2021-08-19 11:30:06 +00:00
|
|
|
import { Repository, FindConditions, SelectQueryBuilder, ObjectLiteral, ILike } from 'typeorm'
|
2020-06-26 21:25:03 +00:00
|
|
|
import { PaginateQuery } from './decorator'
|
|
|
|
import { ServiceUnavailableException } from '@nestjs/common'
|
|
|
|
|
|
|
|
type Column<T> = Extract<keyof T, string>
|
|
|
|
type Order<T> = [Column<T>, 'ASC' | 'DESC']
|
|
|
|
type SortBy<T> = Order<T>[]
|
|
|
|
|
|
|
|
export class Paginated<T> {
|
|
|
|
data: T[]
|
|
|
|
meta: {
|
|
|
|
itemsPerPage: number
|
|
|
|
totalItems: number
|
|
|
|
currentPage: number
|
|
|
|
totalPages: number
|
|
|
|
sortBy: SortBy<T>
|
2020-06-28 17:34:00 +00:00
|
|
|
search: string
|
2020-06-26 21:25:03 +00:00
|
|
|
}
|
|
|
|
links: {
|
|
|
|
first?: string
|
|
|
|
previous?: string
|
|
|
|
current: string
|
|
|
|
next?: string
|
|
|
|
last?: string
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface PaginateConfig<T> {
|
|
|
|
sortableColumns: Column<T>[]
|
2020-06-28 17:34:00 +00:00
|
|
|
searchableColumns?: Column<T>[]
|
2020-06-26 21:25:03 +00:00
|
|
|
maxLimit?: number
|
|
|
|
defaultSortBy?: SortBy<T>
|
|
|
|
defaultLimit?: number
|
|
|
|
where?: FindConditions<T>
|
|
|
|
queryBuilder?: SelectQueryBuilder<T>
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function paginate<T>(
|
|
|
|
query: PaginateQuery,
|
|
|
|
repo: Repository<T> | SelectQueryBuilder<T>,
|
|
|
|
config: PaginateConfig<T>
|
|
|
|
): Promise<Paginated<T>> {
|
|
|
|
let page = query.page || 1
|
2021-08-19 11:30:06 +00:00
|
|
|
const limit = Math.min(query.limit || config.defaultLimit || 20, config.maxLimit || 100)
|
2020-06-28 17:34:00 +00:00
|
|
|
const sortBy = [] as SortBy<T>
|
|
|
|
const search = query.search
|
2020-06-26 21:25:03 +00:00
|
|
|
const path = query.path
|
|
|
|
|
|
|
|
function isEntityKey(sortableColumns: Column<T>[], column: string): column is Column<T> {
|
|
|
|
return !!sortableColumns.find((c) => c === column)
|
|
|
|
}
|
|
|
|
|
|
|
|
const { sortableColumns } = config
|
|
|
|
if (config.sortableColumns.length < 1) throw new ServiceUnavailableException()
|
|
|
|
|
|
|
|
if (query.sortBy) {
|
|
|
|
for (const order of query.sortBy) {
|
|
|
|
if (isEntityKey(sortableColumns, order[0]) && ['ASC', 'DESC'].includes(order[1])) {
|
|
|
|
sortBy.push(order as Order<T>)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!sortBy.length) {
|
2020-06-28 17:34:00 +00:00
|
|
|
sortBy.push(...(config.defaultSortBy || [[sortableColumns[0], 'ASC']]))
|
2020-06-26 21:25:03 +00:00
|
|
|
}
|
|
|
|
|
2020-06-28 17:34:00 +00:00
|
|
|
if (page < 1) page = 1
|
|
|
|
|
2020-06-26 21:25:03 +00:00
|
|
|
let [items, totalItems]: [T[], number] = [[], 0]
|
|
|
|
|
2020-06-28 17:34:00 +00:00
|
|
|
let queryBuilder: SelectQueryBuilder<T>
|
|
|
|
|
2020-06-26 21:25:03 +00:00
|
|
|
if (repo instanceof Repository) {
|
2020-06-28 17:34:00 +00:00
|
|
|
queryBuilder = repo
|
2020-06-26 21:25:03 +00:00
|
|
|
.createQueryBuilder('e')
|
|
|
|
.take(limit)
|
|
|
|
.skip((page - 1) * limit)
|
|
|
|
|
|
|
|
for (const order of sortBy) {
|
2020-06-28 17:34:00 +00:00
|
|
|
queryBuilder.addOrderBy('e.' + order[0], order[1])
|
2020-06-26 21:25:03 +00:00
|
|
|
}
|
|
|
|
} else {
|
2020-06-28 17:34:00 +00:00
|
|
|
queryBuilder = repo.take(limit).skip((page - 1) * limit)
|
2020-06-26 21:25:03 +00:00
|
|
|
|
|
|
|
for (const order of sortBy) {
|
2020-06-28 17:34:00 +00:00
|
|
|
queryBuilder.addOrderBy(repo.alias + '.' + order[0], order[1])
|
2020-06-26 21:25:03 +00:00
|
|
|
}
|
2020-06-28 17:34:00 +00:00
|
|
|
}
|
2020-06-26 21:25:03 +00:00
|
|
|
|
2020-06-28 17:34:00 +00:00
|
|
|
const where: ObjectLiteral[] = []
|
|
|
|
if (search && config.searchableColumns) {
|
|
|
|
for (const column of config.searchableColumns) {
|
2021-08-19 11:30:06 +00:00
|
|
|
where.push({ [column]: ILike(`%${search}%`), ...config.where })
|
2020-06-28 17:34:00 +00:00
|
|
|
}
|
2020-06-26 21:25:03 +00:00
|
|
|
}
|
|
|
|
|
2020-06-28 17:34:00 +00:00
|
|
|
;[items, totalItems] = await queryBuilder.where(where.length ? where : config.where || {}).getManyAndCount()
|
|
|
|
|
2020-06-26 21:25:03 +00:00
|
|
|
let totalPages = totalItems / limit
|
|
|
|
if (totalItems % limit) totalPages = Math.ceil(totalPages)
|
|
|
|
|
2020-06-28 17:34:00 +00:00
|
|
|
const options = `&limit=${limit}${sortBy.map((order) => `&sortBy=${order.join(':')}`).join('')}${
|
|
|
|
search ? `&search=${search}` : ''
|
|
|
|
}`
|
2020-06-26 21:25:03 +00:00
|
|
|
|
|
|
|
const buildLink = (p: number): string => path + '?page=' + p + options
|
|
|
|
|
|
|
|
const results: Paginated<T> = {
|
|
|
|
data: items,
|
|
|
|
meta: {
|
|
|
|
itemsPerPage: limit,
|
|
|
|
totalItems,
|
|
|
|
currentPage: page,
|
|
|
|
totalPages: totalPages,
|
|
|
|
sortBy,
|
2020-06-28 17:34:00 +00:00
|
|
|
search,
|
2020-06-26 21:25:03 +00:00
|
|
|
},
|
|
|
|
links: {
|
|
|
|
first: page == 1 ? undefined : buildLink(1),
|
|
|
|
previous: page - 1 < 1 ? undefined : buildLink(page - 1),
|
|
|
|
current: buildLink(page),
|
|
|
|
next: page + 1 > totalPages ? undefined : buildLink(page + 1),
|
|
|
|
last: page == totalPages ? undefined : buildLink(totalPages),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
return Object.assign(new Paginated<T>(), results)
|
|
|
|
}
|