Skip to content

Flight Search UI

Native FilamentPHP v5 implementation of flight search interface with interactive leg selection, collapsible panels, and component-based architecture.

URL: /admin/flight-searches/flights

Navigation: Admin Panel → Flight Searches → Flights (Aerticket)

  1. Select departure and arrival airports (searchable dropdowns)
  2. Choose dates and passenger counts
  3. Configure fare options and preferences
  4. Click “Search Flights”
  5. Browse results with interactive leg selection
  6. View fare details and book flights

File: app/Filament/Resources/FlightSearches/Pages/SearchFlights.php

Components:

  • Form - Search criteria using FilamentPHP Schema
  • Table - Results display using FilamentPHP Table Builder
  • Actions - Search, clear, and fare details actions
  • Livewire Methods - Interactive leg selection
SearchFlights (Page)
├── Form (Schema)
│ ├── Section: Flight Search
│ │ ├── Grid: Origin/Destination/Dates
│ │ ├── Grid: Passengers/Cabin
│ │ ├── Section: Fare Options
│ │ ├── Section: Flight Preferences
│ │ └── Section: Airline Filtering
├── Table
│ ├── Split: Fare Summary
│ │ ├── Stack: Price Info
│ │ └── Split: Badges (Stops, Duration, Source)
│ └── Panel: Collapsible Flight Legs
│ ├── Stack: Outbound Options
│ │ └── Split: Radio + Leg Details (clickable)
│ └── Stack: Return Options
│ └── Split: Radio + Leg Details (clickable)
└── Actions
├── searchFlights()
├── clearResults()
└── viewFareDetails()
Radio::make('trip_type')
->label('Trip Type')
->options([
'one_way' => 'One Way',
'round_trip' => 'Round Trip',
'multi_city' => 'Multi-City',
])
->default('one_way')
->inline()
->live()
->columnSpanFull()

Reactive Behavior:

  • Return date field visibility tied to trip_type === 'round_trip'
  • Multi-city segments visible when trip_type === 'multi_city'
Repeater::make('segments')
->label('Trip Segments')
->visible(fn (Get $get): bool => $get('trip_type') === 'multi_city')
->minItems(2)
->maxItems(6)
->schema([
Grid::make(3)
->schema([
Select::make('departure_airport'),
Select::make('arrival_airport'),
DatePicker::make('departure_date'),
]),
])

Constraints:

  • Minimum 2 segments, maximum 6 segments
  • All segment dates must be in chronological order
  • Validated at search time
Grid::make(4)
->schema([
TextInput::make('adults')
->numeric()
->default(1)
->minValue(1)
->maxValue(9),
TextInput::make('children')
->label('Children (2-11)')
->default(0),
TextInput::make('infants')
->label('Infants (0-1)')
->default(0),
Select::make('cabin_class')
->options([
'ECONOMY' => 'Economy',
'ECONOMY_PREMIUM' => 'Premium Economy',
'BUSINESS' => 'Business',
'FIRST' => 'First',
]),
])

Validation:

  • Total passengers (adults + children + infants) ≤ 9
  • At least 1 adult required
CheckboxList::make('fare_sources')
->options([
'FSC_IATA' => 'IATA Published Fares',
'FSC_NEGO' => 'Negotiated Fares',
'FSC_CONSO' => 'Consolidator Fares',
'FSC_WEB' => 'Direct Channel & Low-Cost Fares',
])
->default(['FSC_IATA'])
TagsInput::make('included_airlines')
->label('Include Airlines')
->placeholder('Enter airline codes (e.g., LH, BA, AA)')
->separator(',')
TagsInput::make('excluded_airlines')
->label('Exclude Airlines')
->separator(',')

Validation:

  • IATA codes: 2-3 alphanumeric characters
  • Cannot have overlapping included/excluded airlines
Split::make([
// Left side: Price and fare information
Stack::make([
TextColumn::make('formattedPrice')
->weight(FontWeight::Bold)
->color('success'),
TextColumn::make('fareType'),
TextColumn::make('cabinClass')
->icon('heroicon-m-ticket'),
])->space(1),
// Right side: Badges
Split::make([
TextColumn::make('numberOfStops')
->badge()
->color(fn (int $state): string => match ($state) {
0 => 'success',
1 => 'warning',
default => 'danger',
}),
TextColumn::make('formattedDuration')
->badge()
->icon('heroicon-m-clock'),
])->from('sm'),
])->from('md')

Users can select different flight options for each direction:

Panel::make([
Stack::make(fn ($record) => $this->buildFlightLegComponents($record))
->space(3),
])->collapsible()->collapsed(true)

Leg Structure (Round-Trip):

Panel (Collapsible)
├── Outbound Header
├── Leg Option 1 (Split: Radio + Details) [Clickable]
├── Leg Option 2 (Split: Radio + Details) [Clickable]
├── Return Header
├── Leg Option 1 (Split: Radio + Details) [Clickable]
└── Leg Option 2 (Split: Radio + Details) [Clickable]

Leg Structure (Multi-City):

Panel (Collapsible)
├── Leg 1 Header
├── Leg Option 1 [Clickable]
├── Leg 2 Header
├── Leg Option 1 [Clickable]
└── ... (up to 6 legs)
public function selectLeg(string $fareId, string $direction, int $legIndex): void
{
if (!isset($this->selectedLegs[$fareId])) {
$this->selectedLegs[$fareId] = [
'Outbound' => 0,
'Return' => 0,
];
}
$this->selectedLegs[$fareId][$direction] = $legIndex;
}
ViewAncillariesAction::make()
->rawFareData($this->rawFareData)
->selectedLegs($this->selectedLegs)

Features:

  • Opens modal with available ancillary services
  • Component-level caching prevents duplicate API requests
  • Displays services grouped by type (BAGGAGE, MEAL, SEAT)
ViewFareDetailsAction::make()
->rawFareData($this->rawFareData)

Features:

  • Complete fare breakdown
  • Passenger pricing
  • Baggage allowance
  • Booking action
UpsellAction::make()
->rawFareData($this->rawFareData)

Features:

  • Searches for upgraded fare options
  • Uses user’s selected leg choices
  • Replaces search results with upsell results
  • sm - 640px
  • md - 768px
  • lg - 1024px

Usage:

Split::make([/* ... */])->from('md') // Horizontal on medium+
Stack::make([/* ... */]) // Always vertical
  • Full-width fields on mobile
  • Grid collapses to single column
  • Collapsible panels for details
  • Touch-friendly input sizes
  • Limit to 10 results for dropdowns
  • Uses GIN index for sub-millisecond queries
  • See Airport Search documentation
  • Component-level cache prevents duplicate API requests
  • Cache key: fareId|itinerary1:itinerary2
  • Automatic invalidation on search state changes
public function clearResults(): void
{
$this->searchResults = collect();
$this->rawFareData = [];
$this->selectedLegs = [];
$this->fareAncillaryCache = [];
$this->hasSearched = false;
}
/admin/flight-searches/flights
# Access flight search page
# Test scenarios:
1. One-way search: BCN MAD, 1 adult, Economy
2. Round-trip search: LHR JFK, 2 adults + 1 child, Business
3. Multi-city search: BCN BKK CNX BCN, 2 adults
4. Multi-leg selection: Expand panel, click different options
5. Fare details: Click "View Details" action
6. Validation: Try >9 passengers
use function Pest\Livewire\livewire;
test('flight search form renders correctly', function () {
livewire(SearchFlights::class)
->assertFormExists()
->assertFormFieldExists('departure_airport')
->assertFormFieldExists('arrival_airport')
->assertFormFieldExists('departure_date')
->assertFormFieldExists('adults');
});
test('leg selection updates state', function () {
$component = livewire(SearchFlights::class);
$component->selectLeg('fare-123', 'Outbound', 1);
expect($component->selectedLegs['fare-123']['Outbound'])->toBe(1);
});
  1. Check API credentials configured
  2. Verify AerTicket service accessible
  3. Validate airport codes
  4. Check date range (max 11 months)
Terminal window
php artisan aerticket:test-connection
php artisan pail --filter="Aerticket"
  1. Check Livewire scripts loaded
  2. Check browser console for errors
  3. Verify wire:click attribute present

Admin-configured filters applied to all automated flight cache searches (the populator cycle). They do NOT affect the manual Flight Search page or the route validation checks.

URL: /admin/settings/flight-search-filters

Source: backend/app/Filament/Pages/Settings/FlightSearchFiltersSettings.php, backend/app/Services/SettingsService.php

Setting keyTypeDefaultPurpose
flights.search.baggage_policystring (BaggagePolicy)UPSELLABLEBaggage handling policy — see below
flights.search.max_stopsint2Maximum stopovers per leg requested from Aerticket (ignored when direct_flights_only is on)
flights.search.direct_flights_onlyboolfalseNon-stop only; overrides max_stops
flights.search.cabin_class_listlist[]Cabin classes (empty = all)
flights.search.excluded_airline_listlist[]IATA codes to exclude
flights.search.airline_alliance_listlist[]Preferred alliances (empty = all)
flights.search.fare_source_listlistIATA + NEGO + CONSO + WEBFare sources to query
flights.search.closed_user_group_listlist[ALL, TOP]Closed user groups (see below)
flights.search.latest_arrival_timeH:i stringnullPost-cache arrival compliance cutoff

Select one of OFF / STRICT / UPSELLABLE. Controls how fares without an included checked bag are handled during populator runs:

  • OFF — cache everything, never filter by baggage.
  • STRICT — only cache fares with a bag in the base price. Sends needBaggage=true to Aerticket so the filter runs server-side.
  • UPSELLABLE (default) — cache fares that include a bag or have a bag-included sibling for the same physical itinerary inside the same /search response.

See Dynamic Flight Cache — Baggage Resolution for the /search-only resolution flow.

Closed User Groups (closed_user_group_list)

Section titled “Closed User Groups (closed_user_group_list)”

Union filter over Aerticket CUG codes. The API rejects an empty list, so clearing all options falls back to [ALL, TOP] (Cockpit’s default-plus-tour-operator view).

  • ALL — public bucket Cockpit shows by default.
  • TOP — Tour Operator bucket Cockpit hides behind a dropdown. Surfaces consolidator/tour-operator fares that only appear when TOP is explicitly selected in Cockpit. Included by default so Volare can undercut Cockpit’s default “best price”.
  • ETH — ethnic fares.
  • CRU — cruise fares.

H:i cutoff flagging fares whose outbound-leg destination arrival falls outside the daytime window [06:00, cutoff]. Local-clock comparison only — dates are ignored:

  • HH:MM > cutoff → flagged. A 17:30 arrival with a 16:00 cap is late.
  • HH:MM < 06:00 → flagged. Any pre-dawn landing (00:00–05:59) is late regardless of cutoff.
  • Next-day 15:30 with a 16:00 cap → compliant. The clock reads 15:30 on landing.

The populator does not drop flagged fares. They are cached and flagged so operators can distinguish “Aerticket returned nothing” from “Aerticket returned fares that don’t comply”. The admin list shows an Arrival Compliance badge computed from this setting.

Two places apply the cutoff:

  • The admin list, via DynamicFlightCache::violatesLatestArrivalTime() — visual flag only.
  • The auto offer generator, via AutoOfferGeneratorService::rejectNonCompliantArrivals() — drops flagged cached rows before offer creation.

Both use the shared FareTimeWindowFilter::hhmmWithinDaytimeWindow() so the two paths can’t drift. Leave the setting empty to disable both.

Related:

FlightRankingPolicy is the single source of truth for “which flight is better” across the cache populator (fare_position), the auto-offer generator, the upgrade service, the flights:rerank-dynamic-caches console command, and the live-search at checkout. All paths apply the same filter + comparator rules so the cached “position 1”, the generator’s pick, and the live-search winner stay aligned. Pareto pruning is still applied only on the back-office/frontier paths that explicitly need it; checkout display keeps visible trade-offs after the hard filters.

The rules were agreed with operators (issue #1779) and replace the previous 5-level lexicographic order (stops → layover → outbound flying time → price → departure):

  1. Filter — drop any fare that violates baggage, max stops, max layover, return departure, departure window, or nights-at-destination rules for the relevant caller. Max layover is 8 hours (MAX_LAYOVER_MINUTES = 480). Direct flights pass the layover check trivially. The filters run before sorting.
  2. Stop-tier prune — keep the lowest stop-count tier available in the valid result set. If 1-stop options exist, 2-stop options are hidden; if the route only has 2-stop options, they remain visible.
  3. Order — by total round-trip duration, gate-to-gate, summed across outbound + return. Includes both flight time and layover gaps. For cache rows this uses LegDuration::forLeg (timezone-aware where airport IANA timezones are known, falls back to per-segment + layover sum). For live FareResult DTOs it sums segment.duration + layover gaps directly — timezone-safe since layover gaps are at the same airport.
  4. Tiebreaks — price, then fewer stops, then earliest outbound arrival at destination.
  5. Prune (Pareto)pruneDominatedCacheFares / pruneDominatedFareResults drop any fare that is both longer and more expensive than another kept fare (with at least one strict inequality). Longer-but-cheaper fares stay. This subsumes the operator rule “a cheaper direct erases longer pricier stopovers” — no special case needed.

The two comparators (compareFlightCache for DB rows, compareFareResult for live DTOs) target different shapes but implement the same order.

Source: backend/app/Services/Offers/FlightRankingPolicy.php

CallerFilterOrderPruneVisual dedup
DynamicFlightCachePopulatorService (assigns fare_position)yesyesyes (full frontier)no
AutoOfferGeneratorService (picks the offer’s flight)yesyesyes (full frontier)no
OfferFlightUpgradeService::upgradeIfBetternonoPareto (duration, price) guard against current bound row — bypassable by checkout live-winner binding. See Offer Flight Upgradeno
flights:rerank-dynamic-caches console commandyesyesyes (full frontier)no
EconomyFlightSearchService::rankDisplayFares / bindFirstDisplayFare (checkout live winner)yes — same checkout policy plus baggage resolutionyesyes — stop-count display frontier after baggageyes — selected fare must be bindable
EconomyFlightSearchService::transformResults (display, international)receives pre-ranked bound faresyesnoyes — policy winner preferred
EconomyFlightSearchService::searchDomesticFlight (display, domestic)nononono
BusinessFlightSearchService::transformBusinessResults (display)yesyesnoyes — cheapest

EconomyFlightSearchService::rankDisplayFares is the sole authority on which economy live-search fare can be marked “Seleccionado”. The same ranked list is then handed to bindFirstDisplayFare, which materializes the first bindable fare as a DynamicFlightCache row and rebinds the offer before the response is rendered. This keeps checkout display, offer_flights, offer_flight_bindings, and admin in sync.

Before a fare can be ranked or bound it must pass, in order:

  1. Arrival cutoff (applied upstream when building $arrivalFiltered).
  2. FlightRankingPolicy::passesMaxLayoverFare.
  3. FlightRankingPolicy::passesMinReturnDepartureFare.
  4. FlightRankingPolicy::passesMaxStopsFare($fare, $maxStops).
  5. FlightRankingPolicy::passesDepartureWindowFare($fare, ...$offer->outboundDepartureWindow()).
  6. FlightRankingPolicy::passesNightsAtDestinationFare($fare, $targetNights) — when targetNights can be derived. The target is sourced via the private helper targetNightsForOffer(Offer $offer): ?int, which reads FlightRankingPolicy::targetNightsAtDestinationForRoute() off the bound cache row’s route. Returns null (filter skipped) when the offer has no cache-sourced international OfferFlight or the route can’t yield a target.

After filtering, candidates are ordered by the shared policy and pass through the /search-only baggage chain: native bag → in-response fare-family sibling. Candidates that cannot resolve a checked bag on both international legs under an active baggage policy are dropped. The filtered set is then sorted by compareFareResult, visually deduped, trimmed to the stop-count display frontier, and capped to the checkout display limit.

bindFirstDisplayFare iterates the already-ranked display list. The first fare that can be found or materialized through EconomyFlightCacheUpdateService::findOrCreateCacheRowForFare is rebound with OfferFlightUpgradeService::upgradeIfBetter(..., bypassPriceGuard: true) and then pinned as the first rendered card. If no ranked fare can be represented by a cache row, the live response returns no selected fare instead of rendering a card admin cannot reconcile.

App\Services\Flights\Domain\OfferFlightSignature exposes the canonical identity used by the FE reconciler to know which option the backend is currently bound to:

  • fromCacheSegments(Collection $segments): string — per-leg OPERATOR+FLIGHTNUMBER (joined by + for connections) from cache segments.
  • fromFlightSegments(Collection $segments): string — same shape, built from Aerticket FlightSegment DTOs. Used by the live-search response renderer to emit a signature on every leg option.
  • compose(string $outbound, ?string $inbound): string — composes the round-trip option signature as outbound|inbound (or just outbound for one-way).
  • forBoundOffer(Offer $offer): string — returns the canonical outbound|inbound signature of the offer’s currently-bound international OfferFlight, built from the cache row’s primary itinerary per leg (not positional index 1). Returns '' when the offer has no resolvable international binding (draft, manual flight, missing cache row).

This identity is what the checkout response surfaces as bound_flight_signature (root) and per-option signature — see Checkout API → GET /flights.

Checkout display does not preserve a stale bound fare as “Seleccionado”. The selected card is the first live option after hard filters, baggage resolution, sorting, visual deduplication, and the top-10 cap. Economy display uses the same /search-only baggage rule as matched fare resolution and live-winner binding: native bag or in-response siblings only. If the old bound fare violates max stops or any other hard rule, it is removed from the displayed list instead of being pinned.

The international checkout live-search collapses fares that the customer would see as identical cards into a single option. Two fares share the visual signature when, for every segment of both the outbound and inbound primary itinerary, they share the same departure / arrival airports and the same wallclock departure / arrival times. The key is built from the segments actually rendered (the primary itinerary per leg picked by FlightRankingPolicy::primaryItineraryIndexForLeg).

This collapses:

  • Codeshares — e.g. an Iberia-ticketed row and a Qatar-ticketed row sitting on the same operating flight, which previously rendered as two indistinguishable cards.
  • Fare-family clones — Aerticket sometimes publishes the same physical flight under drifted flight numbers across families (BASIC / STANDARD / FLEX), which the bare-flight-number dedup let through.
  • Genuine accidental twins — two fares that publish the same journey at the same wallclock on the same airports.

Within a dedup group, the survivor is picked by pickGroupWinner:

  • Economy — the offer’s locked-in fare wins if it’s in the group (booking identity preserved so the booking-time fare_id fast-path in CheckoutFlightBookingService keeps working). Otherwise the cheapest wins, since the upstream compareFareResult sort places it first when group members share duration (which visual twins do).
  • Business$group->first() always; no matched-fare concept here. Dedup runs before take(10) so the cap returns ten distinct visual flights instead of being thinned later.
  • Domestic legs at checkout. searchDomesticFlightfindDomesticFareMatchingLeg signature-matches the offer’s locked leg and never deduplicates. The locked leg is extracted from the cache by its bound itinerary_index_outbound (the primary itinerary), not by leg_sequence alone — a route’s multiple cached direct alternatives each live under their own itinerary_index (see Dynamic Flight Cache — Segments Table), so pinning the index keeps the extracted leg a single flight (correct flight_numbers / stopover_airports / arrival_time) instead of welding all alternatives into a fake multi-stop connection. A genuine connection (multiple segment_number under one itinerary_index) is preserved. The per-leg flight_signature is built from those same pinned segments, so it identifies one flight and live re-pricing can actually match it. Past domestic-result-shrinkage incidents cannot recur from this change.
  • Offer generation / populator. DynamicFlightCachePopulatorService keeps positions 1–5 of distinct fare-families on purpose so checkout’s bag-resolution chain has in-response siblings when Aerticket publishes bag and no-bag fare families together.
  • Booking-time matcher. CheckoutFlightBookingService resolves identity via stored fare_id then signature fallback — visual dedup preserves the matched fare so this path is unchanged.

Source: EconomyFlightSearchService::getFareKey, EconomyFlightSearchService::pickGroupWinner, BusinessFlightSearchService::getFareKey.

php artisan flights:rerank-dynamic-caches rewrites fare_position on every completed cache row using the current policy. It reports two new metrics so operators can see how aggressively the new rules trimmed the catalogue:

  • Excluded (>8h layover) — rows that failed the filter.
  • Pareto-dominated (skipped from ranking) — rows pruned because another row in the same (route, departure_date, return_date, CUG) bucket was both faster and cheaper.

Both are informational; the rows stay in the DB, they just don’t carry a fare_position.

Source: backend/app/Console/Commands/RerankDynamicFlightCaches.php