mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-07 14:39:15 +01:00
538 lines
21 KiB
JavaScript
538 lines
21 KiB
JavaScript
const express = require('express')
|
|
const Path = require('path')
|
|
const fs = require('fs-extra')
|
|
const date = require('date-and-time')
|
|
const axios = require('axios')
|
|
|
|
const Logger = require('../Logger')
|
|
const { isObject } = require('../utils/index')
|
|
const { parsePodcastRssFeedXml } = require('../utils/podcastUtils')
|
|
|
|
const LibraryController = require('../controllers/LibraryController')
|
|
const UserController = require('../controllers/UserController')
|
|
const CollectionController = require('../controllers/CollectionController')
|
|
const MeController = require('../controllers/MeController')
|
|
const BackupController = require('../controllers/BackupController')
|
|
const LibraryItemController = require('../controllers/LibraryItemController')
|
|
const SeriesController = require('../controllers/SeriesController')
|
|
const AuthorController = require('../controllers/AuthorController')
|
|
const MediaEntityController = require('../controllers/MediaEntityController')
|
|
const SessionController = require('../controllers/SessionController')
|
|
|
|
const BookFinder = require('../finders/BookFinder')
|
|
const AuthorFinder = require('../finders/AuthorFinder')
|
|
const PodcastFinder = require('../finders/PodcastFinder')
|
|
|
|
const Author = require('../objects/entities/Author')
|
|
const Series = require('../objects/entities/Series')
|
|
const FileSystemController = require('../controllers/FileSystemController')
|
|
|
|
class ApiRouter {
|
|
constructor(db, auth, scanner, playbackSessionManager, downloadManager, coverController, backupManager, watcher, cacheManager, emitter, clientEmitter) {
|
|
this.db = db
|
|
this.auth = auth
|
|
this.scanner = scanner
|
|
this.playbackSessionManager = playbackSessionManager
|
|
this.downloadManager = downloadManager
|
|
this.backupManager = backupManager
|
|
this.coverController = coverController
|
|
this.watcher = watcher
|
|
this.cacheManager = cacheManager
|
|
this.emitter = emitter
|
|
this.clientEmitter = clientEmitter
|
|
|
|
this.bookFinder = new BookFinder()
|
|
this.authorFinder = new AuthorFinder()
|
|
this.podcastFinder = new PodcastFinder()
|
|
|
|
this.router = express()
|
|
this.init()
|
|
}
|
|
|
|
init() {
|
|
//
|
|
// Library Routes
|
|
//
|
|
this.router.post('/libraries', LibraryController.create.bind(this))
|
|
this.router.get('/libraries', LibraryController.findAll.bind(this))
|
|
this.router.get('/libraries/:id', LibraryController.middleware.bind(this), LibraryController.findOne.bind(this))
|
|
this.router.patch('/libraries/:id', LibraryController.middleware.bind(this), LibraryController.update.bind(this))
|
|
this.router.delete('/libraries/:id', LibraryController.middleware.bind(this), LibraryController.delete.bind(this))
|
|
|
|
this.router.get('/libraries/:id/items', LibraryController.middleware.bind(this), LibraryController.getLibraryItems.bind(this))
|
|
this.router.get('/libraries/:id/series', LibraryController.middleware.bind(this), LibraryController.getAllSeriesForLibrary.bind(this))
|
|
this.router.get('/libraries/:id/series/:series', LibraryController.middleware.bind(this), LibraryController.getSeriesForLibrary.bind(this))
|
|
this.router.get('/libraries/:id/collections', LibraryController.middleware.bind(this), LibraryController.getCollectionsForLibrary.bind(this))
|
|
this.router.get('/libraries/:id/personalized', LibraryController.middleware.bind(this), LibraryController.getLibraryUserPersonalized.bind(this))
|
|
this.router.get('/libraries/:id/filterdata', LibraryController.middleware.bind(this), LibraryController.getLibraryFilterData.bind(this))
|
|
this.router.get('/libraries/:id/search', LibraryController.middleware.bind(this), LibraryController.search.bind(this))
|
|
this.router.get('/libraries/:id/stats', LibraryController.middleware.bind(this), LibraryController.stats.bind(this))
|
|
this.router.get('/libraries/:id/authors', LibraryController.middleware.bind(this), LibraryController.getAuthors.bind(this))
|
|
this.router.post('/libraries/:id/matchbooks', LibraryController.middleware.bind(this), LibraryController.matchBooks.bind(this))
|
|
|
|
this.router.post('/libraries/order', LibraryController.reorder.bind(this))
|
|
|
|
//
|
|
// Media Entity Routes
|
|
//
|
|
this.router.get('/entities/:id', MediaEntityController.middleware.bind(this), MediaEntityController.findOne.bind(this))
|
|
this.router.get('/entities/:id/item', MediaEntityController.middleware.bind(this), MediaEntityController.findWithItem.bind(this))
|
|
this.router.patch('/entities/:id/tracks', MediaEntityController.middleware.bind(this), MediaEntityController.updateTracks.bind(this))
|
|
this.router.post('/entities/:id/play', MediaEntityController.middleware.bind(this), MediaEntityController.startPlaybackSession.bind(this))
|
|
|
|
//
|
|
// Item Routes
|
|
//
|
|
this.router.delete('/items/all', LibraryItemController.deleteAll.bind(this))
|
|
|
|
this.router.get('/items/:id', LibraryItemController.middleware.bind(this), LibraryItemController.findOne.bind(this))
|
|
this.router.patch('/items/:id', LibraryItemController.middleware.bind(this), LibraryItemController.update.bind(this))
|
|
this.router.delete('/items/:id', LibraryItemController.middleware.bind(this), LibraryItemController.delete.bind(this))
|
|
this.router.patch('/items/:id/media', LibraryItemController.middleware.bind(this), LibraryItemController.updateMedia.bind(this))
|
|
this.router.get('/items/:id/cover', LibraryItemController.middleware.bind(this), LibraryItemController.getCover.bind(this))
|
|
this.router.post('/items/:id/cover', LibraryItemController.middleware.bind(this), LibraryItemController.uploadCover.bind(this))
|
|
this.router.patch('/items/:id/cover', LibraryItemController.middleware.bind(this), LibraryItemController.updateCover.bind(this))
|
|
this.router.delete('/items/:id/cover', LibraryItemController.middleware.bind(this), LibraryItemController.removeCover.bind(this))
|
|
this.router.post('/items/:id/match', LibraryItemController.middleware.bind(this), LibraryItemController.match.bind(this))
|
|
this.router.post('/items/:id/play', LibraryItemController.middleware.bind(this), LibraryItemController.startPlaybackSession.bind(this))
|
|
|
|
this.router.post('/items/batch/delete', LibraryItemController.batchDelete.bind(this))
|
|
this.router.post('/items/batch/update', LibraryItemController.batchUpdate.bind(this))
|
|
this.router.post('/items/batch/get', LibraryItemController.batchGet.bind(this))
|
|
|
|
//
|
|
// User Routes
|
|
//
|
|
this.router.post('/users', UserController.create.bind(this))
|
|
this.router.get('/users', UserController.findAll.bind(this))
|
|
this.router.get('/users/:id', UserController.findOne.bind(this))
|
|
this.router.patch('/users/:id', UserController.update.bind(this))
|
|
this.router.delete('/users/:id', UserController.delete.bind(this))
|
|
|
|
this.router.get('/users/:id/listening-sessions', UserController.getListeningStats.bind(this))
|
|
this.router.get('/users/:id/listening-stats', UserController.getListeningStats.bind(this))
|
|
|
|
//
|
|
// Collection Routes
|
|
//
|
|
this.router.post('/collections', CollectionController.create.bind(this))
|
|
this.router.get('/collections', CollectionController.findAll.bind(this))
|
|
this.router.get('/collections/:id', CollectionController.findOne.bind(this))
|
|
this.router.patch('/collections/:id', CollectionController.update.bind(this))
|
|
this.router.delete('/collections/:id', CollectionController.delete.bind(this))
|
|
|
|
this.router.post('/collections/:id/book', CollectionController.addBook.bind(this))
|
|
this.router.delete('/collections/:id/book/:bookId', CollectionController.removeBook.bind(this))
|
|
this.router.post('/collections/:id/batch/add', CollectionController.addBatch.bind(this))
|
|
this.router.post('/collections/:id/batch/remove', CollectionController.removeBatch.bind(this))
|
|
|
|
//
|
|
// Current User Routes (Me)
|
|
//
|
|
this.router.get('/me/listening-sessions', MeController.getListeningSessions.bind(this))
|
|
this.router.get('/me/listening-stats', MeController.getListeningStats.bind(this))
|
|
this.router.patch('/me/progress/:id', MeController.createUpdateLibraryItemProgress.bind(this))
|
|
this.router.delete('/me/progress/:id', MeController.removeLibraryItemProgress.bind(this))
|
|
this.router.patch('/me/progress/batch/update', MeController.batchUpdateLibraryItemProgress.bind(this))
|
|
this.router.post('/me/item/:id/bookmark', MeController.createBookmark.bind(this))
|
|
this.router.patch('/me/item/:id/bookmark', MeController.updateBookmark.bind(this))
|
|
this.router.delete('/me/item/:id/bookmark/:time', MeController.removeBookmark.bind(this))
|
|
this.router.patch('/me/password', MeController.updatePassword.bind(this))
|
|
this.router.patch('/me/settings', MeController.updateSettings.bind(this))
|
|
|
|
//
|
|
// Backup Routes
|
|
//
|
|
this.router.delete('/backup/:id', BackupController.delete.bind(this))
|
|
this.router.post('/backup/upload', BackupController.upload.bind(this))
|
|
|
|
//
|
|
// Search Routes
|
|
//
|
|
this.router.get('/search/covers', this.findCovers.bind(this))
|
|
this.router.get('/search/books', this.findBooks.bind(this))
|
|
this.router.get('/search/podcast', this.findPodcasts.bind(this))
|
|
this.router.get('/search/authors', this.findAuthor.bind(this))
|
|
|
|
//
|
|
// File System Routes
|
|
//
|
|
this.router.get('/filesystem', FileSystemController.getPaths.bind(this))
|
|
|
|
//
|
|
// Author Routes
|
|
//
|
|
this.router.get('/authors/search', AuthorController.search.bind(this))
|
|
this.router.get('/authors/:id', AuthorController.middleware.bind(this), AuthorController.findOne.bind(this))
|
|
this.router.patch('/authors/:id', AuthorController.middleware.bind(this), AuthorController.update.bind(this))
|
|
this.router.post('/authors/:id/match', AuthorController.middleware.bind(this), AuthorController.match.bind(this))
|
|
this.router.get('/authors/:id/image', AuthorController.middleware.bind(this), AuthorController.getImage.bind(this))
|
|
|
|
//
|
|
// Series Routes
|
|
//
|
|
this.router.get('/series/search', SeriesController.search.bind(this))
|
|
this.router.get('/series/:id', SeriesController.middleware.bind(this), SeriesController.findOne.bind(this))
|
|
|
|
//
|
|
// Playback Session Routes
|
|
//
|
|
this.router.post('/session/:id/sync', SessionController.middleware.bind(this), SessionController.sync.bind(this))
|
|
this.router.post('/session/:id/close', SessionController.middleware.bind(this), SessionController.close.bind(this))
|
|
|
|
//
|
|
// Misc Routes
|
|
//
|
|
this.router.patch('/serverSettings', this.updateServerSettings.bind(this))
|
|
|
|
this.router.post('/authorize', this.authorize.bind(this))
|
|
|
|
this.router.get('/download/:id', this.download.bind(this))
|
|
|
|
this.router.post('/purgecache', this.purgeCache.bind(this))
|
|
|
|
// OLD
|
|
// this.router.post('/syncUserAudiobookData', this.syncUserAudiobookData.bind(this))
|
|
|
|
this.router.post('/getPodcastFeed', this.getPodcastFeed.bind(this))
|
|
}
|
|
|
|
async findBooks(req, res) {
|
|
var provider = req.query.provider || 'google'
|
|
var title = req.query.title || ''
|
|
var author = req.query.author || ''
|
|
var results = await this.bookFinder.search(provider, title, author)
|
|
res.json(results)
|
|
}
|
|
|
|
async findCovers(req, res) {
|
|
var query = req.query
|
|
var result = await this.bookFinder.findCovers(query.provider, query.title, query.author || null)
|
|
res.json(result)
|
|
}
|
|
|
|
async findPodcasts(req, res) {
|
|
var term = req.query.term
|
|
var results = await this.podcastFinder.search(term)
|
|
res.json(results)
|
|
}
|
|
|
|
async findAuthor(req, res) {
|
|
var query = req.query.q
|
|
var author = await this.authorFinder.findAuthorByName(query)
|
|
res.json(author)
|
|
}
|
|
|
|
authorize(req, res) {
|
|
if (!req.user) {
|
|
Logger.error('Invalid user in authorize')
|
|
return res.sendStatus(401)
|
|
}
|
|
res.json({ user: req.user })
|
|
}
|
|
|
|
async updateServerSettings(req, res) {
|
|
if (!req.user.isRoot) {
|
|
Logger.error('User other than root attempting to update server settings', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var settingsUpdate = req.body
|
|
if (!settingsUpdate || !isObject(settingsUpdate)) {
|
|
return res.status(500).send('Invalid settings update object')
|
|
}
|
|
|
|
var madeUpdates = this.db.serverSettings.update(settingsUpdate)
|
|
if (madeUpdates) {
|
|
// If backup schedule is updated - update backup manager
|
|
if (settingsUpdate.backupSchedule !== undefined) {
|
|
this.backupManager.updateCronSchedule()
|
|
}
|
|
|
|
await this.db.updateServerSettings()
|
|
}
|
|
return res.json({
|
|
success: true,
|
|
serverSettings: this.db.serverSettings
|
|
})
|
|
}
|
|
|
|
async download(req, res) {
|
|
if (!req.user.canDownload) {
|
|
Logger.error('User attempting to download without permission', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var downloadId = req.params.id
|
|
Logger.info('Download Request', downloadId)
|
|
var download = this.downloadManager.getDownload(downloadId)
|
|
if (!download) {
|
|
Logger.error('Download request not found', downloadId)
|
|
return res.sendStatus(404)
|
|
}
|
|
|
|
var options = {
|
|
headers: {
|
|
'Content-Type': download.mimeType
|
|
}
|
|
}
|
|
res.download(download.fullPath, download.filename, options, (err) => {
|
|
if (err) {
|
|
Logger.error('Download Error', err)
|
|
}
|
|
})
|
|
}
|
|
|
|
async getDirectories(dir, relpath, excludedDirs, level = 0) {
|
|
try {
|
|
var paths = await fs.readdir(dir)
|
|
|
|
var dirs = await Promise.all(paths.map(async dirname => {
|
|
var fullPath = Path.join(dir, dirname)
|
|
var path = Path.join(relpath, dirname)
|
|
|
|
var isDir = (await fs.lstat(fullPath)).isDirectory()
|
|
if (isDir && !excludedDirs.includes(path) && dirname !== 'node_modules') {
|
|
return {
|
|
path,
|
|
dirname,
|
|
fullPath,
|
|
level,
|
|
dirs: level < 4 ? (await this.getDirectories(fullPath, path, excludedDirs, level + 1)) : []
|
|
}
|
|
} else {
|
|
return false
|
|
}
|
|
}))
|
|
dirs = dirs.filter(d => d)
|
|
return dirs
|
|
} catch (error) {
|
|
Logger.error('Failed to readdir', dir, error)
|
|
return []
|
|
}
|
|
}
|
|
|
|
async syncUserAudiobookData(req, res) {
|
|
// if (!req.body.data) {
|
|
// return res.status(403).send('Invalid local user audiobook data')
|
|
// }
|
|
|
|
// var hasUpdates = false
|
|
|
|
// // Local user audiobook data use the latest update
|
|
// req.body.data.forEach((uab) => {
|
|
// if (!uab || !uab.audiobookId) {
|
|
// Logger.error('[ApiController] Invalid user audiobook data', uab)
|
|
// return
|
|
// }
|
|
// var audiobook = this.db.audiobooks.find(ab => ab.id === uab.audiobookId)
|
|
// if (!audiobook) {
|
|
// Logger.info('[ApiController] syncUserAudiobookData local audiobook data audiobook no longer exists', uab.audiobookId)
|
|
// return
|
|
// }
|
|
// if (req.user.syncLocalUserAudiobookData(uab, audiobook)) {
|
|
// this.clientEmitter(req.user.id, 'current_user_audiobook_update', { id: uab.audiobookId, data: uab })
|
|
// hasUpdates = true
|
|
// }
|
|
// })
|
|
|
|
// if (hasUpdates) {
|
|
// await this.db.updateEntity('user', req.user)
|
|
// }
|
|
|
|
// var allUserAudiobookData = Object.values(req.user.audiobooksToJSON())
|
|
// res.json(allUserAudiobookData)
|
|
}
|
|
|
|
//
|
|
// Helper Methods
|
|
//
|
|
userJsonWithItemProgressDetails(user) {
|
|
var json = user.toJSONForBrowser()
|
|
|
|
json.libraryItemProgress = json.libraryItemProgress.map(lip => {
|
|
var libraryItem = this.db.libraryItems.find(li => li.id === lip.id)
|
|
if (!libraryItem) {
|
|
Logger.warn('[ApiRouter] Library item not found for users progress ' + lip.id)
|
|
return null
|
|
}
|
|
lip.media = libraryItem.media.toJSONExpanded()
|
|
return lip
|
|
}).filter(lip => !!lip)
|
|
|
|
return json
|
|
}
|
|
|
|
async handleDeleteLibraryItem(libraryItem) {
|
|
// Remove libraryItem from users
|
|
for (let i = 0; i < this.db.users.length; i++) {
|
|
var user = this.db.users[i]
|
|
var madeUpdates = user.removeLibraryItemProgress(libraryItem.id)
|
|
if (madeUpdates) {
|
|
await this.db.updateEntity('user', user)
|
|
}
|
|
}
|
|
|
|
// remove any streams open for this audiobook
|
|
// TODO: Change to PlaybackSessionManager to remove open sessions for user
|
|
// var streams = this.streamManager.streams.filter(stream => stream.audiobookId === libraryItem.id)
|
|
// for (let i = 0; i < streams.length; i++) {
|
|
// var stream = streams[i]
|
|
// var client = stream.client
|
|
// await stream.close()
|
|
// if (client && client.user) {
|
|
// client.user.stream = null
|
|
// client.stream = null
|
|
// this.db.updateUserStream(client.user.id, null)
|
|
// }
|
|
// }
|
|
|
|
// remove book from collections
|
|
var collectionsWithBook = this.db.collections.filter(c => c.books.includes(libraryItem.id))
|
|
for (let i = 0; i < collectionsWithBook.length; i++) {
|
|
var collection = collectionsWithBook[i]
|
|
collection.removeBook(libraryItem.id)
|
|
await this.db.updateEntity('collection', collection)
|
|
this.clientEmitter(collection.userId, 'collection_updated', collection.toJSONExpanded(this.db.libraryItems))
|
|
}
|
|
|
|
// purge cover cache
|
|
if (libraryItem.media.coverPath) {
|
|
await this.cacheManager.purgeCoverCache(libraryItem.id)
|
|
}
|
|
|
|
var json = libraryItem.toJSONExpanded()
|
|
await this.db.removeLibraryItem(libraryItem.id)
|
|
this.emitter('item_removed', json)
|
|
}
|
|
|
|
async getUserListeningSessionsHelper(userId) {
|
|
var userSessions = await this.db.selectUserSessions(userId)
|
|
return userSessions.sort((a, b) => b.updatedAt - a.updatedAt)
|
|
}
|
|
|
|
async getUserListeningStatsHelpers(userId) {
|
|
const today = date.format(new Date(), 'YYYY-MM-DD')
|
|
|
|
var listeningSessions = await this.getUserListeningSessionsHelper(userId)
|
|
var listeningStats = {
|
|
totalTime: 0,
|
|
items: {},
|
|
days: {},
|
|
dayOfWeek: {},
|
|
today: 0,
|
|
recentSessions: listeningSessions.slice(0, 10)
|
|
}
|
|
listeningSessions.forEach((s) => {
|
|
if (s.dayOfWeek) {
|
|
if (!listeningStats.dayOfWeek[s.dayOfWeek]) listeningStats.dayOfWeek[s.dayOfWeek] = 0
|
|
listeningStats.dayOfWeek[s.dayOfWeek] += s.timeListening
|
|
}
|
|
if (s.date && s.timeListening > 0) {
|
|
if (!listeningStats.days[s.date]) listeningStats.days[s.date] = 0
|
|
listeningStats.days[s.date] += s.timeListening
|
|
|
|
if (s.date === today) {
|
|
listeningStats.today += s.timeListening
|
|
}
|
|
}
|
|
if (!listeningStats.items[s.libraryItemId]) {
|
|
listeningStats.items[s.libraryItemId] = {
|
|
id: s.libraryItemId,
|
|
timeListening: s.timeListening,
|
|
mediaMetadata: s.mediaMetadata,
|
|
lastUpdate: s.lastUpdate
|
|
}
|
|
} else {
|
|
listeningStats.items[s.libraryItemId].timeListening += s.timeListening
|
|
}
|
|
|
|
listeningStats.totalTime += s.timeListening
|
|
})
|
|
return listeningStats
|
|
}
|
|
|
|
async purgeCache(req, res) {
|
|
if (!req.user.isRoot) {
|
|
return res.sendStatus(403)
|
|
}
|
|
Logger.info(`[ApiRouter] Purging all cache`)
|
|
await this.cacheManager.purgeAll()
|
|
res.sendStatus(200)
|
|
}
|
|
|
|
async createAuthorsAndSeriesForItemUpdate(mediaPayload) {
|
|
if (mediaPayload.metadata) {
|
|
var mediaMetadata = mediaPayload.metadata
|
|
|
|
// Create new authors if in payload
|
|
if (mediaMetadata.authors && mediaMetadata.authors.length) {
|
|
// TODO: validate authors
|
|
var newAuthors = []
|
|
for (let i = 0; i < mediaMetadata.authors.length; i++) {
|
|
if (mediaMetadata.authors[i].id.startsWith('new')) {
|
|
var author = this.db.authors.find(au => au.checkNameEquals(mediaMetadata.authors[i].name))
|
|
if (!author) {
|
|
author = new Author()
|
|
author.setData(mediaMetadata.authors[i])
|
|
Logger.debug(`[ApiRouter] Created new author "${author.name}"`)
|
|
newAuthors.push(author)
|
|
}
|
|
|
|
// Update ID in original payload
|
|
mediaMetadata.authors[i].id = author.id
|
|
}
|
|
}
|
|
if (newAuthors.length) {
|
|
await this.db.insertEntities('author', newAuthors)
|
|
this.emitter('authors_added', newAuthors)
|
|
}
|
|
}
|
|
|
|
// Create new series if in payload
|
|
if (mediaMetadata.series && mediaMetadata.series.length) {
|
|
// TODO: validate series
|
|
var newSeries = []
|
|
for (let i = 0; i < mediaMetadata.series.length; i++) {
|
|
if (mediaMetadata.series[i].id.startsWith('new')) {
|
|
var seriesItem = this.db.series.find(se => se.checkNameEquals(mediaMetadata.series[i].name))
|
|
if (!seriesItem) {
|
|
seriesItem = new Series()
|
|
seriesItem.setData(mediaMetadata.series[i])
|
|
Logger.debug(`[ApiRouter] Created new series "${seriesItem.name}"`)
|
|
newSeries.push(seriesItem)
|
|
}
|
|
|
|
// Update ID in original payload
|
|
mediaMetadata.series[i].id = seriesItem.id
|
|
}
|
|
}
|
|
if (newSeries.length) {
|
|
await this.db.insertEntities('series', newSeries)
|
|
this.emitter('authors_added', newSeries)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
getPodcastFeed(req, res) {
|
|
var url = req.body.rssFeed
|
|
if (!url) {
|
|
return res.status(400).send('Bad request')
|
|
}
|
|
|
|
axios.get(url).then(async (data) => {
|
|
if (!data || !data.data) {
|
|
Logger.error('Invalid podcast feed request response')
|
|
return res.status(500).send('Bad response from feed request')
|
|
}
|
|
var podcast = await parsePodcastRssFeedXml(data.data)
|
|
if (!podcast) {
|
|
return res.status(500).send('Invalid podcast RSS feed')
|
|
}
|
|
res.json(podcast)
|
|
}).catch((error) => {
|
|
console.error('Failed', error)
|
|
res.status(500).send(error)
|
|
})
|
|
}
|
|
}
|
|
module.exports = ApiRouter |