Как разбить результаты на страницы при фильтрации по левому столбцу таблицы соединений
Определим, например, эти сущности:
@Entity()
export class Author {
@OneToMany(() => Book, book => book.author)
books = new Collection<Book>(this);
}
@Entity()
export class Book {
@Property()
name: string;
@ManyToOne()
author: Author;
}
const authors = await authorRepository.find({ books: {name: 'Test'} }, {
limit: 10
});
Как видите, я хочу выбрать всех авторов, у которых есть книги с названием Test, но это сгенерирует следующий запрос:
select `e0`.* from `author` as `e0`
left join `book` as `e1` on `e0`.`id` = `e1`.`author_id`
where `e1`.`name` = 'Test' limit 10
Проблема в том, что когда у меня более 2 авторов, и у каждой из них более 10 книг с именем "Тест", этот запрос вернет только первого автора из-за ограничения.
Я не уверен, что это ошибка ORM или это ожидаемое поведение.
Один из способов исправить это - выбрать все строки без предложения limit и выполнить разбиение на страницы в памяти, как это делается в спящем режиме, но я не уверен, сколько памяти будет использоваться с очень большими таблицами, и это может заблокировать цикл событий в NodeJS при их обработке.
1 ответ
Вы можете вернуться к построителю запросов здесь, чтобы применить предложение group by:
const qb = await authorRepository.createQueryBuilder('a');
qb.select('a.*').where({ books: { name: 'Test' } }).limit(10).groupBy('a.id');
const authors = await qb.getResult();
Подумайте, как поддерживать этот вариант использования напрямую через EM/ API репозитория.