API Integration Examples
Examples for integrating with external APIs.
Introduction
Section titled “Introduction”Common patterns for API integration:
- ✅ REST APIs - GET, POST, PUT, DELETE
- ✅ Webhooks - Handle incoming webhooks
- ✅ Response Mapping - Map API responses to DTOs
- ✅ Error Handling - Handle API errors
REST API Client
Section titled “REST API Client”GET Request
Section titled “GET Request”use event4u\DataHelpers\SimpleDTO\SimpleDTO;use event4u\DataHelpers\SimpleDTO\Attributes\MapFrom;
class UserDTO extends SimpleDTO{    public function __construct(        #[MapFrom('data.id')]        public readonly int $id,
        #[MapFrom('data.attributes.name')]        public readonly string $name,
        #[MapFrom('data.attributes.email')]        public readonly string $email,    ) {}}
// Fetch from API$response = Http::get('https://api.example.com/users/1');$dto = UserDTO::fromArray($response->json());POST Request
Section titled “POST Request”class CreateUserDTO extends SimpleDTO{    public function __construct(        #[Required, Min(3)]        public readonly string $name,
        #[Required, Email]        public readonly string $email,    ) {}}
$dto = CreateUserDTO::fromArray($_POST);$dto->validate();
$response = Http::post('https://api.example.com/users', $dto->toArray());PUT Request
Section titled “PUT Request”class UpdateUserDTO extends SimpleDTO{    public function __construct(        public readonly ?string $name = null,        public readonly ?string $email = null,    ) {}}
$dto = UpdateUserDTO::fromArray($_POST);
$response = Http::put("https://api.example.com/users/{$id}",    array_filter($dto->toArray()));Weather API Integration
Section titled “Weather API Integration”class WeatherDTO extends SimpleDTO{    public function __construct(        #[MapFrom('location.name')]        public readonly string $city,
        #[MapFrom('current.temp_c')]        public readonly float $temperature,
        #[MapFrom('current.condition.text')]        public readonly string $condition,
        #[MapFrom('current.humidity')]        public readonly int $humidity,    ) {}}
$response = Http::get('https://api.weatherapi.com/v1/current.json', [    'key' => env('WEATHER_API_KEY'),    'q' => 'London',]);
$weather = WeatherDTO::fromArray($response->json());
echo "Temperature in {$weather->city}: {$weather->temperature}°C\n";echo "Condition: {$weather->condition}\n";GitHub API Integration
Section titled “GitHub API Integration”class GitHubRepoDTO extends SimpleDTO{    public function __construct(        public readonly int $id,        public readonly string $name,
        #[MapFrom('full_name')]        public readonly string $fullName,
        public readonly string $description,
        #[MapFrom('stargazers_count')]        public readonly int $stars,
        #[MapFrom('forks_count')]        public readonly int $forks,
        #[MapFrom('html_url')]        public readonly string $url,    ) {}}
$response = Http::get('https://api.github.com/repos/event4u-app/data-helpers');$repo = GitHubRepoDTO::fromArray($response->json());
echo "{$repo->fullName}\n";echo "Stars: {$repo->stars}, Forks: {$repo->forks}\n";Stripe API Integration
Section titled “Stripe API Integration”class StripeCustomerDTO extends SimpleDTO{    public function __construct(        public readonly string $id,        public readonly string $email,        public readonly string $name,
        #[Cast(DateTimeCast::class)]        public readonly Carbon $created,    ) {}}
$response = Http::withToken(env('STRIPE_SECRET'))    ->get('https://api.stripe.com/v1/customers/cus_123');
$customer = StripeCustomerDTO::fromArray($response->json());Webhook Handling
Section titled “Webhook Handling”GitHub Webhook
Section titled “GitHub Webhook”class GitHubWebhookDTO extends SimpleDTO{    public function __construct(        public readonly string $action,
        #[MapFrom('repository.name')]        public readonly string $repository,
        #[MapFrom('sender.login')]        public readonly string $sender,    ) {}}
// Handle webhook$payload = json_decode(file_get_contents('php://input'), true);$webhook = GitHubWebhookDTO::fromArray($payload);
match($webhook->action) {    'opened' => handlePullRequestOpened($webhook),    'closed' => handlePullRequestClosed($webhook),    default => null,};Stripe Webhook
Section titled “Stripe Webhook”class StripeWebhookDTO extends SimpleDTO{    public function __construct(        public readonly string $type,
        #[MapFrom('data.object')]        public readonly array $object,    ) {}}
$payload = json_decode(file_get_contents('php://input'), true);$webhook = StripeWebhookDTO::fromArray($payload);
match($webhook->type) {    'payment_intent.succeeded' => handlePaymentSuccess($webhook),    'payment_intent.failed' => handlePaymentFailure($webhook),    default => null,};Pagination
Section titled “Pagination”class PaginatedResponseDTO extends SimpleDTO{    public function __construct(        public readonly array $data,
        #[MapFrom('meta.current_page')]        public readonly int $currentPage,
        #[MapFrom('meta.last_page')]        public readonly int $lastPage,
        #[MapFrom('meta.total')]        public readonly int $total,    ) {}}
$response = Http::get('https://api.example.com/users', [    'page' => 1,    'per_page' => 20,]);
$paginated = PaginatedResponseDTO::fromArray($response->json());
echo "Page {$paginated->currentPage} of {$paginated->lastPage}\n";echo "Total: {$paginated->total}\n";Error Handling
Section titled “Error Handling”class ApiErrorDTO extends SimpleDTO{    public function __construct(        public readonly string $message,        public readonly int $code,        public readonly ?array $errors = null,    ) {}}
try {    $response = Http::post('https://api.example.com/users', $data);
    if ($response->failed()) {        $error = ApiErrorDTO::fromArray($response->json());        throw new ApiException($error->message, $error->code);    }
    $user = UserDTO::fromArray($response->json());} catch (ApiException $e) {    Log::error('API error', ['message' => $e->getMessage()]);}Rate Limiting
Section titled “Rate Limiting”class RateLimitDTO extends SimpleDTO{    public function __construct(        #[MapFrom('X-RateLimit-Limit')]        public readonly int $limit,
        #[MapFrom('X-RateLimit-Remaining')]        public readonly int $remaining,
        #[MapFrom('X-RateLimit-Reset')]        public readonly int $reset,    ) {}}
$response = Http::get('https://api.github.com/user');$rateLimit = RateLimitDTO::fromArray($response->headers());
if ($rateLimit->remaining < 10) {    Log::warning('API rate limit low', [        'remaining' => $rateLimit->remaining,        'reset' => Carbon::createFromTimestamp($rateLimit->reset),    ]);}See Also
Section titled “See Also”- Property Mapping - MapFrom attribute
- Type Casting - Type casting
- Validation - Validation rules
