motomedialab / connector
A super lightweight request/connector pattern to make light work of API authentication and calls
Installs: 37
Dependents: 1
Suggesters: 0
Security: 0
Stars: 0
Watchers: 0
Forks: 0
Open Issues: 0
pkg:composer/motomedialab/connector
Requires
- php: ^8.2
- laravel/framework: ^11.0|^12.0
Requires (Dev)
- larastan/larastan: ^2.0|^3.0
- laravel/pint: ^v1.23
- orchestra/testbench: ^9.0|^10.0
- pestphp/pest: ^2.0|^3.0|^4.0
README
A super lightweight, opinionated connector pattern for Laravel to make light work of consuming third-party APIs.
Introduction
Integrating with external APIs often involves repetitive boilerplate for handling authentication, endpoints, and requests. This package provides a structured and reusable pattern to streamline this process, keeping your code clean, consistent, and easy to maintain.
Installation
You can install the package via composer:
composer require motomedialab/connector
Core Concepts
The package is built around two core concepts: Connectors and Requests.
Connectors
The Connector is responsible for defining the base URL and authentication method for an API. Your connector must extend the Motomedialab\Connector\BaseConnector abstract class.
// app/Connectors/ExampleConnector.php use Illuminate\Http\Client\PendingRequest; use Motomedialab\Connector\BaseConnector; class ExampleConnector extends BaseConnector { /** * Optionally authenticate all requests from this connector. * For example, an access or bearer token. */ public function authenticateRequest(PendingRequest $request): PendingRequest { return $request->withToken('your-secret-token'); } /** * Define the base URL for the API you are connecting with */ public function apiUrl(): string { return 'https://api.example.com/v2/'; } }
Requests
The Request defines the specific details of an API call, such as the endpoint, method, headers,
and payload. Your request classes must extend the Motomedialab\Connector\BaseRequest abstract class
and should implement the Motomedialab\Connector\Contracts\RequestInterface contract.
The BaseRequest provides sensible defaults, so you only need to define what you need to override.
Simple GET Request
Here's a minimal example for a simple GET request:
// app/Requests/ExampleGetRequest.php use Illuminate\Http\Client\Response; use Motomedialab\Connector\BaseRequest; use Motomedialab\Connector\Contracts\RequestInterface; /** * @implements RequestInterface<array> */ readonly class ExampleGetRequest extends BaseRequest implements RequestInterface { public function __construct(private string $id) { // } public function endpoint(): string { return "users/{$this->id}"; } public function toResponse(Response $response): array { return $response->json(); } }
Advanced POST Request
This example demonstrates all available methods for customising a request.
// app/Requests/ExamplePostRequest.php use Illuminate\Http\Client\Response; use Motomedialab\Connector\BaseRequest; use Motomedialab\Connector\Enums\RequestMethod; use Motomedialab\Connector\Contracts\RequestInterface; /** * @implements RequestInterface<array> */ readonly class ExamplePostRequest extends BaseRequest implements RequestInterface { public function __construct(public array $inputData = []) { // } // Define the HTTP method. Defaults to GET. public function method(): RequestMethod { return RequestMethod::POST; } // Specify the request timeout in seconds. Defaults to 5. public function timeout(): int { return 10; } // Add query parameters to the URL. public function queryParams(): array { return ['include' => 'posts']; } // Add or override headers. Defaults include JSON content type. public function headers(): array { return [ ...parent::headers(), 'X-Custom-Header' => 'CustomValue', ]; } // REQUIRED: Define the endpoint, appended to the connector's apiUrl. public function endpoint(): string { return 'users'; } // Define the request payload. public function body(): array { return $this->inputData; } // Determine if the request requires authentication. Defaults to true. public function authenticated(): bool { return true; } // Transform the successful response. public function toResponse(Response $response): array { if ($response->failed()) { // You can handle error responses here. // For example, return a default structure or throw an exception. return ['error' => true, 'status' => $response->status()]; } return $response->json(); } }
Usage
Sending a Request
To send a request, simply instantiate your connector and request, then call the send() method.
$connector = new ExampleConnector(); $request = new ExamplePostRequest(['name' => 'Chris']); // The response will be whatever you return from toResponse() $response = $connector->send($request);
Asynchronous Requests
You can also send requests concurrently using the sendAsync() method. This is great for
performance when you need to make multiple independent API calls.
use GuzzleHttp\Promise\Utils; $connector = new ExampleConnector(); // Create an array of requests $requests = [ new ExampleGetRequest('user-1'), new ExampleGetRequest('user-2'), new ExampleGetRequest('user-3'), ]; // Map requests to promises $promises = array_map( fn($request) => $connector->sendAsync($request), $requests ); // Wait for all promises to resolve $responses = Utils::unwrap($promises);
Testing
This package is designed to work seamlessly with Laravel's Http::fake(). You can write your
tests as you normally would.
use Illuminate\Support\Facades\Http; use App\Connectors\ExampleConnector; use App\Requests\ExampleGetRequest; it('sends a get request', function () { Http::fake([ 'api.example.com/*' => Http::response(['name' => 'John']), ]); $connector = new ExampleConnector(); $response = $connector->send(new ExampleGetRequest('user-1')); expect($response['name'])->toBe('John'); });
Contributing
Please see CONTRIBUTING for details.
License
The MIT License (MIT). Please see License File for more information.