php-sitemap package
php-sitemap is a modern, framework-agnostic PHP package for generating sitemaps in XML, TXT, HTML, and Google News formats. It works seamlessly with Laravel, Symfony, or any PHP project. Features include high test coverage, robust CI, extensible adapters, and support for images, videos, translations, alternates, and Google News.
This is the PHP implementation of our multi-language sitemap library:
- π php-sitemap - PHP 8.2+ implementation with Laravel & Symfony support (this package)
- π npm-sitemap - TypeScript/JavaScript implementation for Node.js and frontend frameworks
- π· go-sitemap - Go implementation for high-performance applications
All implementations share the same API design and features, making it easy to switch between languages or maintain consistency across polyglot projects.
If you find php-sitemap useful, you might also be interested in these related projects:
- π php-seo - Comprehensive SEO toolkit for meta tags, structured data, and search optimization
- π€ php-chatbot - Conversational AI and chatbot framework for PHP applications
- π° php-feed - RSS, Atom, and JSON feed generator for content syndication
- π php-geolocation - IP geolocation, geocoding, and geographic data utilities
- Framework-agnostic: Use in Laravel, Symfony, or any PHP project
- Multiple formats: XML, TXT, HTML, Google News, mobile
- Rich content: Supports images, videos, translations, alternates, Google News
- Modern PHP: Type-safe, extensible, and robust (PHP 8.2+)
- High test coverage: 100% code coverage, CI/CD ready
- Easy integration: Simple API, drop-in for controllers/routes
- Extensible: Adapters for Laravel, Symfony, and more
- Quality tools: PHPStan Level max, PSR-12, comprehensive testing
- Input validation: Built-in URL, priority, and frequency validation
- Type-safe configuration: Fluent configuration with
SitemapConfigclass - Fluent interface: Method chaining for elegant, readable code
- π Installation
- π Usage Examples
- π§ͺ Testing & Development
- π€ Contributing
- π Security Policy
- π Support & Funding
- π License
- PHP 8.2+
- Composer
composer require rumenx/php-sitemapNo additional configuration required! The package works out of the box.
Controller method:
use Rumenx\Sitemap\Sitemap;
public function sitemap(Sitemap $sitemap)
{
$sitemap->add('https://example.com/', now(), '1.0', 'daily');
$sitemap->add('https://example.com/about', now(), '0.8', 'monthly', images: [
['url' => 'https://example.com/img/about.jpg', 'title' => 'About Us']
]);
// Add more items as needed...
// Render XML using a view template
$items = $sitemap->getModel()->getItems();
return response()->view('sitemap.xml', compact('items'), 200, ['Content-Type' => 'application/xml']);
}Route registration:
Route::get('/sitemap.xml', [SitemapController::class, 'sitemap']);Advanced:
// Add with translations, videos, alternates, Google News
$sitemap->add(
'https://example.com/news',
now(),
'0.7',
'weekly',
images: [['url' => 'https://example.com/img/news.jpg', 'title' => 'News Image']],
title: 'News Article',
translations: [['language' => 'fr', 'url' => 'https://example.com/fr/news']],
videos: [['title' => 'News Video', 'description' => 'Video description']],
googlenews: [
'sitename' => 'Example News',
'language' => 'en',
'publication_date' => now(),
],
alternates: [['media' => 'print', 'url' => 'https://example.com/news-print']]
);Controller:
use Rumenx\Sitemap\Sitemap;
use Symfony\Component\HttpFoundation\Response;
class SitemapController
{
public function sitemap(): Response
{
$sitemap = new Sitemap();
$sitemap->add('https://example.com/', (new \DateTime())->format(DATE_ATOM), '1.0', 'daily');
$sitemap->add('https://example.com/contact', (new \DateTime())->format(DATE_ATOM), '0.5', 'monthly');
// Add more items as needed...
// Render XML
$xml = $sitemap->renderXml();
return new Response($xml, 200, ['Content-Type' => 'application/xml']);
}
}Route registration:
# config/routes.yaml
sitemap:
path: /sitemap.xml
controller: App\Controller\SitemapController::sitemaprequire 'vendor/autoload.php';
use Rumenx\Sitemap\Sitemap;
$sitemap = new Sitemap();
$sitemap->add('https://example.com/', date('c'), '1.0', 'daily');
$sitemap->add(
'https://example.com/products',
date('c'),
'0.9',
'weekly',
images: [['url' => 'https://example.com/img/product.jpg', 'title' => 'Product Image']]
);
// Output XML
header('Content-Type: application/xml');
echo $sitemap->renderXml();// Add with all supported fields
$sitemap->add(
'https://example.com/news',
date('c'),
'0.8',
'daily',
images: [['url' => 'https://example.com/img/news.jpg', 'title' => 'News Image']],
title: 'News Article',
translations: [['language' => 'fr', 'url' => 'https://example.com/fr/news']],
videos: [['title' => 'News Video', 'description' => 'Video description']],
googlenews: [
'sitename' => 'Example News',
'language' => 'en',
'publication_date' => date('c'),
],
alternates: [['media' => 'print', 'url' => 'https://example.com/news-print']]
);
// Generate XML using renderXml() method
$xml = $sitemap->renderXml();
file_put_contents('sitemap.xml', $xml);
// Or use view templates for more control (create your own views based on src/views/)
$items = $sitemap->getModel()->getItems();
// Pass $items to your view templateYou can add sitemap entries using either the add() or addItem() methods:
add() β Simple, type-safe, one-at-a-time:
// Recommended for most use cases
$sitemap->add(
'https://example.com/',
date('c'),
'1.0',
'daily',
images: [['url' => 'https://example.com/img.jpg', 'title' => 'Image']],
title: 'Homepage'
);addItem() β Advanced, array-based, supports batch:
// Add a single item with an array (all fields as keys)
$sitemap->addItem([
'loc' => 'https://example.com/about',
'lastmod' => date('c'),
'priority' => '0.8',
'freq' => 'monthly',
'title' => 'About Us',
'images' => [['url' => 'https://example.com/img/about.jpg', 'title' => 'About Us']],
]);
// Add multiple items at once (batch add)
$sitemap->addItem([
[
'loc' => 'https://example.com/page1',
'title' => 'Page 1',
],
[
'loc' => 'https://example.com/page2',
'title' => 'Page 2',
],
]);- Use
add()for simple, explicit, one-at-a-time additions (recommended for most users). - Use
addItem()for advanced, batch, or programmatic additions with arrays (e.g., when looping over database results).
Chain methods for more elegant and readable code:
$sitemap = (new Sitemap())
->add('https://example.com/', date('c'), '1.0', 'daily')
->add('https://example.com/about', date('c'), '0.8', 'monthly')
->add('https://example.com/contact', date('c'), '0.6', 'yearly')
->store('xml', 'sitemap', './public');Configure sitemaps with a fluent, type-safe configuration class:
use Rumenx\Sitemap\Config\SitemapConfig;
$config = (new SitemapConfig())
->setEscaping(true)
->setStrictMode(true)
->setUseGzip(true)
->setDefaultFormat('xml');
$sitemap = new Sitemap($config);Enable strict mode to automatically validate all input:
$config = new SitemapConfig(strictMode: true);
$sitemap = new Sitemap($config);
// Valid data works fine
$sitemap->add('https://example.com', '2023-12-01', '0.8', 'daily');
// Invalid data throws InvalidArgumentException
try {
$sitemap->add('not-a-url', '2023-12-01', '2.0', 'sometimes');
} catch (\InvalidArgumentException $e) {
echo "Validation error: " . $e->getMessage();
}Render sitemaps in different formats:
$sitemap = new Sitemap();
$sitemap->add('https://example.com/', date('c'), '1.0', 'daily');
// Render as XML
$xml = $sitemap->render('xml');
// Render as HTML
$html = $sitemap->render('html');
// Render as plain text
$txt = $sitemap->render('txt');
// Save to file
$sitemap->store('xml', 'sitemap', './public');The package provides multiple ways to generate sitemap output:
$sitemap = new Sitemap();
$sitemap->add('https://example.com/', date('c'), '1.0', 'daily');
$xml = $sitemap->renderXml(); // Returns XML stringFor more control, use the included view templates or create your own:
$sitemap = new Sitemap();
$sitemap->add('https://example.com/', date('c'), '1.0', 'daily');
// Get the data for your view
$items = $sitemap->getModel()->getItems();
// Laravel: Use response()->view() or view()->render()
return response()->view('sitemap.xml', compact('items'), 200, ['Content-Type' => 'application/xml']);
// Symfony: Use Twig templates
return $this->render('sitemap.xml.twig', ['items' => $items], new Response('', 200, ['Content-Type' => 'application/xml']));
// Generic PHP: Include view templates
ob_start();
include 'vendor/rumenx/php-sitemap/src/views/xml.php';
$xml = ob_get_clean();Available view templates in src/views/:
xml.php- Standard XML sitemapxml-mobile.php- Mobile-specific sitemapgoogle-news.php- Google News sitemapsitemapindex.php- Sitemap indextxt.php- Plain text formathtml.php- HTML format
# Run all tests
composer test
# Run tests with text coverage report
composer coverage
# Generate full HTML coverage report
composer coverage-html# Run static analysis (PHPStan Level 6)
composer analyze
# Check coding standards (PSR-12)
composer style
# Auto-fix coding standards
composer style-fix# Run specific test file
./vendor/bin/pest tests/Unit/SitemapTest.php
# Run tests in watch mode
./vendor/bin/pest --watchWe welcome contributions! Please see our Contributing Guide for details on:
- Development setup
- Coding standards
- Testing requirements
- Pull request process
If you discover a security vulnerability, please review our Security Policy for responsible disclosure guidelines.
If you find this package helpful, consider:
- β Starring the repository
- π Supporting development
- π Reporting issues
- π€ Contributing improvements