2021-10-27 08:37:04 -04:00
|
|
|
import { getServerActor } from '@server/models/application/application'
|
2022-09-27 10:19:36 -04:00
|
|
|
import { logger } from '@uploadx/core'
|
2021-08-27 08:32:44 -04:00
|
|
|
import express from 'express'
|
2021-07-26 09:04:37 -04:00
|
|
|
import { truncate } from 'lodash'
|
2022-09-27 10:19:36 -04:00
|
|
|
import { SitemapStream, streamToPromise, ErrorLevel } from 'sitemap'
|
2021-07-26 09:04:37 -04:00
|
|
|
import { buildNSFWFilter } from '../helpers/express-utils'
|
|
|
|
import { ROUTE_CACHE_LIFETIME, WEBSERVER } from '../initializers/constants'
|
|
|
|
import { asyncMiddleware } from '../middlewares'
|
|
|
|
import { cacheRoute } from '../middlewares/cache/cache'
|
|
|
|
import { AccountModel } from '../models/account/account'
|
2018-12-05 11:27:24 -05:00
|
|
|
import { VideoModel } from '../models/video/video'
|
|
|
|
import { VideoChannelModel } from '../models/video/video-channel'
|
|
|
|
|
|
|
|
const botsRouter = express.Router()
|
|
|
|
|
|
|
|
// Special route that add OpenGraph and oEmbed tags
|
|
|
|
// Do not use a template engine for a so little thing
|
|
|
|
botsRouter.use('/sitemap.xml',
|
2021-07-22 05:15:17 -04:00
|
|
|
cacheRoute(ROUTE_CACHE_LIFETIME.SITEMAP),
|
2018-12-05 11:27:24 -05:00
|
|
|
asyncMiddleware(getSitemap)
|
|
|
|
)
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export {
|
|
|
|
botsRouter
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
async function getSitemap (req: express.Request, res: express.Response) {
|
|
|
|
let urls = getSitemapBasicUrls()
|
|
|
|
|
|
|
|
urls = urls.concat(await getSitemapLocalVideoUrls())
|
|
|
|
urls = urls.concat(await getSitemapVideoChannelUrls())
|
|
|
|
urls = urls.concat(await getSitemapAccountUrls())
|
|
|
|
|
2022-09-27 10:19:36 -04:00
|
|
|
const sitemapStream = new SitemapStream({
|
|
|
|
hostname: WEBSERVER.URL,
|
|
|
|
errorHandler: (err: Error, level: ErrorLevel) => {
|
|
|
|
if (level === 'warn') {
|
|
|
|
logger.warn('Warning in sitemap generation.', { err })
|
|
|
|
} else if (level === 'throw') {
|
|
|
|
logger.error('Error in sitemap generation.', { err })
|
|
|
|
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
2018-12-05 11:27:24 -05:00
|
|
|
|
2020-04-01 11:02:52 -04:00
|
|
|
for (const urlObj of urls) {
|
|
|
|
sitemapStream.write(urlObj)
|
2020-04-01 08:16:19 -04:00
|
|
|
}
|
|
|
|
sitemapStream.end()
|
|
|
|
|
|
|
|
const xml = await streamToPromise(sitemapStream)
|
2018-12-05 11:27:24 -05:00
|
|
|
|
2019-10-21 08:50:55 -04:00
|
|
|
res.header('Content-Type', 'application/xml')
|
|
|
|
res.send(xml)
|
2018-12-05 11:27:24 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
async function getSitemapVideoChannelUrls () {
|
|
|
|
const rows = await VideoChannelModel.listLocalsForSitemap('createdAt')
|
|
|
|
|
|
|
|
return rows.map(channel => ({
|
2019-04-11 05:33:44 -04:00
|
|
|
url: WEBSERVER.URL + '/video-channels/' + channel.Actor.preferredUsername
|
2018-12-05 11:27:24 -05:00
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getSitemapAccountUrls () {
|
|
|
|
const rows = await AccountModel.listLocalsForSitemap('createdAt')
|
|
|
|
|
|
|
|
return rows.map(channel => ({
|
2019-04-11 05:33:44 -04:00
|
|
|
url: WEBSERVER.URL + '/accounts/' + channel.Actor.preferredUsername
|
2018-12-05 11:27:24 -05:00
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getSitemapLocalVideoUrls () {
|
2021-10-27 08:37:04 -04:00
|
|
|
const serverActor = await getServerActor()
|
|
|
|
|
2020-01-08 08:15:16 -05:00
|
|
|
const { data } = await VideoModel.listForApi({
|
2018-12-05 11:27:24 -05:00
|
|
|
start: 0,
|
|
|
|
count: undefined,
|
|
|
|
sort: 'createdAt',
|
2021-10-27 08:37:04 -04:00
|
|
|
displayOnlyForFollower: {
|
|
|
|
actorId: serverActor.id,
|
|
|
|
orLocalVideos: true
|
|
|
|
},
|
|
|
|
isLocal: true,
|
2018-12-05 11:27:24 -05:00
|
|
|
nsfw: buildNSFWFilter(),
|
2020-01-08 08:15:16 -05:00
|
|
|
countVideos: false
|
2018-12-05 11:27:24 -05:00
|
|
|
})
|
|
|
|
|
2020-01-08 08:15:16 -05:00
|
|
|
return data.map(v => ({
|
2021-07-26 09:04:37 -04:00
|
|
|
url: WEBSERVER.URL + v.getWatchStaticPath(),
|
2018-12-05 11:27:24 -05:00
|
|
|
video: [
|
|
|
|
{
|
2022-09-27 10:19:36 -04:00
|
|
|
// Sitemap title should be < 100 characters
|
|
|
|
title: truncate(v.name, { length: 100, omission: '...' }),
|
2018-12-05 11:27:24 -05:00
|
|
|
// Sitemap description should be < 2000 characters
|
|
|
|
description: truncate(v.description || v.name, { length: 2000, omission: '...' }),
|
2021-07-26 09:04:37 -04:00
|
|
|
player_loc: WEBSERVER.URL + v.getEmbedStaticPath(),
|
2019-04-23 03:50:57 -04:00
|
|
|
thumbnail_loc: WEBSERVER.URL + v.getMiniatureStaticPath()
|
2018-12-05 11:27:24 -05:00
|
|
|
}
|
|
|
|
]
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
|
|
|
function getSitemapBasicUrls () {
|
|
|
|
const paths = [
|
|
|
|
'/about/instance',
|
|
|
|
'/videos/local'
|
|
|
|
]
|
|
|
|
|
2019-04-11 05:33:44 -04:00
|
|
|
return paths.map(p => ({ url: WEBSERVER.URL + p }))
|
2018-12-05 11:27:24 -05:00
|
|
|
}
|