| Package Data | |
|---|---|
| Maintainer Username: | RumenX |
| Maintainer Contact: | contact@rumenx.com (Rumen Damyanov) |
| Package Create Date: | 2013-05-30 |
| Package Last Update: | 2026-02-16 |
| Home Page: | |
| Language: | PHP |
| License: | MIT |
| Last Refreshed: | 2026-03-02 15:00:04 |
| Package Statistics | |
|---|---|
| Total Downloads: | 11,832 |
| Monthly Downloads: | 2,540 |
| Daily Downloads: | 177 |
| Total Stars: | 1,340 |
| Total Watchers: | 36 |
| Total Forks: | 274 |
| Total Open Issues: | 1 |
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:
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:
SitemapConfig classcomposer require rumenx/php-sitemap
No 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::sitemap
require '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 template
You 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',
],
]);
add() for simple, explicit, one-at-a-time additions (recommended for most users).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 string
For 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 --watch
We welcome contributions! Please see our Contributing Guide for details on:
If you discover a security vulnerability, please review our Security Policy for responsible disclosure guidelines.
If you find this package helpful, consider: