Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(anthropic): text stream support #266

Open
wants to merge 40 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
899a60a
anthropic stream support
Mar 15, 2025
c33102f
anthropic stream tests and sse files
Mar 15, 2025
1eee715
updated for v0.51
Mar 15, 2025
7156c1b
remove comments
Mar 15, 2025
bb482ae
remove complexity in ToolCallMap
Mar 15, 2025
2316c30
added doc blocks
Mar 15, 2025
5ef4580
Revert "added doc blocks"
Mar 15, 2025
c91b5a9
added doc blocks
Mar 15, 2025
01ac8cc
tool call improvement + exception on argument failure
Mar 16, 2025
471e592
sendRequest tidyup
Mar 16, 2025
23bfdc8
next line improvement
Mar 16, 2025
9e336d7
finish reason fix
Mar 16, 2025
95ca741
stop reason was needed
Mar 16, 2025
9843acd
properly handle retry-after header in Anthropic ProcessesRateLimits
Mar 17, 2025
b337883
fixed parameter name for tools + mapToolCalls moved to stream class
Mar 17, 2025
44d0c90
stream with citations using MessagePartWithCitations
Mar 17, 2025
8bfb4f2
rate limit trait to match abstract rate limit
Mar 17, 2025
04cc5c7
code tidy up
Mar 21, 2025
58a4ec8
thinking test
Mar 21, 2025
f708c9c
formatting
Mar 21, 2025
6bfad7a
formatting and phpstan
Mar 22, 2025
843a686
rm comment
Mar 22, 2025
bf5461f
thinking sse file - rm empty spaces
Mar 22, 2025
e6fa801
Merge branch 'main' of https://github.com/prism-php/prism into feat/a…
Mar 25, 2025
1baa2ee
exception handling
ChrisB-TL Mar 26, 2025
5b50646
add meta to final chunk
ChrisB-TL Mar 26, 2025
4602539
refactor to use match for readability + reasoning
ChrisB-TL Mar 26, 2025
9b239d8
simplify/clarify state
ChrisB-TL Mar 26, 2025
e696ab0
add comments to match statement + yield thinking chunks
ChrisB-TL Mar 26, 2025
83819f5
fix citations
ChrisB-TL Mar 26, 2025
ce814f7
add note to docs re citation streaming
ChrisB-TL Mar 26, 2025
83d5123
use text request payload method
ChrisB-TL Mar 26, 2025
354fe7b
stop_reason and FinishReason fix
Mar 28, 2025
665b6d8
formatting
Mar 28, 2025
0e20894
'error' case added
Mar 29, 2025
e58271a
PrismProviderOverloadedException exception for "overloaded" case
Mar 29, 2025
c60a008
phpstan
Mar 30, 2025
77dcd03
pint formatting
Apr 3, 2025
465a6c7
Merge branch 'main' into feat/anthropic-stream-0.51
Apr 3, 2025
0ed862b
added peck.json from main
Apr 3, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 7 additions & 2 deletions src/Providers/Anthropic/Anthropic.php
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,9 @@
use Prism\Prism\Contracts\Provider;
use Prism\Prism\Embeddings\Request as EmbeddingRequest;
use Prism\Prism\Embeddings\Response as EmbeddingResponse;
use Prism\Prism\Exceptions\PrismException;
use Prism\Prism\Providers\Anthropic\Handlers\Structured;
use Prism\Prism\Providers\Anthropic\Handlers\Text;
use Prism\Prism\Providers\Anthropic\Handlers\Stream;
use Prism\Prism\Structured\Request as StructuredRequest;
use Prism\Prism\Structured\Response as StructuredResponse;
use Prism\Prism\Text\Request as TextRequest;
Expand Down Expand Up @@ -57,7 +57,12 @@ public function structured(StructuredRequest $request): StructuredResponse
#[\Override]
public function stream(TextRequest $request): Generator
{
throw PrismException::unsupportedProviderAction(__METHOD__, class_basename($this));
$handler = new Stream($this->client(
$request->clientOptions(),
$request->clientRetry()
));

return $handler->handle($request);
}

#[\Override]
Expand Down
77 changes: 77 additions & 0 deletions src/Providers/Anthropic/Concerns/ProcessesRateLimits.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
<?php

declare(strict_types=1);

namespace Prism\Prism\Providers\Anthropic\Concerns;

use Illuminate\Http\Client\Response;
use Illuminate\Support\Arr;
use Illuminate\Support\Carbon;
use Illuminate\Support\Str;
use Prism\Prism\ValueObjects\ProviderRateLimit;

trait ProcessesRateLimits
{
/**
* Process rate limit headers from Anthropic API responses.
*
* @return ProviderRateLimit[]
*/
protected function processRateLimits(Response $response): array
{
$limitHeaders = array_filter($response->getHeaders(), fn ($headerName) => Str::startsWith($headerName, 'x-ratelimit-'), ARRAY_FILTER_USE_KEY);

$rateLimits = [];

foreach ($limitHeaders as $headerName => $headerValues) {
// Parse header name according to Anthropic's format: x-ratelimit-{limit/remaining/reset}-{resource_type}
$parts = explode('-', $headerName);

// Expect format like x-ratelimit-limit-requests or x-ratelimit-remaining-tokens
if (count($parts) >= 3) {
$fieldName = $parts[1]; // limit, remaining, or reset
$limitName = $parts[2]; // requests, tokens, etc.

$rateLimits[$limitName][$fieldName] = $headerValues[0];
}
}

// Also check for retry-after header
$retryAfter = $response->header('retry-after');
if ($retryAfter !== null) {
$rateLimits['global']['reset'] = $retryAfter . 's';
}

return array_values(Arr::map($rateLimits, function ($fields, $limitName): ProviderRateLimit {
$resetsAt = data_get($fields, 'reset', '');

// Anthropic typically provides reset times in seconds
if (is_numeric($resetsAt)) {
$resetSeconds = (int) $resetsAt;
$resetMilliseconds = 0;
$resetMinutes = 0;
} elseif (str_contains($resetsAt, 's')) {
$resetSeconds = (int) Str::of($resetsAt)->before('s')->toString();
$resetMilliseconds = 0;
$resetMinutes = 0;
} else {
$resetSeconds = 0;
$resetMilliseconds = 0;
$resetMinutes = 0;
}

return new ProviderRateLimit(
name: $limitName,
limit: data_get($fields, 'limit') !== null
? (int) data_get($fields, 'limit')
: null,
remaining: data_get($fields, 'remaining') !== null
? (int) data_get($fields, 'remaining')
: null,
resetsAt: data_get($fields, 'reset') !== null
? Carbon::now()->addMinutes((int) $resetMinutes)->addSeconds((int) $resetSeconds)->addMilliseconds((int) $resetMilliseconds)
: null
);
}));
}
}
Loading