in-square/opendxp-sitemap-bundle

Static XML sitemap generator for OpenDXP (Documents + DataObjects).

Maintainers

Package info

github.com/in-square/opendxp-sitemap-bundle

Type:opendxp-bundle

pkg:composer/in-square/opendxp-sitemap-bundle

Statistics

Installs: 15

Dependents: 0

Suggesters: 0

Stars: 0

Open Issues: 0

v1.1.2 2026-05-12 19:21 UTC

This package is not auto-updated.

Last update: 2026-05-12 19:22:48 UTC


README

Static XML sitemap generator for OpenDXP (Documents + DataObjects) with multi-site and multi-locale support.

Requirements

  • PHP 8.3+
  • OpenDXP 1.x / Symfony 7.4
  • Symfony Messenger (for queue processing)
  • Elements Process Manager (commands integrate with Process Manager)

Installation

composer require in-square/opendxp-sitemap-bundle

Enable the bundle in config/bundles.php:

return [
    InSquare\OpendxpSitemapBundle\InSquareOpendxpSitemapBundle::class => ['all' => true],
];

Configuration

Create config/packages/in_square_opendxp_sitemap.yaml:

in_square_opendxp_sitemap:
  sites:
    - id: 0
      host: 'example.com'
      languages: ['pl']
      objects:
        - 'OpenDXP\Model\DataObject\Post'
        - 'OpenDXP\Model\DataObject\PostCategory'

    - id: 1
      host: 'example.org'
      languages: ['pl', 'en']
      objects:
        - 'OpenDXP\Model\DataObject\Product'

  object_generators:
    post: 'App\\Sitemap\\PostGenerator'
    postCategory: 'App\\Sitemap\\PostCategoryGenerator'
    product: 'App\\Sitemap\\ProductGenerator'

  output:
    dir: '%kernel.project_dir%/public/sitemap'
    max_urls_per_file: 50000

  hreflang:
    enabled: true
    x_default_language: 'en' # optional
    x_default_fallback_language: 'pl' # optional fallback used only when x_default_language is missing

Notes:

  • sites[*].objects defines DataObject classes to collect for each site.
  • object_generators keys must match ObjectGeneratorInterface::getId(); keys are used in sitemap filenames.
  • hreflang.enabled controls generation of <xhtml:link rel="alternate" ... /> entries.
  • hreflang.x_default_language selects preferred locale for x-default.
  • hreflang.x_default_fallback_language is an explicit fallback locale if preferred one is missing; without it, x-default is omitted.

Commands

  • bin/console insquare:sitemap:install – create sitemap_item table.
  • bin/console insquare:sitemap:collect – dispatch sitemap messages to Messenger.
  • bin/console insquare:sitemap:dump – generate XML files from database.
  • bin/console insquare:sitemap:delete – delete XML files and truncate the table.

Add routing for Messenger in config/packages/framework.yaml:

framework:
  messenger:
    routing:
      'InSquare\OpendxpSitemapBundle\Message\SitemapItemCreateMessage': async

Run Messenger worker for the queue:

bin/console messenger:consume async

Controller

The bundle exposes /sitemap.xml. The controller selects the correct site and serves the pre-generated file from public/sitemap/sitemap.{siteId}.xml.

Object generators

Implement InSquare\OpendxpSitemapBundle\Generator\ObjectGeneratorWithContextInterface in your app to receive a normalized host from sitemap configuration:

<?php

declare(strict_types=1);

namespace App\Sitemap;

use InSquare\OpendxpSitemapBundle\Generator\ObjectGeneratorWithContextInterface;
use InSquare\OpendxpSitemapBundle\Generator\SitemapItemData;
use InSquare\OpendxpSitemapBundle\Generator\SitemapGeneratorContext;
use OpenDXP\Model\DataObject\Post;

final class PostGenerator implements ObjectGeneratorWithContextInterface
{
    public function getId(): string
    {
        return 'post';
    }

    public function getObjectClass(): string
    {
        return Post::class;
    }

    public function buildItem(object $object, SitemapGeneratorContext $context): ?SitemapItemData
    {
        if (!$object instanceof Post) {
            return null;
        }

        if (!$object->isPublished()) {
            return null;
        }

        $linkGenerator = $object->getClass()?->getLinkGenerator();
        if ($linkGenerator === null) {
            return null;
        }

        $locale = $context->getLocale();
        $host = $context->getHost();
        $path = $linkGenerator->generate($object, [
            'locale' => $locale,
            'siteId' => $context->getSiteId(),
        ]);
        $url = $host . '/' . ltrim((string) $path, '/');

        $lastmod = (new \DateTimeImmutable())->setTimestamp($object->getModificationDate());

        return new SitemapItemData(
            $object->getId(),
            $object::class,
            $url,
            $lastmod
        );
    }
}

The generator returns a SitemapItemData DTO used to persist rows in sitemap_item.

Notes:

  • SitemapGeneratorContext::getHost() is normalized to an absolute host (for example https://example.com).
  • ObjectGeneratorInterface is deprecated and supported only for backward compatibility.