shineforge/compass

A modern, robust, and PSR-7 compatible URL manipulation library for PHP.

1.0.0 2025-07-21 16:04 UTC

This package is auto-updated.

Last update: 2025-07-21 16:05:06 UTC


README

License Latest Version PHP Version Main Status Release Status Develop Status

Description

Compass provides a powerful URL class that simplifies parsing, building, and manipulating URLs in your PHP applications. It offers an intuitive, immutable API for handling all parts of a URL, from the scheme to the fragment.

Key features include:

  • PSR-7 UriInterface Compatibility: Use it as a drop-in replacement wherever a PSR-7 URI is needed.
  • Immutable API: All modification methods (with...) return a new instance, ensuring predictable and safe state management.
  • Advanced Path Resolution: Easily convert between absolute and relative URLs, and canonicalize paths (resolving . and .. segments).
  • Robust Parsing: Handles complex URLs gracefully.

Installation

Install the library using Composer:

composer require shineforge/compass

Usage

Creating a URL

You can create a URL instance from a string.

<?php

require 'vendor/autoload.php';

use Compass\URL;

// Create via constructor
$url = new URL('https://user:pass@example.com:8080/path/to/file?query=string#fragment');

// Or using the static factory method
$url = URL::create('https://example.com/some/path');

If the provided string is an invalid URL, an InvalidArgumentException will be thrown.

Accessing URL Components

The URL class provides getter methods for all parts of a URL, compliant with Psr\Http\Message\UriInterface.

<?php

$url = new Compass\URL('https://user:pass@example.com:8080/path/to/file?query=string#fragment');

echo $url->getScheme();   // "https"
echo $url->getAuthority(); // "user:pass@example.com:8080"
echo $url->getUserInfo();  // "user:pass"
echo $url->getHost();      // "example.com"
echo $url->getPort();      // 8080
echo $url->getPath();      // "/path/to/file"
echo $url->getQuery();     // "query=string"
echo $url->getFragment();  // "fragment"

Modifying a URL (Immutability)

The URL object is immutable. All methods that modify the URL, such as withPath() or withScheme(), return a new URL instance with the change, leaving the original object untouched.

<?php

$url = Compass\URL::create('https://example.com/path');

$newUrl = $url->withScheme('http')->withPort(8080);

echo $url;     // "https://example.com/path"
echo $newUrl;  // "http://example.com:8080/path"

Absolute vs. Relative URLs

You can easily check if a URL is absolute or relative.

<?php

$absolute = new Compass\URL('https://example.com/path');
$relative = new Compass\URL('/path/only');

var_dump($absolute->isAbsolute()); // bool(true)
var_dump($absolute->isRelative()); // bool(false)

var_dump($relative->isAbsolute()); // bool(false)
var_dump($relative->isRelative()); // bool(true)

Path Resolution

Compass excels at handling complex path resolution tasks.

Making a URL Absolute

You can resolve a relative URL against a base URL to make it absolute.

<?php

$base = Compass\URL::create('https://example.com/a/b/c');
$relativeUrl = Compass\URL::create('../../d/e');

$absoluteUrl = $relativeUrl->makeAbsolute($base);

echo $absoluteUrl; // "https://example.com/a/d/e"

Making a URL Relative

You can also compute a relative path from one absolute URL to another.

<?php

$base = Compass\URL::create('https://example.com/a/b/c');
$target = Compass\URL::create('https://example.com/a/d/e');

$relativeUrl = $target->makeRelative($base);

echo $relativeUrl; // "../d/e"

You can also make a URL root-relative by passing null as the base. This will strip the scheme and authority, leaving a path that is absolute from the root of a domain.

<?php

$url = Compass\URL::create('https://example.com/some/path');
$rootRelative = $url->makeRelative(null);

echo $rootRelative; // "/some/path"

Converting to a String

The URL object can be easily cast to a string, which will return the full, canonicalized URL.

<?php

$url = Compass\URL::create('https://example.com/a/b/../c/');

echo $url; // "https://example.com/a/c/"