- Made `/api/auth/me` public and return explicit allowlist (name, email, two_factor_confirmed_at, tier, subscription fields) instead of spreading `$user->toArray()` which leaked is_admin, stripe_id, pm_type, pm_last_four, postcode. Returns `null` when unauthenticated rather than 401. - Moved `/auth/logout` to remain behind auth:sanctum gate. - Added 3×200ms retry with exponential backoff to EiaBrentPriceSource and FredBrentPriceSource on ConnectionException or 5xx responses. Timeout raised from 10s to 30s. - Both sources now throw typed BrentPriceFetchException on exhausted retries instead of silently returning null + logging. Updated tests to assert exception message includes HTTP status or "connection failed".
60 lines
2.1 KiB
PHP
60 lines
2.1 KiB
PHP
<?php
|
|
|
|
namespace App\Services\BrentPriceSources;
|
|
|
|
use App\Services\ApiLogger;
|
|
use Illuminate\Http\Client\ConnectionException;
|
|
use Illuminate\Http\Client\RequestException;
|
|
use Illuminate\Support\Facades\Http;
|
|
use Throwable;
|
|
|
|
final class EiaBrentPriceSource
|
|
{
|
|
private const string URL = 'https://api.eia.gov/v2/petroleum/pri/spt/data/';
|
|
|
|
public function __construct(private readonly ApiLogger $apiLogger) {}
|
|
|
|
/**
|
|
* @return array{date: string, price_usd: float}[]|null null only when the response carried no usable rows
|
|
*
|
|
* @throws BrentPriceFetchException on network failure or non-2xx response after retries
|
|
*/
|
|
public function fetch(): ?array
|
|
{
|
|
try {
|
|
$response = $this->apiLogger->send('eia', 'GET', self::URL, fn () => Http::timeout(30)
|
|
->retry(3, 200, fn (Throwable $e) => $this->shouldRetry($e))
|
|
->throw()
|
|
->get(self::URL, [
|
|
'api_key' => config('services.eia.api_key'),
|
|
'frequency' => 'daily',
|
|
'data[0]' => 'value',
|
|
'facets[series][]' => 'RBRTE',
|
|
'sort[0][column]' => 'period',
|
|
'sort[0][direction]' => 'desc',
|
|
'length' => 30,
|
|
]));
|
|
} catch (ConnectionException $e) {
|
|
throw new BrentPriceFetchException("EIA connection failed: {$e->getMessage()}", previous: $e);
|
|
} catch (RequestException $e) {
|
|
throw new BrentPriceFetchException("EIA returned HTTP {$e->response->status()}", previous: $e);
|
|
}
|
|
|
|
$rows = collect($response->json('response.data') ?? [])
|
|
->filter(fn (array $row) => ($row['value'] ?? '.') !== '.')
|
|
->map(fn (array $row) => [
|
|
'date' => $row['period'],
|
|
'price_usd' => (float) $row['value'],
|
|
])
|
|
->all();
|
|
|
|
return $rows === [] ? null : $rows;
|
|
}
|
|
|
|
private function shouldRetry(Throwable $e): bool
|
|
{
|
|
return $e instanceof ConnectionException
|
|
|| ($e instanceof RequestException && $e->response->serverError());
|
|
}
|
|
}
|