cheesytech/booking

A flexible and powerful booking system for Laravel applications

Installs: 1

Dependents: 0

Suggesters: 0

Security: 0

Stars: 3

Watchers: 0

Forks: 0

Open Issues: 0

pkg:composer/cheesytech/booking

v1.0.0 2025-06-18 06:37 UTC

This package is auto-updated.

Last update: 2025-12-14 09:27:19 UTC


README

A flexible and powerful booking system for Laravel applications that supports polymorphic relationships, time slot management, and advanced status tracking.

Features

  • ๐Ÿ•’ Time slot management with overlap prevention
  • ๐Ÿ”„ Polymorphic relationships for bookable and bookerable resources
  • โš™๏ธ Configurable validation and overlap rules
  • ๐ŸŽฏ Custom booking rules via OverlapRule interface
  • ๐Ÿ“… Business hours and custom rules validation
  • ๐Ÿ”’ Booking duration and minimum interval limits
  • ๐ŸŽจ Easy to extend with traits and interfaces
  • ๐Ÿ“Š Status tracking with full history and metadata
  • ๐Ÿ”„ Event-driven architecture (BookingCreated, BookingUpdated, etc.)
  • ๐Ÿงช Comprehensive testing support and model factories
  • ๐Ÿ” Advanced querying and duration-based scopes (cross-DB)
  • ๐Ÿ’พ Multi-database support (MySQL, PostgreSQL, SQLite, SQL Server)
  • โฑ๏ธ Flexible duration calculations (minutes, hours, days)
  • ๐Ÿ› ๏ธ Publishable config and migration files

Requirements

  • PHP 8.1 or higher
  • Laravel 10.0, 11.0, or 12.0
  • Carbon 2.0 or higher

Installation

  1. Install the package via composer:
composer require cheesytech/booking
  1. The package will automatically register its service provider.

  2. Publish and run the migrations:

php artisan vendor:publish --provider="CheeasyTech\Booking\BookingServiceProvider" --tag="migrations"
php artisan migrate
  1. Publish the configuration file:
php artisan vendor:publish --provider="CheeasyTech\Booking\BookingServiceProvider" --tag="config"

Or use the install command for all at once:

php artisan package:install cheesytech/booking

This will create a config/booking.php file in your config directory.

Configuration

The package is highly configurable through the config/booking.php file. Example:

return [
    'statuses' => [
        'pending' => [
            'label' => 'Pending',
            'color' => '#FFA500',
            'can_transition_to' => ['confirmed', 'cancelled'],
        ],
        'confirmed' => [
            'label' => 'Confirmed',
            'color' => '#008000',
            'can_transition_to' => ['cancelled', 'completed'],
        ],
        'cancelled' => [
            'label' => 'Cancelled',
            'color' => '#FF0000',
            'can_transition_to' => [],
        ],
        'completed' => [
            'label' => 'Completed',
            'color' => '#0000FF',
            'can_transition_to' => [],
        ],
    ],
    'overlap' => [
        'enabled' => true,
        'allow_same_booker' => false,
        'min_time_between' => 0,
        'max_duration' => 0,
        'rules' => [
            'business_hours' => [
                'enabled' => false,
                'class' => \CheeasyTech\Booking\Rules\BusinessHoursRule::class,
            ],
        ],
    ],
    'events' => [
        'enabled' => true,
        'classes' => [
            'created' => \CheeasyTech\Booking\Events\BookingCreated::class,
            'updated' => \CheeasyTech\Booking\Events\BookingUpdated::class,
            'deleted' => \CheeasyTech\Booking\Events\BookingDeleted::class,
            'status_changed' => \CheeasyTech\Booking\Events\BookingStatusChanged::class,
        ],
    ],
];

Model Setup

Implement the provided interfaces and use the traits for your models:

use CheeasyTech\Booking\Contracts\Bookable;
use CheeasyTech\Booking\Traits\HasBookings;
use Illuminate\Database\Eloquent\Model;

class Room extends Model implements Bookable {
    use HasBookings;
    // ...
    public function getBookableId(): int { return $this->id; }
    public function getBookableType(): string { return static::class; }
}

use CheeasyTech\Booking\Contracts\Bookerable;
use CheeasyTech\Booking\Traits\HasBookers;

class User extends Model implements Bookerable {
    use HasBookers;
    // ...
    public function getBookerableId(): int|string { return $this->id; }
    public function getBookerableType(): string { return static::class; }
}

Quick Start

use CheeasyTech\Booking\Models\Booking;
use Carbon\Carbon;

$room = Room::find(1);
$user = User::find(1);

$booking = new Booking();
$booking->bookable()->associate($room);
$booking->bookerable()->associate($user);
$booking->start_time = Carbon::tomorrow()->setHour(10);
$booking->end_time = Carbon::tomorrow()->setHour(11);
$booking->status = 'pending';
$booking->save();

// Change booking status
$booking->changeStatus('confirmed', 'Approved by admin', ['key' => 'value']);

// Check for overlap
$isAvailable = !$booking->hasOverlap(
    Carbon::tomorrow()->setHour(10),
    Carbon::tomorrow()->setHour(11)
);

Traits & Interfaces

  • HasBookings: Add to bookable models (e.g., Room) for convenient booking management (newBooking, deleteBooking, etc.).
  • HasBookers: Add to bookerable models (e.g., User) for managing bookings made by the entity.
  • Bookable: Interface for resources to be booked (must implement getBookableId, getBookableType).
  • Bookerable: Interface for entities making bookings (must implement getBookerableId, getBookerableType).
  • OverlapRule: Interface for custom overlap validation rules.

Database Schema

The package creates the following database table:

Schema::create('bookings', function (Blueprint $table) {
    $table->id();
    $table->morphs('bookable');
    $table->morphs('bookerable');
    $table->dateTime('start_time');
    $table->dateTime('end_time');
    $table->string('status')->default('pending');
    $table->json('status_history')->nullable();
    $table->timestamp('status_changed_at')->nullable();
    $table->timestamps();
});

Advanced Usage

Query Scopes

// Get bookings longer than 2 hours
Booking::durationLongerThan(120)->get();
// Get bookings between 1 and 2 hours
Booking::durationBetween(60, 120)->get();
// Combine with other conditions
Booking::durationLongerThan(120)->where('status', 'confirmed')->get();

Status Management

$booking->changeStatus('confirmed', 'Approved by supervisor', ['approver_id' => 123]);
$status = $booking->getCurrentStatus();
$history = $booking->getStatusHistory();
if ($booking->hasStatus('confirmed')) { /* ... */ }

Overlap & Custom Rules

  • Prevents overlapping bookings by default.
  • Supports minimum interval and max duration.
  • Add custom rules by implementing OverlapRule and registering in config.
  • Example: BusinessHoursRule restricts bookings to business hours.

Events

Events are fired for all major actions:

  • BookingCreated
  • BookingUpdated
  • BookingDeleted
  • BookingStatusChanged

Testing & Factories

Use provided factories for testing:

$booking = Booking::factory()->pending()->create();
$room = Room::factory()->create();
$user = User::factory()->create();

Updating PHPDoc

To update PHPDoc for models, run:

./update-phpdoc.sh

Contributing

Thank you for considering contributing to the Laravel Booking package! Please feel free to submit pull requests or create issues for bugs and feature requests.

License

The Laravel Booking package is open-sourced software licensed under the MIT license.