Loading...
Loading...
Use Saloon to build elegant API integrations in modern Laravel applications, including connectors, requests, responses, authentication, testing, pagination, and SDK patterns.
npx skill4agent add whoami15/claude-code-laravel-skills saloon-for-laravel$connector->send($request)app/Http/Integrations/{ServiceName}/config/saloon.phpcomposer require saloonphp/laravel-pluginsaloonphp/saloonphp artisan vendor:publish --tag=saloon-configcomposer require saloonphp/pagination-plugin # Paginated API responses
composer require saloonphp/cache-plugin # Response caching
composer require saloonphp/rate-limit-plugin # Rate limit handling
composer require saloonphp/xml-wrangler # XML reading/writingapp/Http/Integrations/
└── Acme/
├── AcmeConnector.php
├── Requests/
│ ├── GetUserRequest.php
│ ├── ListProjectsRequest.php
│ └── CreateProjectRequest.php
├── Responses/
│ └── AcmeResponse.php
└── Data/
└── UserDTO.phpphp artisan saloon:connector Acme AcmeConnectornamespace App\Http\Integrations\Acme;
use Saloon\Http\Connector;
use Saloon\Traits\Plugins\AcceptsJson;
use Saloon\Traits\Plugins\AlwaysThrowOnErrors;
class AcmeConnector extends Connector
{
use AcceptsJson;
use AlwaysThrowOnErrors;
public function __construct(
protected readonly string $token,
) {}
public function resolveBaseUrl(): string
{
return config('services.acme.base_url');
}
protected function defaultHeaders(): array
{
return [
'Authorization' => 'Bearer ' . $this->token,
];
}
}php artisan saloon:request Acme GetUserRequestnamespace App\Http\Integrations\Acme\Requests;
use Saloon\Enums\Method;
use Saloon\Http\Request;
class GetUserRequest extends Request
{
protected Method $method = Method::GET;
public function __construct(
protected readonly string $username,
) {}
public function resolveEndpoint(): string
{
return '/users/' . $this->username;
}
}HasBodyuse Saloon\Contracts\Body\HasBody;
use Saloon\Enums\Method;
use Saloon\Http\Request;
use Saloon\Traits\Body\HasJsonBody;
class CreateProjectRequest extends Request implements HasBody
{
use HasJsonBody;
protected Method $method = Method::POST;
public function __construct(
protected readonly string $name,
protected readonly bool $private = false,
) {}
public function resolveEndpoint(): string
{
return '/projects';
}
protected function defaultBody(): array
{
return [
'name' => $this->name,
'private' => $this->private,
];
}
}HasBody| Trait | Use Case |
|---|---|
| JSON payloads ( |
| Form data ( |
| File uploads ( |
| XML payloads |
| Raw string body |
| Stream/binary body |
$connector = new AcmeConnector(token: config('services.acme.token'));
$request = new GetUserRequest('johndoe');
$response = $connector->send($request);class UserController extends Controller
{
public function show(string $username, AcmeConnector $connector)
{
$response = $connector->send(new GetUserRequest($username));
return $response->json();
}
}// AppServiceProvider::register()
$this->app->bind(AcmeConnector::class, function () {
return new AcmeConnector(token: config('services.acme.token'));
});$response = $connector->send($request);
// Status checks
$response->successful(); // 200-299
$response->ok(); // 200
$response->failed(); // 400+
$response->status(); // int
// Reading data
$response->json(); // array
$response->json('user.name'); // nested key access
$response->object(); // stdClass
$response->collect(); // Illuminate\Support\Collection
$response->body(); // raw string
// Headers
$response->headers()->all();
$response->header('Content-Type');
// Error handling
$response->throw(); // throw if failed
$response->onError(fn ($response) => logger()->error('API failed', [
'status' => $response->status(),
]));use Saloon\Http\Auth\TokenAuthenticator;
use Saloon\Http\Auth\BasicAuthenticator;
use Saloon\Http\Auth\QueryAuthenticator;
use Saloon\Http\Auth\HeaderAuthenticator;
// Bearer token (most common)
$connector->authenticate(new TokenAuthenticator('my-api-token'));
// Basic auth
$connector->authenticate(new BasicAuthenticator('user', 'password'));
// Query parameter auth
$connector->authenticate(new QueryAuthenticator('api_key', 'my-key'));
// Custom header
$connector->authenticate(new HeaderAuthenticator('my-token', 'X-API-Key'));class AcmeConnector extends Connector
{
public function __construct(protected readonly string $token) {}
protected function defaultAuth(): TokenAuthenticator
{
return new TokenAuthenticator($this->token);
}
}AlwaysThrowOnErrorsuse Saloon\Traits\Plugins\AlwaysThrowOnErrors;
class AcmeConnector extends Connector
{
use AlwaysThrowOnErrors;
}use Saloon\Exceptions\Request\RequestException;
use Saloon\Exceptions\Request\ClientException; // 4xx
use Saloon\Exceptions\Request\ServerException; // 5xx
use Saloon\Exceptions\Request\FatalRequestException; // network failure
try {
$response = $connector->send($request);
} catch (ClientException $e) {
// 4xx error — $e->getResponse() gives the Response object
} catch (ServerException $e) {
// 5xx error
} catch (FatalRequestException $e) {
// Connection failure, timeout, etc.
}public function hasRequestFailed(Response $response): ?bool
{
return $response->json('success') === false;
}createDtoFromResponseuse Saloon\Http\Request;
use Saloon\Http\Response;
class GetUserRequest extends Request
{
// ...
public function createDtoFromResponse(Response $response): UserDTO
{
return new UserDTO(
name: $response->json('name'),
email: $response->json('email'),
avatar: $response->json('avatar_url'),
);
}
}$user = $connector->send(new GetUserRequest('johndoe'))->dto();
$user = $connector->send(new GetUserRequest('johndoe'))->dtoOrFail();class AcmeConnector extends Connector
{
public function __construct()
{
// Request middleware
$this->middleware()->onRequest(function (PendingRequest $pendingRequest) {
$pendingRequest->headers()->add('X-Request-ID', Str::uuid()->toString());
});
// Response middleware
$this->middleware()->onResponse(function (Response $response) {
logger()->info('Acme API responded', ['status' => $response->status()]);
});
}
}bootpublic function boot(PendingRequest $pendingRequest): void
{
$pendingRequest->config()->add('timeout', 60);
}Saloonuse Saloon\Laravel\Facades\Saloon;
use Saloon\Http\Faking\MockResponse;
it('fetches a user from the API', function () {
Saloon::fake([
GetUserRequest::class => MockResponse::make(
body: ['name' => 'Sam', 'email' => 'sam@example.com'],
status: 200,
),
]);
$response = $this->get('/api/users/johndoe');
$response->assertOk();
Saloon::assertSent(GetUserRequest::class);
Saloon::assertSentCount(1);
});Saloon::fake([
GetUserRequest::class => MockResponse::sequence([
MockResponse::make(['attempt' => 1], 500),
MockResponse::make(['attempt' => 2], 200),
]),
]);Saloon::assertSent(GetUserRequest::class);
Saloon::assertSent(fn (Request $request) => $request->resolveEndpoint() === '/users/johndoe');
Saloon::assertNotSent(DeleteUserRequest::class);
Saloon::assertSentCount(2);
Saloon::assertNothingSent();Saloon::fake();
// or
Saloon::allowStrayRequests();class AcmeConnector extends Connector
{
protected int $tries = 3;
protected int $retryInterval = 500; // milliseconds
protected bool $useExponentialBackoff = true;
protected bool $throwOnMaxTries = true;
public function handleRetry(FatalRequestException|RequestException $exception, Request $request): bool
{
// Return false to stop retrying
if ($exception instanceof RequestException && $exception->getResponse()->status() === 422) {
return false;
}
return true;
}
}composer require saloonphp/pagination-pluginHasPaginationuse Saloon\Http\Connector;
use Saloon\PaginationPlugin\PagedPaginator;
use Saloon\PaginationPlugin\Contracts\HasPagination;
class AcmeConnector extends Connector implements HasPagination
{
public function paginate(Request $request): PagedPaginator
{
return new class(connector: $this, request: $request) extends PagedPaginator
{
protected function isLastPage(Response $response): bool
{
return empty($response->json('next_page_url'));
}
protected function getPageItems(Response $response, Response $originalResponse): array
{
return $response->json('data');
}
};
}
}$paginator = $connector->paginate(new ListProjectsRequest);
foreach ($paginator as $response) {
$projects = $response->json('data');
}
// Or collect all items
$allProjects = $paginator->collect()->all();PagedPaginatorOffsetPaginatorCursorPaginatorcomposer require saloonphp/rate-limit-pluginuse Saloon\Http\Connector;
use Saloon\RateLimitPlugin\Contracts\RateLimitStore;
use Saloon\RateLimitPlugin\Limit;
use Saloon\RateLimitPlugin\Stores\LaravelCacheStore;
use Saloon\RateLimitPlugin\Traits\HasRateLimits;
class AcmeConnector extends Connector
{
use HasRateLimits;
protected function resolveLimits(): array
{
return [
Limit::allow(5000)->everyHour(),
];
}
protected function resolveRateLimitStore(): RateLimitStore
{
return new LaravelCacheStore(cache()->store());
}
}composer require saloonphp/cache-pluginuse Saloon\CachePlugin\Contracts\Cacheable;
use Saloon\CachePlugin\Contracts\Driver;
use Saloon\CachePlugin\Drivers\LaravelCacheDriver;
use Saloon\CachePlugin\Traits\HasCaching;
use Saloon\Http\Request;
class GetUserRequest extends Request implements Cacheable
{
use HasCaching;
public function resolveCacheDriver(): Driver
{
return new LaravelCacheDriver(cache()->store());
}
public function cacheExpiryInSeconds(): int
{
return 3600; // 1 hour
}
}use Saloon\Http\Response;
$pool = $connector->pool(
requests: [
new GetUserRequest('alice'),
new GetUserRequest('bob'),
new GetUserRequest('charlie'),
],
concurrency: 5,
responseHandler: function (Response $response, int $key) {
logger()->info('User fetched', ['data' => $response->json('name')]);
},
exceptionHandler: function (Exception $exception, int $key) {
logger()->error('Request failed', ['key' => $key]);
},
);
$pool->send()->wait();php artisan saloon:connector Acme AcmeOAuthConnectoruse Saloon\Http\Connector;
use Saloon\Http\Auth\AccessTokenAuthenticator;
use Saloon\Traits\OAuth2\AuthorizationCodeGrant;
class AcmeOAuthConnector extends Connector
{
use AuthorizationCodeGrant;
public function resolveBaseUrl(): string
{
return config('services.acme.base_url');
}
protected function resolveAccessTokenUrl(): string
{
return config('services.acme.base_url') . '/oauth/token';
}
protected function resolveAuthorizationUrl(): string
{
return config('services.acme.base_url') . '/oauth/authorize';
}
}// Redirect to OAuth provider
public function redirect(AcmeOAuthConnector $connector)
{
return $connector->getAuthorizationUrl(
scopes: ['read', 'write'],
state: session()->get('state'),
);
}
// Handle callback
public function callback(Request $request, AcmeOAuthConnector $connector)
{
$authenticator = $connector->getAccessToken(
code: $request->query('code'),
);
// Store for later — use encrypted cast on your model
$user->update(['acme_token' => $authenticator]);
}use Saloon\Laravel\Casts\EncryptedOAuthAuthenticatorCast;
class User extends Authenticatable
{
protected function casts(): array
{
return [
'acme_token' => EncryptedOAuthAuthenticatorCast::class,
];
}
}class AcmeConnector extends Connector
{
public function getUser(string $username): Response
{
return $this->send(new GetUserRequest($username));
}
public function listProjects(string $username): Response
{
return $this->send(new ListProjectsRequest($username));
}
public function createProject(string $name, bool $private = false): Response
{
return $this->send(new CreateProjectRequest($name, $private));
}
}$acme = new AcmeConnector(token: config('services.acme.token'));
$user = $acme->getUser('johndoe')->dto();
$projects = $acme->listProjects('johndoe')->collect('data');// config/saloon.php
'default_sender' => \Saloon\Laravel\HttpSender::class,HasBodysaloon:connectorsaloon:requestSaloon\Enums\Methodsaloonphp/pagination-pluginHttpSenderSaloon::fake()