Extbase repository
All Extbase repositories inherit from
        \TYPO3\.
A repository is always responsible for precisely one type of domain object (model).
The naming of the repositories is important:
If the domain object is, for example, Blog (with full name
        \Friends),
then the corresponding repository is named BlogRepository (with the full name
        \Friends).
The 
        \TYPO3\ already offers a large
number of useful methods. Therefore it is sufficient that a class only extends the
        Repository class.
The 
        Blog sets the default orderings and is otherwise empty:
class BlogRepository extends Repository
{
}Table of contents
See also
If you are using the friendsoftypo3/kickstarter you can use the command vendor/bin/typo3 make:repository to generate an Extbase repository.
Find methods
Changed in version 14.0
The "magic" find methods 
        find, 
        find and 
        count have
been removed. See Migration
The (not-magic) methods 
        find and 
        find have not
been deprecated or removed, and are still valid.
Using these methods will fetch a given domain object by it's UID, ignoring possible storage
page settings - unlike 
        find, which does respect those settings.
The 
        Repository class provides the following methods for querying against
arbitrary criteria:
- find- By - (array $criteria, array $order - By = null, int $limit = null, int $offset = null): Query - Result - Interface 
- Finds all objects with the provided criteria.
- find- One - By - (array $criteria, array $order - By = null): object - |null 
- Returns the first object found with the provided criteria.
- count- (array $criteria): int 
- Counts all objects with the provided criteria.
Example:
$this->blogRepository->findBy(['author' => 1, 'published' => true]);Attention
Saving a 
        Query to a cache is not possible, if objects in the
        Query contain closures. This is typically the case for models
which use lazy loading on properties.
Custom find methods
Custom find methods can be implemented. They can be used for complex queries.
Attention
As Extbase repositories turn the results into objects, querying large amounts of data is resource-intensive.
Example:
The 
        Post of the 
    t3docs/blog-example
 example extension implements
several custom find methods, two of them are shown below:
use T3docs\BlogExample\Domain\Model\Blog;
use TYPO3\CMS\Extbase\Persistence\QueryInterface;
use TYPO3\CMS\Extbase\Persistence\QueryResultInterface;
class PostRepository extends Repository
{
    public function findByTagAndBlog(
        string $tag,
        Blog $blog,
    ): QueryResultInterface {
        $query = $this->createQuery();
        return $query
            ->matching(
                $query->logicalAnd(
                    $query->equals('blog', $blog),
                    $query->equals('tags.name', $tag),
                ),
            )
            ->execute();
    }
    public function findAllSortedByCategory(array $uids): QueryResultInterface
    {
        $q = $this->createQuery();
        $q->matching($q->in('uid', $uids));
        $q->setOrderings([
            'categories.title' => QueryInterface::ORDER_ASCENDING,
            'uid' => QueryInterface::ORDER_ASCENDING,
        ]);
        return $q->execute();
    }
}Query settings
If the query settings should be used for all methods in the repository,
they should be set in the method 
        initialize method.
class CommentRepository extends Repository
{
    public function initializeObject(): void
    {
        $querySettings = $this->createQuery()->getQuerySettings();
        // Show comments from all pages
        $querySettings->setRespectStoragePage(false);
        $this->setDefaultQuerySettings($querySettings);
    }
}Attention
Depending on the query settings, hidden or even deleted objects can become visible. This might cause sensitive information to be disclosed. Use with care.
If you only want to change the query settings for a specific method, they can be set in the method itself:
use TYPO3\CMS\Extbase\Persistence\QueryResultInterface;
class CommentRepository extends Repository
{
    public function findAllIgnoreEnableFields(): QueryResultInterface
    {
        $query = $this->createQuery();
        $query->getQuerySettings()->setIgnoreEnableFields(true);
        return $query->execute();
    }
}Repository API
- class Repository
- 
                    - Fully qualified name
- 
                        \TYPO3\CMS\ Extbase\ Persistence\ Repository 
 The base repository - will usually be extended by a more concrete repository. - injectPersistenceManager ( \TYPO3\CMS\Extbase\Persistence\PersistenceManagerInterface $persistenceManager)
- 
        - param $persistenceManager
- 
            
    the persistenceManager 
 
 - injectEventDispatcher ( \Psr\EventDispatcher\EventDispatcherInterface $eventDispatcher)
- 
        - param $eventDispatcher
- 
            
    the eventDispatcher 
 
 - remove ( ?object $object)
- 
        
    Removes an object from this repository. - param $object
- 
            
    The object to remove 
 
 - update ( ?object $modifiedObject)
- 
        
    Replaces an existing object with the same identifier by the given object - param $modifiedObject
- 
            
    The modified object 
 
 - countAll ( )
- 
        
    Returns the total number objects of this repository. - Return description
- 
            
    The object count 
 - Returns
- 
                    int
 
 - findByUid ( ?int $uid)
- 
        
    Finds an object matching the given identifier. - param $uid
- 
            
    The identifier of the object to find 
- Return description
- 
            
    The matching object if found, otherwise NULL 
 - Returns
- 
                    object|null 
 
 - findByIdentifier ( ?mixed $identifier)
- 
        
    Finds an object matching the given identifier. - param $identifier
- 
            
    The identifier of the object to find 
- Return description
- 
            
    The matching object if found, otherwise NULL 
 - Returns
- 
                    object|null 
 
 - setDefaultOrderings ( array $defaultOrderings)
- 
        
    Sets the property names to order the result by per default. Expected like this: array( 'foo' => TYPO3CMSExtbasePersistenceQueryInterface::ORDER_ASCENDING, 'bar' => TYPO3CMSExtbasePersistenceQueryInterface::ORDER_DESCENDING ) - param $defaultOrderings
- 
            
    The property names to order by 
 
 - setDefaultQuerySettings ( \TYPO3\CMS\Extbase\Persistence\Generic\QuerySettingsInterface $defaultQuerySettings)
- 
        
    Sets the default query settings to be used in this repository. A typical use case is an initializeObject() method that creates a QuerySettingsInterface object, configures it and sets it to be used for all queries created by the repository. Warning: Using this setter fully overrides native query settings created by QueryFactory->create(). This especially means that storagePid settings from configuration are not applied anymore, if not explicitly set. Make sure to apply these to your own QuerySettingsInterface object if needed, when using this method. - param $defaultQuerySettings
- 
            
    the defaultQuerySettings 
 
 - findBy ( array $criteria, ?array $orderBy = NULL, ?int $limit = NULL, ?int $offset = NULL)
- 
        - param $criteria
- 
            
    the criteria 
- param $orderBy
- 
            
    the orderBy, default: NULL 
- param $limit
- 
            
    the limit, default: NULL 
- param $offset
- 
            
    the offset, default: NULL 
 - Returns
- 
                    \QueryResult Interface 
 
 
Typo3QuerySettings and localization
Extbase renders the translated records in the same way as TypoScript rendering.
The following methods can be used to set and get the language aspect  from any
        \TYPO3\:
- Query- Settings - Interface:: - get - Language - Aspect - (): Language - Aspect 
- Query- Settings - Interface:: - set - Language - Aspect - (Language - Aspect $aspect) 
You can specify a custom language aspect per query as defined in the query settings in any repository class:
Example to use the fallback to the default language when working with overlays:
<?php
declare(strict_types=1);
namespace MyVendor\MyExtension\Domain\Repository;
use TYPO3\CMS\Core\Context\LanguageAspect;
use TYPO3\CMS\Extbase\Persistence\Repository;
final class MyRepository extends Repository
{
    public function findSomethingByLanguage(int $languageId, int $contentId)
    {
        $query = $this->createQuery();
        $query->getQuerySettings()->setLanguageAspect(
            new LanguageAspect(
                $languageId,
                $contentId,
                LanguageAspect::OVERLAYS_MIXED,
            ),
        );
        // query something
    }
}
Debugging an Extbase query
When using complex queries in Extbase repositories it sometimes comes handy to debug them using the Extbase debug utilities.
<?php
declare(strict_types=1);
namespace MyVendor\MyExtension\Domain\Repository;
use TYPO3\CMS\Core\Utility\GeneralUtility;
use TYPO3\CMS\Extbase\Persistence\Generic\Storage\Typo3DbQueryParser;
use TYPO3\CMS\Extbase\Persistence\QueryResultInterface;
use TYPO3\CMS\Extbase\Persistence\Repository;
use TYPO3\CMS\Extbase\Utility\DebuggerUtility;
final class MyRepository extends Repository
{
    public function findBySomething(string $something, bool $debugOn = false): QueryResultInterface
    {
        $query = $this->createQuery();
        $query = $query->matching($query->equals('some_field', $something));
        if ($debugOn) {
            $typo3DbQueryParser = GeneralUtility::makeInstance(Typo3DbQueryParser::class);
            $queryBuilder = $typo3DbQueryParser->convertQueryToDoctrineQueryBuilder($query);
            DebuggerUtility::var_dump($queryBuilder->getSQL());
            DebuggerUtility::var_dump($queryBuilder->getParameters());
        }
        return $query->execute();
    }
}
Please note that 
        \TYPO3\
is marked as @internal and subject to unannounced changes.