Laravel AI SDK (Laravel AI SDK)
소개 (Introduction)
Laravel AI SDK는 OpenAI, Anthropic, Gemini 등 다양한 AI provider와 상호작용할 수 있는 통합된 API를 제공합니다. AI SDK를 사용하면 도구와 구조화된 출력을 갖춘 지능형 에이전트를 만들고, 이미지를 생성하고, 오디오를 합성하거나 음성을 텍스트로 변환하고, 벡터 임베딩을 생성하는 등 다양한 작업을 일관된 Laravel 친화적 인터페이스로 처리할 수 있습니다.
설치 (Installation)
Composer를 통해 Laravel AI SDK를 설치할 수 있습니다:
composer require laravel/ai
다음으로 vendor:publish Artisan 명령어를 사용하여 AI SDK의 설정 및 마이그레이션 파일을 게시해야 합니다:
php artisan vendor:publish --provider="Laravel\Ai\AiServiceProvider"
마지막으로 애플리케이션의 데이터베이스 마이그레이션을 실행합니다. 이 작업을 통해 AI SDK가 대화 기록을 저장하는 데 사용하는 agent_conversations 및 agent_conversation_messages 테이블이 생성됩니다:
php artisan migrate
설정
AI provider의 인증 정보는 애플리케이션의 config/ai.php 설정 파일이나 .env 파일의 환경 변수로 지정할 수 있습니다:
ANTHROPIC_API_KEY=
COHERE_API_KEY=
ELEVENLABS_API_KEY=
GEMINI_API_KEY=
MISTRAL_API_KEY=
OLLAMA_API_KEY=
OPENAI_API_KEY=
JINA_API_KEY=
VOYAGEAI_API_KEY=
XAI_API_KEY=
텍스트, 이미지, 오디오, 음성 전사, 임베딩에 사용할 기본 모델도 config/ai.php 설정 파일에서 지정할 수 있습니다.
사용자 지정 Base URL
기본적으로 Laravel AI SDK는 각 provider의 공개 API 엔드포인트에 직접 연결합니다. 하지만 프록시 서비스를 통해 요청을 우회해야 하는 경우도 있습니다. 예를 들어 API 키 관리를 중앙화하거나, rate limiting을 적용하거나, 사내 게이트웨이를 통해 트래픽을 라우팅해야 할 수 있습니다.
이런 경우 provider 설정에 url 파라미터를 추가해 사용자 지정 Base URL을 설정할 수 있습니다:
'providers' => [
'openai' => [
'driver' => 'openai',
'key' => env('OPENAI_API_KEY'),
'url' => env('OPENAI_BASE_URL'),
],
'anthropic' => [
'driver' => 'anthropic',
'key' => env('ANTHROPIC_API_KEY'),
'url' => env('ANTHROPIC_BASE_URL'),
],
],
이 기능은 LiteLLM이나 Azure OpenAI Gateway 같은 프록시 서비스를 통해 요청을 라우팅하거나 대체 엔드포인트를 사용할 때 유용합니다.
다음 provider는 사용자 지정 Base URL을 지원합니다: OpenAI, Anthropic, Gemini, Groq, Cohere, DeepSeek, xAI, OpenRouter.
Provider 지원
AI SDK는 기능별로 여러 provider를 지원합니다. 다음 표는 각 기능에서 사용할 수 있는 provider를 정리한 것입니다:
| 기능 | Provider |
|---|---|
| Text | OpenAI, Anthropic, Gemini, Azure, Groq, xAI, DeepSeek, Mistral, Ollama |
| Images | OpenAI, Gemini, xAI |
| TTS | OpenAI, ElevenLabs |
| STT | OpenAI, ElevenLabs, Mistral |
| Embeddings | OpenAI, Gemini, Azure, Cohere, Mistral, Jina, VoyageAI |
| Reranking | Cohere, Jina |
| Files | OpenAI, Anthropic, Gemini |
코드에서는 문자열 대신 Laravel\Ai\Enums\Lab enum으로 provider를 참조할 수 있습니다:
use Laravel\Ai\Enums\Lab;
Lab::Anthropic;
Lab::OpenAI;
Lab::Gemini;
// ...
에이전트 (Agents)
에이전트는 Laravel AI SDK에서 AI provider와 상호작용할 때 사용하는 기본 구성 요소입니다. 각 에이전트는 대형 언어 모델과 상호작용하는 데 필요한 지침, 대화 컨텍스트, 도구, 출력 스키마를 캡슐화한 전용 PHP 클래스입니다. 에이전트는 영업 코치, 문서 분석기, 고객 지원 봇처럼 특정 역할에 맞춘 어시스턴트라고 생각하면 됩니다. 한 번 정의해 두면 애플리케이션 전반에서 필요할 때마다 프롬프트할 수 있습니다.
make:agent Artisan 명령어로 에이전트를 생성할 수 있습니다:
php artisan make:agent SalesCoach
php artisan make:agent SalesCoach --structured
생성된 에이전트 클래스에서는 시스템 프롬프트 또는 지침, 메시지 컨텍스트, 사용 가능한 도구, 출력 스키마(해당하는 경우)를 정의할 수 있습니다:
<?php
namespace App\Ai\Agents;
use App\Ai\Tools\RetrievePreviousTranscripts;
use App\Models\History;
use App\Models\User;
use Illuminate\Contracts\JsonSchema\JsonSchema;
use Laravel\Ai\Contracts\Agent;
use Laravel\Ai\Contracts\Conversational;
use Laravel\Ai\Contracts\HasStructuredOutput;
use Laravel\Ai\Contracts\HasTools;
use Laravel\Ai\Messages\Message;
use Laravel\Ai\Promptable;
use Stringable;
class SalesCoach implements Agent, Conversational, HasTools, HasStructuredOutput
{
use Promptable;
public function __construct(public User $user) {}
/**
* Get the instructions that the agent should follow.
*/
public function instructions(): Stringable|string
{
return 'You are a sales coach, analyzing transcripts and providing feedback and an overall sales strength score.';
}
/**
* Get the list of messages comprising the conversation so far.
*/
public function messages(): iterable
{
return History::where('user_id', $this->user->id)
->latest()
->limit(50)
->get()
->reverse()
->map(function ($message) {
return new Message($message->role, $message->content);
})->all();
}
/**
* Get the tools available to the agent.
*
* @return Tool[]
*/
public function tools(): iterable
{
return [
new RetrievePreviousTranscripts,
];
}
/**
* Get the agent's structured output schema definition.
*/
public function schema(JsonSchema $schema): array
{
return [
'feedback' => $schema->string()->required(),
'score' => $schema->integer()->min(1)->max(10)->required(),
];
}
}
프롬프트 작성
에이전트에 프롬프트를 보내려면 먼저 make 메서드 또는 일반적인 인스턴스화로 인스턴스를 만든 뒤 prompt를 호출합니다:
$response = (new SalesCoach)
->prompt('Analyze this sales transcript...');
$response = SalesCoach::make()
->prompt('Analyze this sales transcript...');
return (string) $response;
make 메서드는 컨테이너에서 에이전트를 resolve하므로 자동 의존성 주입을 활용할 수 있습니다. 에이전트 생성자에 인수를 전달하는 것도 가능합니다:
$agent = SalesCoach::make(user: $user);
prompt 메서드에 추가 인수를 전달하면 프롬프트를 보낼 때 기본 provider, 모델, HTTP 타임아웃을 재정의할 수 있습니다:
$response = (new SalesCoach)->prompt(
'Analyze this sales transcript...',
provider: Lab::Anthropic,
model: 'claude-haiku-4-5-20251001',
timeout: 120,
);
대화 컨텍스트
에이전트가 Conversational 인터페이스를 구현하는 경우 messages 메서드로 이전 대화 컨텍스트를 반환할 수 있습니다:
use App\Models\History;
use Laravel\Ai\Messages\Message;
/**
* Get the list of messages comprising the conversation so far.
*/
public function messages(): iterable
{
return History::where('user_id', $this->user->id)
->latest()
->limit(50)
->get()
->reverse()
->map(function ($message) {
return new Message($message->role, $message->content);
})->all();
}
대화 기억하기
Note:
RemembersConversations트레이트를 사용하기 전에vendor:publishArtisan 명령어를 사용하여 AI SDK 마이그레이션을 게시하고 실행해야 합니다. 이 마이그레이션은 대화를 저장하는 데 필요한 데이터베이스 테이블을 생성합니다.
Laravel이 에이전트의 대화 기록을 자동으로 저장하고 다시 불러오게 하려면 RemembersConversations 트레이트를 사용할 수 있습니다. 이 트레이트는 Conversational 인터페이스를 수동으로 구현하지 않아도 대화 메시지를 데이터베이스에 영속적으로 저장할 수 있는 간단한 방법을 제공합니다:
<?php
namespace App\Ai\Agents;
use Laravel\Ai\Concerns\RemembersConversations;
use Laravel\Ai\Contracts\Agent;
use Laravel\Ai\Contracts\Conversational;
use Laravel\Ai\Promptable;
class SalesCoach implements Agent, Conversational
{
use Promptable, RemembersConversations;
/**
* Get the instructions that the agent should follow.
*/
public function instructions(): string
{
return 'You are a sales coach...';
}
}
사용자의 새 대화를 시작하려면 프롬프트 전에 forUser 메서드를 호출합니다:
$response = (new SalesCoach)->forUser($user)->prompt('Hello!');
$conversationId = $response->conversationId;
대화 ID는 응답에서 반환되며 나중에 참조할 수 있도록 저장할 수 있습니다. 또는 agent_conversations 테이블에서 직접 사용자의 모든 대화를 조회할 수 있습니다.
기존 대화를 계속하려면 continue 메서드를 사용합니다:
$response = (new SalesCoach)
->continue($conversationId, as: $user)
->prompt('Tell me more about that.');
RemembersConversations 트레이트를 사용할 때 이전 메시지는 프롬프트 시 자동으로 로드되어 대화 컨텍스트에 포함됩니다. 새 메시지(사용자 및 어시스턴트 모두)는 각 상호작용 후 자동으로 저장됩니다.
구조화된 출력
에이전트가 구조화된 출력을 반환하게 하려면 HasStructuredOutput 인터페이스를 구현하세요. 이 인터페이스를 사용하려면 에이전트에 schema 메서드를 정의해야 합니다:
<?php
namespace App\Ai\Agents;
use Illuminate\Contracts\JsonSchema\JsonSchema;
use Laravel\Ai\Contracts\Agent;
use Laravel\Ai\Contracts\HasStructuredOutput;
use Laravel\Ai\Promptable;
class SalesCoach implements Agent, HasStructuredOutput
{
use Promptable;
// ...
/**
* Get the agent's structured output schema definition.
*/
public function schema(JsonSchema $schema): array
{
return [
'score' => $schema->integer()->required(),
];
}
}
구조화된 출력을 반환하는 에이전트에 프롬프트할 때는 반환된 StructuredAgentResponse를 배열처럼 다룰 수 있습니다:
$response = (new SalesCoach)->prompt('Analyze this sales transcript...');
return $response['score'];
첨부 파일
프롬프트 시 모델이 이미지와 문서를 검토할 수 있도록 첨부 파일을 함께 전달할 수 있습니다:
use App\Ai\Agents\SalesCoach;
use Laravel\Ai\Files;
$response = (new SalesCoach)->prompt(
'Analyze the attached sales transcript...',
attachments: [
Files\Document::fromStorage('transcript.pdf') // Attach a document from a filesystem disk...
Files\Document::fromPath('/home/laravel/transcript.md') // Attach a document from a local path...
$request->file('transcript'), // Attach an uploaded file...
]
);
마찬가지로 Laravel\Ai\Files\Image 클래스를 사용하여 프롬프트에 이미지를 첨부할 수 있습니다:
use App\Ai\Agents\ImageAnalyzer;
use Laravel\Ai\Files;
$response = (new ImageAnalyzer)->prompt(
'What is in this image?',
attachments: [
Files\Image::fromStorage('photo.jpg') // Attach an image from a filesystem disk...
Files\Image::fromPath('/home/laravel/photo.jpg') // Attach an image from a local path...
$request->file('photo'), // Attach an uploaded file...
]
);
스트리밍
Agent의 응답을 스트리밍하려면 stream 메서드를 호출합니다. 반환된 StreamableAgentResponse는 라우트에서 반환하여 자동으로 스트리밍 응답(SSE)을 클라이언트에 보낼 수 있습니다:
use App\Ai\Agents\SalesCoach;
Route::get('/coach', function () {
return (new SalesCoach)->stream('Analyze this sales transcript...');
});
then 메서드를 사용하면 전체 응답이 클라이언트에 스트리밍된 후 실행될 클로저를 제공할 수 있습니다:
use App\Ai\Agents\SalesCoach;
use Laravel\Ai\Responses\StreamedAgentResponse;
Route::get('/coach', function () {
return (new SalesCoach)
->stream('Analyze this sales transcript...')
->then(function (StreamedAgentResponse $response) {
// $response->text, $response->events, $response->usage...
});
});
또는 스트리밍된 이벤트를 수동으로 반복 처리할 수 있습니다:
$stream = (new SalesCoach)->stream('Analyze this sales transcript...');
foreach ($stream as $event) {
// ...
}
Vercel AI SDK 프로토콜을 사용한 스트리밍
스트리밍 가능한 응답에서 usingVercelDataProtocol 메서드를 호출하면 Vercel AI SDK 스트림 프로토콜을 사용하여 이벤트를 스트리밍할 수 있습니다:
use App\Ai\Agents\SalesCoach;
Route::get('/coach', function () {
return (new SalesCoach)
->stream('Analyze this sales transcript...')
->usingVercelDataProtocol();
});
브로드캐스팅
스트리밍된 이벤트를 여러 방법으로 브로드캐스트할 수 있습니다. 먼저, 스트리밍된 이벤트에서 broadcast 또는 broadcastNow 메서드를 호출할 수 있습니다:
use App\Ai\Agents\SalesCoach;
use Illuminate\Broadcasting\Channel;
$stream = (new SalesCoach)->stream('Analyze this sales transcript...');
foreach ($stream as $event) {
$event->broadcast(new Channel('channel-name'));
}
또는 에이전트의 broadcastOnQueue 메서드를 호출해 에이전트 작업을 큐에 넣고, 스트리밍 이벤트를 사용할 수 있게 되면 브로드캐스트할 수 있습니다:
(new SalesCoach)->broadcastOnQueue(
'Analyze this sales transcript...'
new Channel('channel-name'),
);
큐 처리
Agent의 queue 메서드를 사용하면 agent에 프롬프트를 보내되, 백그라운드에서 응답을 처리하여 애플리케이션의 반응성을 유지할 수 있습니다. then 및 catch 메서드를 사용하여 응답이 사용 가능해지거나 예외가 발생했을 때 호출될 클로저를 등록할 수 있습니다:
use Illuminate\Http\Request;
use Laravel\Ai\Responses\AgentResponse;
use Throwable;
Route::post('/coach', function (Request $request) {
return (new SalesCoach)
->queue($request->input('transcript'))
->then(function (AgentResponse $response) {
// ...
})
->catch(function (Throwable $e) {
// ...
});
return back();
});
도구
도구를 사용하면 에이전트가 프롬프트에 응답할 때 활용할 수 있는 추가 기능을 제공할 수 있습니다. 도구는 make:tool Artisan 명령어로 생성할 수 있습니다:
php artisan make:tool RandomNumberGenerator
생성된 도구는 애플리케이션의 app/Ai/Tools 디렉터리에 생성됩니다. 각 도구에는 에이전트가 도구를 사용해야 할 때 호출되는 handle 메서드가 포함되어 있습니다:
<?php
namespace App\Ai\Tools;
use Illuminate\Contracts\JsonSchema\JsonSchema;
use Laravel\Ai\Contracts\Tool;
use Laravel\Ai\Tools\Request;
use Stringable;
class RandomNumberGenerator implements Tool
{
/**
* Get the description of the tool's purpose.
*/
public function description(): Stringable|string
{
return 'This tool may be used to generate cryptographically secure random numbers.';
}
/**
* Execute the tool.
*/
public function handle(Request $request): Stringable|string
{
return (string) random_int($request['min'], $request['max']);
}
/**
* Get the tool's schema definition.
*/
public function schema(JsonSchema $schema): array
{
return [
'min' => $schema->integer()->min(0)->required(),
'max' => $schema->integer()->required(),
];
}
}
도구를 정의한 뒤에는 에이전트의 tools 메서드에서 반환할 수 있습니다:
use App\Ai\Tools\RandomNumberGenerator;
/**
* Get the tools available to the agent.
*
* @return Tool[]
*/
public function tools(): iterable
{
return [
new RandomNumberGenerator,
];
}
유사도 검색
SimilaritySearch 도구를 사용하면 에이전트가 데이터베이스에 저장된 벡터 임베딩으로 주어진 쿼리와 유사한 문서를 검색할 수 있습니다. 애플리케이션 데이터를 검색할 수 있는 권한을 에이전트에 부여하고 싶을 때 검색 증강 생성(RAG)에 유용합니다.
유사도 검색 도구를 만드는 가장 간단한 방법은 벡터 임베딩이 있는 Eloquent 모델과 함께 usingModel 메서드를 사용하는 것입니다:
use App\Models\Document;
use Laravel\Ai\Tools\SimilaritySearch;
public function tools(): iterable
{
return [
SimilaritySearch::usingModel(Document::class, 'embedding'),
];
}
첫 번째 인수는 Eloquent 모델 클래스이고, 두 번째 인수는 벡터 임베딩이 포함된 컬럼입니다.
0.0에서 1.0 사이의 최소 유사도 임계값과 쿼리를 사용자 지정하기 위한 클로저를 제공할 수도 있습니다:
SimilaritySearch::usingModel(
model: Document::class,
column: 'embedding',
minSimilarity: 0.7,
limit: 10,
query: fn ($query) => $query->where('published', true),
),
더 세밀한 제어가 필요하다면 검색 결과를 반환하는 커스텀 클로저로 유사도 검색 도구를 생성할 수 있습니다:
use App\Models\Document;
use Laravel\Ai\Tools\SimilaritySearch;
public function tools(): iterable
{
return [
new SimilaritySearch(using: function (string $query) {
return Document::query()
->where('user_id', $this->user->id)
->whereVectorSimilarTo('embedding', $query)
->limit(10)
->get();
}),
];
}
withDescription 메서드를 사용하여 도구의 설명을 사용자 지정할 수 있습니다:
SimilaritySearch::usingModel(Document::class, 'embedding')
->withDescription('Search the knowledge base for relevant articles.'),
Provider 도구
Provider 도구는 웹 검색, URL 가져오기, 파일 검색 같은 기능을 제공하도록 AI provider가 네이티브로 구현한 특수 도구입니다. 일반 도구와 달리 provider 도구는 애플리케이션이 아니라 provider 자체에서 실행됩니다.
Provider 도구는 에이전트의 tools 메서드에서 반환할 수 있습니다.
웹 검색
WebSearch provider 도구를 사용하면 에이전트가 실시간 정보를 얻기 위해 웹을 검색할 수 있습니다. 최신 이벤트, 최근 데이터, 또는 모델의 학습 기준일 이후 바뀌었을 수 있는 주제에 대한 질문에 답할 때 유용합니다.
지원 Provider: Anthropic, OpenAI, Gemini
use Laravel\Ai\Providers\Tools\WebSearch;
public function tools(): iterable
{
return [
new WebSearch,
];
}
웹 검색 도구를 설정하여 검색 횟수를 제한하거나 결과를 특정 도메인으로 제한할 수 있습니다:
(new WebSearch)->max(5)->allow(['laravel.com', 'php.net']),
사용자 위치를 기반으로 검색 결과를 세분화하려면 location 메서드를 사용합니다:
(new WebSearch)->location(
city: 'New York',
region: 'NY',
country: 'US'
);
웹 페이지 가져오기
WebFetch provider 도구를 사용하면 에이전트가 웹 페이지 내용을 가져와 읽을 수 있습니다. 특정 URL을 분석하거나 알려진 웹 페이지에서 자세한 정보를 검색해야 할 때 유용합니다.
지원 Provider: Anthropic, Gemini
use Laravel\Ai\Providers\Tools\WebFetch;
public function tools(): iterable
{
return [
new WebFetch,
];
}
웹 페이지 가져오기 도구를 설정하여 가져오기 횟수를 제한하거나 특정 도메인으로 제한할 수 있습니다:
(new WebFetch)->max(3)->allow(['docs.laravel.com']),
파일 검색
FileSearch provider 도구를 사용하면 에이전트가 벡터 스토어에 저장된 파일을 검색할 수 있습니다. 업로드된 문서에서 관련 정보를 찾을 수 있게 해 주므로 검색 증강 생성(RAG)에 유용합니다.
지원 Provider: OpenAI, Gemini
use Laravel\Ai\Providers\Tools\FileSearch;
public function tools(): iterable
{
return [
new FileSearch(stores: ['store_id']),
];
}
여러 벡터 스토어를 검색하기 위해 여러 벡터 스토어 ID를 제공할 수 있습니다:
new FileSearch(stores: ['store_1', 'store_2']);
파일에 메타데이터가 있는 경우 where 인수를 제공하여 검색 결과를 필터링할 수 있습니다. 간단한 동등 필터의 경우 배열을 전달합니다:
new FileSearch(stores: ['store_id'], where: [
'author' => 'Taylor Otwell',
'year' => 2026,
]);
더 복잡한 필터의 경우 FileSearchQuery 인스턴스를 받는 클로저를 전달할 수 있습니다:
use Laravel\Ai\Providers\Tools\FileSearchQuery;
new FileSearch(stores: ['store_id'], where: fn (FileSearchQuery $query) =>
$query->where('author', 'Taylor Otwell')
->whereNot('status', 'draft')
->whereIn('category', ['news', 'updates'])
);
미들웨어
에이전트는 미들웨어를 지원하므로 프롬프트가 provider로 전송되기 전에 가로채거나 수정할 수 있습니다. 미들웨어는 make:agent-middleware Artisan 명령어로 생성할 수 있습니다:
php artisan make:agent-middleware LogPrompts
생성된 미들웨어는 애플리케이션의 app/Ai/Middleware 디렉터리에 생성됩니다. 에이전트에 미들웨어를 추가하려면 HasMiddleware 인터페이스를 구현하고, 미들웨어 클래스 배열을 반환하는 middleware 메서드를 정의합니다:
<?php
namespace App\Ai\Agents;
use App\Ai\Middleware\LogPrompts;
use Laravel\Ai\Contracts\Agent;
use Laravel\Ai\Contracts\HasMiddleware;
use Laravel\Ai\Promptable;
class SalesCoach implements Agent, HasMiddleware
{
use Promptable;
// ...
/**
* Get the agent's middleware.
*/
public function middleware(): array
{
return [
new LogPrompts,
];
}
}
각 미들웨어 클래스는 AgentPrompt와 프롬프트를 다음 미들웨어로 전달하는 Closure를 받는 handle 메서드를 정의해야 합니다:
<?php
namespace App\Ai\Middleware;
use Closure;
use Laravel\Ai\Prompts\AgentPrompt;
class LogPrompts
{
/**
* Handle the incoming prompt.
*/
public function handle(AgentPrompt $prompt, Closure $next)
{
Log::info('Prompting agent', ['prompt' => $prompt->prompt]);
return $next($prompt);
}
}
응답의 then 메서드를 사용하면 에이전트가 처리를 마친 뒤 코드를 실행할 수 있습니다. 이 방식은 동기 응답과 스트리밍 응답 모두에서 동작합니다:
public function handle(AgentPrompt $prompt, Closure $next)
{
return $next($prompt)->then(function (AgentResponse $response) {
Log::info('Agent responded', ['text' => $response->text]);
});
}
익명 에이전트
때로는 전용 에이전트 클래스를 만들지 않고 빠르게 모델과 상호작용하고 싶을 수 있습니다. agent 함수를 사용하면 즉석에서 익명 에이전트를 만들 수 있습니다:
use function Laravel\Ai\{agent};
$response = agent(
instructions: 'You are an expert at software development.',
messages: [],
tools: [],
)->prompt('Tell me about Laravel')
익명 에이전트도 구조화된 출력을 생성할 수 있습니다:
use Illuminate\Contracts\JsonSchema\JsonSchema;
use function Laravel\Ai\{agent};
$response = agent(
schema: fn (JsonSchema $schema) => [
'number' => $schema->integer()->required(),
],
)->prompt('Generate a random number less than 100')
에이전트 설정
PHP 어트리뷰트를 사용해 에이전트의 텍스트 생성 옵션을 설정할 수 있습니다. 사용할 수 있는 어트리뷰트는 다음과 같습니다:
MaxSteps: Agent가 도구를 사용할 때 수행할 수 있는 최대 단계 수.MaxTokens: 모델이 생성할 수 있는 최대 토큰 수.Model: Agent가 사용할 모델.Provider: Agent에 사용할 AI provider (또는 failover용 복수 provider).Temperature: 생성에 사용할 샘플링 온도 (0.0에서 1.0).Timeout: 에이전트 요청의 HTTP 타임아웃 (초 단위, 기본값: 60).UseCheapestModel: 비용 최적화를 위해 provider의 가장 저렴한 텍스트 모델 사용.UseSmartestModel: 복잡한 작업을 위해 provider의 가장 강력한 텍스트 모델 사용.
<?php
namespace App\Ai\Agents;
use Laravel\Ai\Attributes\MaxSteps;
use Laravel\Ai\Attributes\MaxTokens;
use Laravel\Ai\Attributes\Model;
use Laravel\Ai\Attributes\Provider;
use Laravel\Ai\Attributes\Temperature;
use Laravel\Ai\Attributes\Timeout;
use Laravel\Ai\Contracts\Agent;
use Laravel\Ai\Enums\Lab;
use Laravel\Ai\Promptable;
#[Provider(Lab::Anthropic)]
#[Model('claude-haiku-4-5-20251001')]
#[MaxSteps(10)]
#[MaxTokens(4096)]
#[Temperature(0.7)]
#[Timeout(120)]
class SalesCoach implements Agent
{
use Promptable;
// ...
}
UseCheapestModel과 UseSmartestModel 어트리뷰트를 사용하면 모델 이름을 지정하지 않고도 주어진 provider에서 가장 비용 효율적이거나 가장 강력한 모델을 자동으로 선택할 수 있습니다. 이는 다양한 provider에서 비용이나 성능을 최적화하고자 할 때 유용합니다:
use Laravel\Ai\Attributes\UseCheapestModel;
use Laravel\Ai\Attributes\UseSmartestModel;
use Laravel\Ai\Contracts\Agent;
use Laravel\Ai\Promptable;
#[UseCheapestModel]
class SimpleSummarizer implements Agent
{
use Promptable;
// Will use the cheapest model (e.g., Haiku)...
}
#[UseSmartestModel]
class ComplexReasoner implements Agent
{
use Promptable;
// Will use the most capable model (e.g., Opus)...
}
이미지 (Images)
Laravel\Ai\Image 클래스를 사용하여 openai, gemini, 또는 xai provider를 통해 이미지를 생성할 수 있습니다:
use Laravel\Ai\Image;
$image = Image::of('A donut sitting on the kitchen counter')->generate();
$rawContent = (string) $image;
square, portrait, landscape 메서드를 사용하여 이미지의 종횡비를 제어할 수 있으며, quality 메서드를 사용하여 최종 이미지 품질(high, medium, low)을 가이드할 수 있습니다. timeout 메서드를 사용하여 HTTP 타임아웃을 초 단위로 지정할 수 있습니다:
use Laravel\Ai\Image;
$image = Image::of('A donut sitting on the kitchen counter')
->quality('high')
->landscape()
->timeout(120)
->generate();
attachments 메서드를 사용하여 참조 이미지를 첨부할 수 있습니다:
use Laravel\Ai\Files;
use Laravel\Ai\Image;
$image = Image::of('Update this photo of me to be in the style of an impressionist painting.')
->attachments([
Files\Image::fromStorage('photo.jpg'),
// Files\Image::fromPath('/home/laravel/photo.jpg'),
// Files\Image::fromUrl('https://example.com/photo.jpg'),
// $request->file('photo'),
])
->landscape()
->generate();
생성된 이미지는 애플리케이션의 config/filesystems.php 설정 파일에 구성된 기본 디스크에 쉽게 저장할 수 있습니다:
$image = Image::of('A donut sitting on the kitchen counter');
$path = $image->store();
$path = $image->storeAs('image.jpg');
$path = $image->storePublicly();
$path = $image->storePubliclyAs('image.jpg');
이미지 생성은 큐에 넣을 수도 있습니다:
use Laravel\Ai\Image;
use Laravel\Ai\Responses\ImageResponse;
Image::of('A donut sitting on the kitchen counter')
->portrait()
->queue()
->then(function (ImageResponse $image) {
$path = $image->store();
// ...
});
오디오 (Audio)
Laravel\Ai\Audio 클래스를 사용하여 주어진 텍스트에서 오디오를 생성할 수 있습니다:
use Laravel\Ai\Audio;
$audio = Audio::of('I love coding with Laravel.')->generate();
$rawContent = (string) $audio;
male, female, voice 메서드를 사용하여 생성된 오디오의 음성을 결정할 수 있습니다:
$audio = Audio::of('I love coding with Laravel.')
->female()
->generate();
$audio = Audio::of('I love coding with Laravel.')
->voice('voice-id-or-name')
->generate();
마찬가지로, instructions 메서드를 사용하여 생성된 오디오가 어떻게 들려야 하는지 모델에 동적으로 가이드할 수 있습니다:
$audio = Audio::of('I love coding with Laravel.')
->female()
->instructions('Said like a pirate')
->generate();
생성된 오디오는 애플리케이션의 config/filesystems.php 설정 파일에 구성된 기본 디스크에 쉽게 저장할 수 있습니다:
$audio = Audio::of('I love coding with Laravel.')->generate();
$path = $audio->store();
$path = $audio->storeAs('audio.mp3');
$path = $audio->storePublicly();
$path = $audio->storePubliclyAs('audio.mp3');
오디오 생성은 큐에 넣을 수도 있습니다:
use Laravel\Ai\Audio;
use Laravel\Ai\Responses\AudioResponse;
Audio::of('I love coding with Laravel.')
->queue()
->then(function (AudioResponse $audio) {
$path = $audio->store();
// ...
});
음성 전사 (Transcriptions)
Laravel\Ai\Transcription 클래스를 사용해 주어진 오디오의 전사본을 생성할 수 있습니다:
use Laravel\Ai\Transcription;
$transcript = Transcription::fromPath('/home/laravel/audio.mp3')->generate();
$transcript = Transcription::fromStorage('audio.mp3')->generate();
$transcript = Transcription::fromUpload($request->file('audio'))->generate();
return (string) $transcript;
diarize 메서드를 사용하면 원문 텍스트 전사본과 함께 화자별로 구분된 전사본도 응답에 포함되도록 할 수 있습니다:
$transcript = Transcription::fromStorage('audio.mp3')
->diarize()
->generate();
음성 전사 생성은 큐에 넣을 수도 있습니다:
use Laravel\Ai\Transcription;
use Laravel\Ai\Responses\TranscriptionResponse;
Transcription::fromStorage('audio.mp3')
->queue()
->then(function (TranscriptionResponse $transcript) {
// ...
});
임베딩 (Embeddings)
Laravel의 Stringable 클래스에서 제공하는 toEmbeddings 메서드를 사용하여 주어진 문자열에 대한 벡터 임베딩을 쉽게 생성할 수 있습니다:
use Illuminate\Support\Str;
$embeddings = Str::of('Napa Valley has great wine.')->toEmbeddings();
또는 Embeddings 클래스를 사용하여 여러 입력에 대한 임베딩을 한 번에 생성할 수 있습니다:
use Laravel\Ai\Embeddings;
$response = Embeddings::for([
'Napa Valley has great wine.',
'Laravel is a PHP framework.',
])->generate();
$response->embeddings; // [[0.123, 0.456, ...], [0.789, 0.012, ...]]
임베딩의 차원과 provider를 지정할 수 있습니다:
$response = Embeddings::for(['Napa Valley has great wine.'])
->dimensions(1536)
->generate(Lab::OpenAI, 'text-embedding-3-small');
임베딩 검색
임베딩을 생성한 후에는 일반적으로 나중에 조회할 수 있도록 데이터베이스의 vector 컬럼에 저장합니다. Laravel은 pgvector 확장을 통해 PostgreSQL에서 벡터 컬럼에 대한 네이티브 지원을 제공합니다. 시작하려면 마이그레이션에서 차원 수를 지정하여 vector 컬럼을 정의합니다:
Schema::ensureVectorExtensionExists();
Schema::create('documents', function (Blueprint $table) {
$table->id();
$table->string('title');
$table->text('content');
$table->vector('embedding', dimensions: 1536);
$table->timestamps();
});
유사도 검색을 빠르게 하기 위해 벡터 인덱스를 추가할 수도 있습니다. 벡터 컬럼에서 index를 호출하면 Laravel이 자동으로 코사인 거리를 사용하는 HNSW 인덱스를 생성합니다:
$table->vector('embedding', dimensions: 1536)->index();
Eloquent 모델에서 벡터 컬럼을 array로 캐스팅해야 합니다:
protected function casts(): array
{
return [
'embedding' => 'array',
];
}
유사한 레코드를 조회하려면 whereVectorSimilarTo 메서드를 사용합니다. 이 메서드는 최소 코사인 유사도(0.0에서 1.0 사이, 1.0은 동일)로 결과를 필터링하고 유사도순으로 결과를 정렬합니다:
use App\Models\Document;
$documents = Document::query()
->whereVectorSimilarTo('embedding', $queryEmbedding, minSimilarity: 0.4)
->limit(10)
->get();
$queryEmbedding은 실수 배열 또는 일반 문자열일 수 있습니다. 문자열이 주어지면 Laravel이 자동으로 임베딩을 생성합니다:
$documents = Document::query()
->whereVectorSimilarTo('embedding', 'best wineries in Napa Valley')
->limit(10)
->get();
더 세밀한 제어가 필요하다면 하위 수준의 whereVectorDistanceLessThan, selectVectorDistance, orderByVectorDistance 메서드를 독립적으로 사용할 수 있습니다:
$documents = Document::query()
->select('*')
->selectVectorDistance('embedding', $queryEmbedding, as: 'distance')
->whereVectorDistanceLessThan('embedding', $queryEmbedding, maxDistance: 0.3)
->orderByVectorDistance('embedding', $queryEmbedding)
->limit(10)
->get();
에이전트에 도구로서 유사도 검색 기능을 제공하고 싶다면 유사도 검색 도구 문서를 참고하세요.
[!NOTE] 벡터 쿼리는 현재
pgvector확장을 사용하는 PostgreSQL 연결에서만 지원됩니다.
임베딩 캐싱
동일한 입력에 대한 중복 API 호출을 피하기 위해 임베딩 생성을 캐시할 수 있습니다. 캐싱을 활성화하려면 ai.caching.embeddings.cache 설정 옵션을 true로 설정합니다:
'caching' => [
'embeddings' => [
'cache' => true,
'store' => env('CACHE_STORE', 'database'),
// ...
],
],
캐싱이 활성화되면 임베딩은 30일 동안 캐시됩니다. 캐시 키는 provider, 모델, 차원, 입력 내용을 기반으로 하므로 동일한 요청은 캐시된 결과를 반환하고 다른 구성은 새로운 임베딩을 생성합니다.
전역 캐싱이 비활성화되어 있어도 cache 메서드를 사용하여 특정 요청에 대한 캐싱을 활성화할 수 있습니다:
$response = Embeddings::for(['Napa Valley has great wine.'])
->cache()
->generate();
커스텀 캐시 기간을 초 단위로 지정할 수 있습니다:
$response = Embeddings::for(['Napa Valley has great wine.'])
->cache(seconds: 3600) // Cache for 1 hour
->generate();
toEmbeddings Stringable 메서드도 cache 인수를 받습니다:
// Cache with default duration...
$embeddings = Str::of('Napa Valley has great wine.')->toEmbeddings(cache: true);
// Cache for a specific duration...
$embeddings = Str::of('Napa Valley has great wine.')->toEmbeddings(cache: 3600);
리랭킹 (Reranking)
리랭킹을 사용하면 주어진 쿼리와의 관련성을 기준으로 문서 목록의 순서를 다시 정렬할 수 있습니다. 검색 결과를 의미적으로 더 정확하게 개선할 때 유용합니다:
Laravel\Ai\Reranking 클래스를 사용해 문서를 리랭킹할 수 있습니다:
use Laravel\Ai\Reranking;
$response = Reranking::of([
'Django is a Python web framework.',
'Laravel is a PHP web application framework.',
'React is a JavaScript library for building user interfaces.',
])->rerank('PHP frameworks');
// Access the top result...
$response->first()->document; // "Laravel is a PHP web application framework."
$response->first()->score; // 0.95
$response->first()->index; // 1 (original position)
limit 메서드를 사용하여 반환되는 결과 수를 제한할 수 있습니다:
$response = Reranking::of($documents)
->limit(5)
->rerank('search query');
컬렉션 리랭킹
편의를 위해 rerank 매크로를 사용해 Laravel 컬렉션을 리랭킹할 수 있습니다. 첫 번째 인수는 리랭킹에 사용할 필드를 지정하고, 두 번째 인수는 쿼리입니다:
// Rerank by a single field...
$posts = Post::all()
->rerank('body', 'Laravel tutorials');
// Rerank by multiple fields (sent as JSON)...
$reranked = $posts->rerank(['title', 'body'], 'Laravel tutorials');
// Rerank using a closure to build the document...
$reranked = $posts->rerank(
fn ($post) => $post->title.': '.$post->body,
'Laravel tutorials'
);
결과 수를 제한하고 provider를 지정할 수도 있습니다:
$reranked = $posts->rerank(
by: 'content',
query: 'Laravel tutorials',
limit: 10,
provider: Lab::Cohere
);
파일 (Files)
Laravel\Ai\Files 클래스 또는 개별 파일 클래스를 사용하여 AI provider에 파일을 저장하고 나중에 대화에서 사용할 수 있습니다. 이는 여러 번 참조하고 싶은 대용량 문서나 파일을 다시 업로드하지 않고 사용할 때 유용합니다:
use Laravel\Ai\Files\Document;
use Laravel\Ai\Files\Image;
// Store a file from a local path...
$response = Document::fromPath('/home/laravel/document.pdf')->put();
$response = Image::fromPath('/home/laravel/photo.jpg')->put();
// Store a file that is stored on a filesystem disk...
$response = Document::fromStorage('document.pdf', disk: 'local')->put();
$response = Image::fromStorage('photo.jpg', disk: 'local')->put();
// Store a file that is stored on a remote URL...
$response = Document::fromUrl('https://example.com/document.pdf')->put();
$response = Image::fromUrl('https://example.com/photo.jpg')->put();
return $response->id;
원시 콘텐츠나 업로드된 파일을 저장할 수도 있습니다:
use Laravel\Ai\Files;
use Laravel\Ai\Files\Document;
// Store raw content...
$stored = Document::fromString('Hello, World!', 'text/plain')->put();
// Store an uploaded file...
$stored = Document::fromUpload($request->file('document'))->put();
파일이 저장되면 다시 업로드하지 않고도 에이전트를 통한 텍스트 생성에서 해당 파일을 참조할 수 있습니다:
use App\Ai\Agents\SalesCoach;
use Laravel\Ai\Files;
$response = (new SalesCoach)->prompt(
'Analyze the attached sales transcript...'
attachments: [
Files\Document::fromId('file-id') // Attach a stored document...
]
);
이전에 저장된 파일을 조회하려면 파일 인스턴스의 get 메서드를 사용합니다:
use Laravel\Ai\Files\Document;
$file = Document::fromId('file-id')->get();
$file->id;
$file->mimeType();
Provider에서 파일을 삭제하려면 delete 메서드를 사용합니다:
Document::fromId('file-id')->delete();
기본적으로 Files 클래스는 애플리케이션의 config/ai.php 설정 파일에 구성된 기본 AI provider를 사용합니다. 대부분의 작업에서 provider 인수를 사용하여 다른 provider를 지정할 수 있습니다:
$response = Document::fromPath(
'/home/laravel/document.pdf'
)->put(provider: Lab::Anthropic);
저장된 파일을 대화에서 사용하기
Provider에 파일이 저장되면 Document 또는 Image 클래스의 fromId 메서드를 사용해 에이전트 대화에서 참조할 수 있습니다:
use App\Ai\Agents\DocumentAnalyzer;
use Laravel\Ai\Files;
use Laravel\Ai\Files\Document;
$stored = Document::fromPath('/path/to/report.pdf')->put();
$response = (new DocumentAnalyzer)->prompt(
'Summarize this document.',
attachments: [
Document::fromId($stored->id),
],
);
마찬가지로 저장된 이미지는 Image 클래스를 사용하여 참조할 수 있습니다:
use Laravel\Ai\Files;
use Laravel\Ai\Files\Image;
$stored = Image::fromPath('/path/to/photo.jpg')->put();
$response = (new ImageAnalyzer)->prompt(
'What is in this image?',
attachments: [
Image::fromId($stored->id),
],
);
벡터 스토어 (Vector Stores)
벡터 스토어를 사용하면 검색 증강 생성(RAG)에 사용할 수 있는 검색 가능한 파일 컬렉션을 생성할 수 있습니다. Laravel\Ai\Stores 클래스는 벡터 스토어를 생성, 조회, 삭제하는 메서드를 제공합니다:
use Laravel\Ai\Stores;
// Create a new vector store...
$store = Stores::create('Knowledge Base');
// Create a store with additional options...
$store = Stores::create(
name: 'Knowledge Base',
description: 'Documentation and reference materials.',
expiresWhenIdleFor: days(30),
);
return $store->id;
기존 벡터 스토어를 ID로 조회하려면 get 메서드를 사용합니다:
use Laravel\Ai\Stores;
$store = Stores::get('store_id');
$store->id;
$store->name;
$store->fileCounts;
$store->ready;
벡터 스토어를 삭제하려면 Stores 클래스 또는 스토어 인스턴스의 delete 메서드를 사용합니다:
use Laravel\Ai\Stores;
// Delete by ID...
Stores::delete('store_id');
// Or delete via a store instance...
$store = Stores::get('store_id');
$store->delete();
스토어에 파일 추가
벡터 스토어가 있으면 add 메서드를 사용하여 파일을 추가할 수 있습니다. 스토어에 추가된 파일은 파일 검색 provider 도구를 사용한 의미 검색을 위해 자동으로 인덱싱됩니다:
use Laravel\Ai\Files\Document;
use Laravel\Ai\Stores;
$store = Stores::get('store_id');
// Add a file that has already been stored with the provider...
$document = $store->add('file_id');
$document = $store->add(Document::fromId('file_id'));
// Or, store and add a file in one step...
$document = $store->add(Document::fromPath('/path/to/document.pdf'));
$document = $store->add(Document::fromStorage('manual.pdf'));
$document = $store->add($request->file('document'));
$document->id;
$document->fileId;
Note: 일반적으로 이전에 저장된 파일을 벡터 스토어에 추가할 때 반환된 문서 ID는 파일의 이전에 할당된 ID와 일치합니다. 그러나 일부 벡터 스토리지 provider는 새로운 다른 "문서 ID"를 반환할 수 있습니다. 따라서 나중에 참조할 수 있도록 항상 두 ID를 데이터베이스에 저장하는 것이 좋습니다.
스토어에 파일을 추가할 때 메타데이터를 첨부할 수 있습니다. 이 메타데이터는 나중에 파일 검색 provider 도구를 사용할 때 검색 결과를 필터링하는 데 사용할 수 있습니다:
$store->add(Document::fromPath('/path/to/document.pdf'), metadata: [
'author' => 'Taylor Otwell',
'department' => 'Engineering',
'year' => 2026,
]);
스토어에서 파일을 제거하려면 remove 메서드를 사용합니다:
$store->remove('file_id');
벡터 스토어에서 파일을 제거해도 provider의 파일 스토리지에서는 제거되지 않습니다. 벡터 스토어에서 파일을 제거하고 파일 스토리지에서도 영구 삭제하려면 deleteFile 인수를 사용합니다:
$store->remove('file_abc123', deleteFile: true);
페일오버 (Failover)
프롬프트를 보내거나 다른 미디어를 생성할 때 provider/모델 배열을 제공하여 기본 provider에서 서비스 중단이나 속도 제한이 발생하면 자동으로 백업 provider/모델로 전환할 수 있습니다:
use App\Ai\Agents\SalesCoach;
use Laravel\Ai\Image;
$response = (new SalesCoach)->prompt(
'Analyze this sales transcript...',
provider: [Lab::OpenAI, Lab::Anthropic],
);
$image = Image::of('A donut sitting on the kitchen counter')
->generate(provider: [Lab::Gemini, Lab::xAI]);
테스트 (Testing)
에이전트
테스트 중 에이전트의 응답을 페이크하려면 에이전트 클래스에서 fake 메서드를 호출합니다. 필요하다면 응답 배열이나 클로저를 제공할 수 있습니다:
use App\Ai\Agents\SalesCoach;
use Laravel\Ai\Prompts\AgentPrompt;
// Automatically generate a fixed response for every prompt...
SalesCoach::fake();
// Provide a list of prompt responses...
SalesCoach::fake([
'First response',
'Second response',
]);
// Dynamically handle prompt responses based on the incoming prompt...
SalesCoach::fake(function (AgentPrompt $prompt) {
return 'Response for: '.$prompt->prompt;
});
Note: 구조화된 출력을 반환하는 에이전트에서
Agent::fake()를 호출하면 Laravel이 에이전트에 정의된 출력 스키마에 맞는 페이크 데이터를 자동으로 생성합니다.
Agent에 프롬프트를 보낸 후 수신된 프롬프트에 대한 assertion을 만들 수 있습니다:
use Laravel\Ai\Prompts\AgentPrompt;
SalesCoach::assertPrompted('Analyze this...');
SalesCoach::assertPrompted(function (AgentPrompt $prompt) {
return $prompt->contains('Analyze');
});
SalesCoach::assertNotPrompted('Missing prompt');
SalesCoach::assertNeverPrompted();
큐에 넣은 에이전트 호출은 큐 assertion 메서드로 검증할 수 있습니다:
use Laravel\Ai\QueuedAgentPrompt;
SalesCoach::assertQueued('Analyze this...');
SalesCoach::assertQueued(function (QueuedAgentPrompt $prompt) {
return $prompt->contains('Analyze');
});
SalesCoach::assertNotQueued('Missing prompt');
SalesCoach::assertNeverQueued();
모든 에이전트 호출에 대응하는 페이크 응답이 있는지 확인하려면 preventStrayPrompts를 사용할 수 있습니다. 정의된 페이크 응답 없이 에이전트가 호출되면 예외가 발생합니다:
SalesCoach::fake()->preventStrayPrompts();
이미지
Image 클래스에서 fake 메서드를 호출하여 이미지 생성을 페이크할 수 있습니다. 이미지가 페이크되면 기록된 이미지 생성 프롬프트에 대해 다양한 assertion을 수행할 수 있습니다:
use Laravel\Ai\Image;
use Laravel\Ai\Prompts\ImagePrompt;
use Laravel\Ai\Prompts\QueuedImagePrompt;
// Automatically generate a fixed response for every prompt...
Image::fake();
// Provide a list of prompt responses...
Image::fake([
base64_encode($firstImage),
base64_encode($secondImage),
]);
// Dynamically handle prompt responses based on the incoming prompt...
Image::fake(function (ImagePrompt $prompt) {
return base64_encode('...');
});
이미지를 생성한 후 수신된 프롬프트에 대한 assertion을 만들 수 있습니다:
Image::assertGenerated(function (ImagePrompt $prompt) {
return $prompt->contains('sunset') && $prompt->isLandscape();
});
Image::assertNotGenerated('Missing prompt');
Image::assertNothingGenerated();
큐에 넣은 이미지 생성의 경우 큐 assertion 메서드를 사용합니다:
Image::assertQueued(
fn (QueuedImagePrompt $prompt) => $prompt->contains('sunset')
);
Image::assertNotQueued('Missing prompt');
Image::assertNothingQueued();
모든 이미지 생성에 대응하는 페이크 응답이 있는지 확인하려면 preventStrayImages를 사용할 수 있습니다. 정의된 페이크 응답 없이 이미지가 생성되면 예외가 발생합니다:
Image::fake()->preventStrayImages();
오디오
Audio 클래스에서 fake 메서드를 호출하여 오디오 생성을 페이크할 수 있습니다. 오디오가 페이크되면 기록된 오디오 생성 프롬프트에 대해 다양한 assertion을 수행할 수 있습니다:
use Laravel\Ai\Audio;
use Laravel\Ai\Prompts\AudioPrompt;
use Laravel\Ai\Prompts\QueuedAudioPrompt;
// Automatically generate a fixed response for every prompt...
Audio::fake();
// Provide a list of prompt responses...
Audio::fake([
base64_encode($firstAudio),
base64_encode($secondAudio),
]);
// Dynamically handle prompt responses based on the incoming prompt...
Audio::fake(function (AudioPrompt $prompt) {
return base64_encode('...');
});
오디오를 생성한 후 수신된 프롬프트에 대한 assertion을 만들 수 있습니다:
Audio::assertGenerated(function (AudioPrompt $prompt) {
return $prompt->contains('Hello') && $prompt->isFemale();
});
Audio::assertNotGenerated('Missing prompt');
Audio::assertNothingGenerated();
큐에 넣은 오디오 생성의 경우 큐 assertion 메서드를 사용합니다:
Audio::assertQueued(
fn (QueuedAudioPrompt $prompt) => $prompt->contains('Hello')
);
Audio::assertNotQueued('Missing prompt');
Audio::assertNothingQueued();
모든 오디오 생성에 대응하는 페이크 응답이 있는지 확인하려면 preventStrayAudio를 사용할 수 있습니다. 정의된 페이크 응답 없이 오디오가 생성되면 예외가 발생합니다:
Audio::fake()->preventStrayAudio();
음성 전사
Transcription 클래스에서 fake 메서드를 호출해 음성 전사 생성을 페이크할 수 있습니다. 전사 생성이 페이크되면 기록된 전사 프롬프트에 대해 다양한 assertion을 수행할 수 있습니다:
use Laravel\Ai\Transcription;
use Laravel\Ai\Prompts\TranscriptionPrompt;
use Laravel\Ai\Prompts\QueuedTranscriptionPrompt;
// Automatically generate a fixed response for every prompt...
Transcription::fake();
// Provide a list of prompt responses...
Transcription::fake([
'First transcription text.',
'Second transcription text.',
]);
// Dynamically handle prompt responses based on the incoming prompt...
Transcription::fake(function (TranscriptionPrompt $prompt) {
return 'Transcribed text...';
});
전사를 생성한 뒤 수신된 프롬프트에 대한 assertion을 만들 수 있습니다:
Transcription::assertGenerated(function (TranscriptionPrompt $prompt) {
return $prompt->language === 'en' && $prompt->isDiarized();
});
Transcription::assertNotGenerated(
fn (TranscriptionPrompt $prompt) => $prompt->language === 'fr'
);
Transcription::assertNothingGenerated();
큐에 넣은 음성 전사 생성은 큐 assertion 메서드로 검증할 수 있습니다:
Transcription::assertQueued(
fn (QueuedTranscriptionPrompt $prompt) => $prompt->isDiarized()
);
Transcription::assertNotQueued(
fn (QueuedTranscriptionPrompt $prompt) => $prompt->language === 'fr'
);
Transcription::assertNothingQueued();
모든 음성 전사 생성에 대응하는 페이크 응답이 있는지 확인하려면 preventStrayTranscriptions를 사용할 수 있습니다. 정의된 페이크 응답 없이 전사가 생성되면 예외가 발생합니다:
Transcription::fake()->preventStrayTranscriptions();
임베딩
Embeddings 클래스에서 fake 메서드를 호출하여 임베딩 생성을 페이크할 수 있습니다. 임베딩이 페이크되면 기록된 임베딩 생성 프롬프트에 대해 다양한 assertion을 수행할 수 있습니다:
use Laravel\Ai\Embeddings;
use Laravel\Ai\Prompts\EmbeddingsPrompt;
use Laravel\Ai\Prompts\QueuedEmbeddingsPrompt;
// Automatically generate fake embeddings of the proper dimensions for every prompt...
Embeddings::fake();
// Provide a list of prompt responses...
Embeddings::fake([
[$firstEmbeddingVector],
[$secondEmbeddingVector],
]);
// Dynamically handle prompt responses based on the incoming prompt...
Embeddings::fake(function (EmbeddingsPrompt $prompt) {
return array_map(
fn () => Embeddings::fakeEmbedding($prompt->dimensions),
$prompt->inputs
);
});
임베딩을 생성한 후 수신된 프롬프트에 대한 assertion을 만들 수 있습니다:
Embeddings::assertGenerated(function (EmbeddingsPrompt $prompt) {
return $prompt->contains('Laravel') && $prompt->dimensions === 1536;
});
Embeddings::assertNotGenerated(
fn (EmbeddingsPrompt $prompt) => $prompt->contains('Other')
);
Embeddings::assertNothingGenerated();
큐에 넣은 임베딩 생성의 경우 큐 assertion 메서드를 사용합니다:
Embeddings::assertQueued(
fn (QueuedEmbeddingsPrompt $prompt) => $prompt->contains('Laravel')
);
Embeddings::assertNotQueued(
fn (QueuedEmbeddingsPrompt $prompt) => $prompt->contains('Other')
);
Embeddings::assertNothingQueued();
모든 임베딩 생성에 대응하는 페이크 응답이 있는지 확인하려면 preventStrayEmbeddings를 사용할 수 있습니다. 정의된 페이크 응답 없이 임베딩이 생성되면 예외가 발생합니다:
Embeddings::fake()->preventStrayEmbeddings();
리랭킹
Reranking 클래스에서 fake 메서드를 호출해 리랭킹 작업을 페이크할 수 있습니다:
use Laravel\Ai\Reranking;
use Laravel\Ai\Prompts\RerankingPrompt;
use Laravel\Ai\Responses\Data\RankedDocument;
// Automatically generate a fake reranked responses...
Reranking::fake();
// Provide custom responses...
Reranking::fake([
[
new RankedDocument(index: 0, document: 'First', score: 0.95),
new RankedDocument(index: 1, document: 'Second', score: 0.80),
],
]);
리랭킹 후 수행된 작업에 대한 assertion을 만들 수 있습니다:
Reranking::assertReranked(function (RerankingPrompt $prompt) {
return $prompt->contains('Laravel') && $prompt->limit === 5;
});
Reranking::assertNotReranked(
fn (RerankingPrompt $prompt) => $prompt->contains('Django')
);
Reranking::assertNothingReranked();
파일
Files 클래스에서 fake 메서드를 호출하여 파일 작업을 페이크할 수 있습니다:
use Laravel\Ai\Files;
Files::fake();
파일 작업이 페이크되면 발생한 업로드와 삭제에 대한 assertion을 만들 수 있습니다:
use Laravel\Ai\Contracts\Files\StorableFile;
use Laravel\Ai\Files\Document;
// Store files...
Document::fromString('Hello, Laravel!', mimeType: 'text/plain')
->as('hello.txt')
->put();
// Make assertions...
Files::assertStored(fn (StorableFile $file) =>
(string) $file === 'Hello, Laravel!' &&
$file->mimeType() === 'text/plain';
);
Files::assertNotStored(fn (StorableFile $file) =>
(string) $file === 'Hello, World!'
);
Files::assertNothingStored();
파일 삭제에 대한 assertion의 경우 파일 ID를 전달할 수 있습니다:
Files::assertDeleted('file-id');
Files::assertNotDeleted('file-id');
Files::assertNothingDeleted();
벡터 스토어
Stores 클래스에서 fake 메서드를 호출하여 벡터 스토어 작업을 페이크할 수 있습니다. 스토어를 페이크하면 파일 작업도 자동으로 페이크됩니다:
use Laravel\Ai\Stores;
Stores::fake();
스토어 작업이 페이크되면 생성되거나 삭제된 스토어에 대한 assertion을 만들 수 있습니다:
use Laravel\Ai\Stores;
// Create store...
$store = Stores::create('Knowledge Base');
// Make assertions...
Stores::assertCreated('Knowledge Base');
Stores::assertCreated(fn (string $name, ?string $description) =>
$name === 'Knowledge Base'
);
Stores::assertNotCreated('Other Store');
Stores::assertNothingCreated();
스토어 삭제에 대한 assertion의 경우 스토어 ID를 제공할 수 있습니다:
Stores::assertDeleted('store_id');
Stores::assertNotDeleted('other_store_id');
Stores::assertNothingDeleted();
스토어에서 파일이 추가되거나 제거되었는지 assert하려면 주어진 Store 인스턴스의 assertion 메서드를 사용합니다:
Stores::fake();
$store = Stores::get('store_id');
// Add / remove files...
$store->add('added_id');
$store->remove('removed_id');
// Make assertions...
$store->assertAdded('added_id');
$store->assertRemoved('removed_id');
$store->assertNotAdded('other_file_id');
$store->assertNotRemoved('other_file_id');
파일이 provider의 파일 스토리지에 저장되고 같은 요청에서 벡터 스토어에 추가되는 경우 파일의 provider ID를 모를 수 있습니다. 이 경우 assertAdded 메서드에 클로저를 전달하여 추가된 파일의 내용에 대해 assert할 수 있습니다:
use Laravel\Ai\Contracts\Files\StorableFile;
use Laravel\Ai\Files\Document;
$store->add(Document::fromString('Hello, World!', 'text/plain')->as('hello.txt'));
$store->assertAdded(fn (StorableFile $file) => $file->name() === 'hello.txt');
$store->assertAdded(fn (StorableFile $file) => $file->content() === 'Hello, World!');
이벤트 (Events)
Laravel AI SDK는 다음을 포함한 다양한 이벤트를 발행합니다:
AddingFileToStoreAgentPromptedAgentStreamedAudioGeneratedCreatingStoreEmbeddingsGeneratedFileAddedToStoreFileDeletedFileRemovedFromStoreFileStoredGeneratingAudioGeneratingEmbeddingsGeneratingImageGeneratingTranscriptionImageGeneratedInvokingToolPromptingAgentRemovingFileFromStoreRerankedRerankingStoreCreatedStoringFileStreamingAgentToolInvokedTranscriptionGenerated