marcincook / laravel-renteon-api-client
Laravel client for the Renteon REST API (multi-country, manager-style facade).
Package info
github.com/marcincook/laravel-renteon-api-client
pkg:composer/marcincook/laravel-renteon-api-client
Requires
- php: ^8.3
- illuminate/cache: ^11.0|^12.0|^13.0
- illuminate/contracts: ^11.0|^12.0|^13.0
- illuminate/http: ^11.0|^12.0|^13.0
- illuminate/support: ^11.0|^12.0|^13.0
Requires (Dev)
- laravel/boost: ^2.0
- laravel/pint: ^1.0
- orchestra/testbench: ^9.0|^10.0
- pestphp/pest: ^3.0|^4.0
- pestphp/pest-plugin-laravel: ^3.0|^4.0
README
A Laravel client for the Renteon REST API, with first-class support for multi-country setups (PL / ES / LT / …) and a manager-style facade familiar to Laravel developers.
Official Renteon API reference: https://demo.s2.renteon.com/en/Api/Help/Referenceex
Status
🚧 Early development — the API surface may still change before v1.0.0.
Requirements
- PHP 8.3+
- Laravel 11, 12 or 13
Installation
composer require marcincook/laravel-renteon-api-client
Publish the config:
php artisan vendor:publish --tag=renteon-config
Configuration
Set environment variables for each country you operate in:
RENTEON_DEFAULT_COUNTRY=pl RENTEON_PL_API_ENABLED=true RENTEON_PL_API_BASE_URL=https://your-tenant.renteon.com RENTEON_PL_API_USERNAME=... RENTEON_PL_API_PASSWORD=... RENTEON_PL_API_CLIENT_ID=... RENTEON_PL_API_SECRET=... RENTEON_PL_API_SALT=00000000 # Optional — used only by reports (RealizationByOffice et al.) RENTEON_REPORT_TOKEN= RENTEON_REPORT_USERNAME= RENTEON_REPORT_PASSWORD= RENTEON_REPORT_OFFICE_ID=
The same set of variables is available for RENTEON_ES_* and RENTEON_LT_*. Authentication is performed transparently: the first call exchanges your credentials for an access token (POST /token with a Base64(SHA512(…)) signature), then re-uses it for the rest of the request lifecycle.
Quick start
use MarcinCook\RenteonApi\RenteonManager; $renteon = app(RenteonManager::class); // Default country — config('renteon.default') $offices = $renteon->offices()->getOffices(); // Explicit country $cars = $renteon->for('es')->cars()->getAllCars(['IsActive' => true]);
You can also use the Renteon facade:
use Renteon; $countries = Renteon::for('pl')->countries()->getCountries();
Resources
Each resource maps to a slice of the Renteon REST API. Method signatures are kept close to the upstream model names so the official reference is your primary documentation.
| Resource | Key methods |
|---|---|
offices() |
getOffices(bool $onlyActive = true), fetchOfficesFromApi(), clearCache() |
countries() |
getCountries() (projected to {id, name}), clearCache() |
cars() |
searchCars($filters), getAllCars($filters), getCar($id) |
carCategories() |
searchCarCategories($filters), getCarCategory($id) |
addressBook() |
searchAddressBook($filters), findByEmail($email), getAddressBook($id), createAddressBook($payload), updateAddressBook($id, $payload) |
bookings() |
availability(...), create(...), calculate(...), save(...), getByNumber($number), cancel($number), checkOut($number, $officeCode), searchBookings(...), getBookingsForMonth($y, $m), getBookingById($id) |
carActivities() |
searchCarActivities($filters), getCarActivityDefinitions() |
finance() |
searchFinanceDocuments($filters), searchExportInvoices($filters), findExportInvoiceByInvoiceId($id), getCachedExPaymentTypes(), getCachedExportInvoiceDocumentTypes() |
reports() |
getRealizationByOffice($params) — uses the separate report-token flow |
Booking flow example
The typical "public-facing booking site → Renteon" pipeline looks like this:
use MarcinCook\RenteonApi\RenteonManager; $renteon = app(RenteonManager::class)->for('pl'); // 1) Lookup dictionaries for the storefront (cache these). $offices = $renteon->offices()->getOffices(); $countries = $renteon->countries()->getCountries(); $categories = $renteon->carCategories()->searchCarCategories(['IsActive' => true]); // 2) Identify the customer (email-based lookup). $customer = $renteon->addressBook()->findByEmail('jane@example.com', [ // Optional pre-filter — keeps the search payload small. 'AddressBookTypeIds' => [1], 'IsActive' => true, ]); if ($customer === null) { $customer = $renteon->addressBook()->createAddressBook([ 'Name' => 'Jane Doe', 'AddressBookTypeId' => 1, 'Email' => 'jane@example.com', 'IsAgency' => false, 'IsAgent' => false, 'IsArtisan' => false, 'IsEmployee' => false, 'IsForeigner' => false, 'IsVatPayer' => false, 'AllowB2CAccess' => true, 'BillingDelayInDays' => 0, 'SurveyDoNotEmail' => false, 'AddressBookPhones' => [ ['Number' => '+48123456789', 'PhoneTypeId' => 1], ], ]); } // 3) Check availability for the chosen date window + office pair. $availability = $renteon->bookings()->availability([ 'OfficeOutId' => 1, 'OfficeInId' => 1, 'DateTimeOut' => '2026-06-10T10:00:00+02:00', 'DateTimeIn' => '2026-06-15T10:00:00+02:00', 'AvailableOnly' => true, 'CarCategoryIds' => [/* optionally narrow down */], ]); $pickedCategory = $availability[0]['AvailabilityCarCategories'][0] ?? null; // 4) Create the booking (server computes prices + mandatory additions). $draft = $renteon->bookings()->create([ 'OfficeOutId' => 1, 'OfficeInId' => 1, 'DateTimeOut' => '2026-06-10T10:00:00+02:00', 'DateTimeIn' => '2026-06-15T10:00:00+02:00', 'ClientId' => $customer['Id'], 'BookingTypeId' => 1, 'CurrencyId' => 1, 'AvailabilityCarCategory' => $pickedCategory, ]); // (optional) Recalculate after the customer adds extras / insurance. $draft['Services'][] = ['ServiceId' => 7, 'Quantity' => 1]; $draft = $renteon->bookings()->calculate($draft); // 5) After your payment gateway clears the deposit & extras — persist it. $booking = $renteon->bookings()->save(array_merge($draft, [ 'CarId' => 42, 'ClientId' => $customer['Id'], 'ClientName' => 'Jane Doe', 'ClientEmail' => 'jane@example.com', 'TotalDeposit' => 365.38, 'Remark' => 'Paid online via Stripe', ])); // $booking['Number'] is now safe to persist in your local DB for the // customer's booking history.
Reports
Reports::getRealizationByOffice() uses a different authentication flow — set either RENTEON_REPORT_TOKEN (a raw bearer token captured from the Renteon panel) or RENTEON_REPORT_USERNAME/RENTEON_REPORT_PASSWORD/RENTEON_REPORT_OFFICE_ID. If neither is set, the client falls back to the country token.
$rows = $renteon->reports()->getRealizationByOffice([ 'DateTimeInFrom' => '01.06.2026.', 'DateTimeInTo' => '30.06.2026.', 'IncludeExternalDocuments' => false, ]);
Caching
Dictionary endpoints (offices, countries, payment types, document types) are cached using the Laravel cache repository, keyed per country. TTL defaults to 24h and is configurable via RENTEON_DICT_CACHE_TTL (seconds). Use the clearCache() method on each resource to invalidate.
Error handling
All HTTP failures bubble up as exceptions:
MarcinCook\RenteonApi\Exceptions\RenteonAuthException—/tokenendpoint failed or returned noaccess_token.MarcinCook\RenteonApi\Exceptions\RenteonHttpException— any other non-2xx response. The originalIlluminate\Http\Client\Responseis attached as->response.MarcinCook\RenteonApi\Exceptions\RenteonException— base class for both.
The two top-level dictionary getters (getOffices, getCountries) deliberately swallow API failures and log a warning — they're used in dropdowns where a transient outage shouldn't break the page.
Testing
vendor/bin/pest vendor/bin/pint --test
License
MIT — see LICENSE.