]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/controllers/bots.ts
Share playlists state
[github/Chocobozzz/PeerTube.git] / server / controllers / bots.ts
CommitLineData
2feebf3e
C
1import * as express from 'express'
2import { asyncMiddleware } from '../middlewares'
74dc3bca 3import { ROUTE_CACHE_LIFETIME, WEBSERVER } from '../initializers/constants'
2feebf3e 4import * as sitemapModule from 'sitemap'
2feebf3e
C
5import { VideoModel } from '../models/video/video'
6import { VideoChannelModel } from '../models/video/video-channel'
7import { AccountModel } from '../models/account/account'
8import { cacheRoute } from '../middlewares/cache'
9import { buildNSFWFilter } from '../helpers/express-utils'
10import { truncate } from 'lodash'
11
12const botsRouter = express.Router()
13
14// Special route that add OpenGraph and oEmbed tags
15// Do not use a template engine for a so little thing
16botsRouter.use('/sitemap.xml',
17 asyncMiddleware(cacheRoute(ROUTE_CACHE_LIFETIME.SITEMAP)),
18 asyncMiddleware(getSitemap)
19)
20
21// ---------------------------------------------------------------------------
22
23export {
24 botsRouter
25}
26
27// ---------------------------------------------------------------------------
28
29async function getSitemap (req: express.Request, res: express.Response) {
30 let urls = getSitemapBasicUrls()
31
32 urls = urls.concat(await getSitemapLocalVideoUrls())
33 urls = urls.concat(await getSitemapVideoChannelUrls())
34 urls = urls.concat(await getSitemapAccountUrls())
35
36 const sitemap = sitemapModule.createSitemap({
6dd9de95 37 hostname: WEBSERVER.URL,
2feebf3e
C
38 urls: urls
39 })
40
d5d9b6d7 41 const xml = sitemap.toXML()
2feebf3e 42
d5d9b6d7
C
43 res.header('Content-Type', 'application/xml')
44 res.send(xml)
2feebf3e
C
45}
46
47async function getSitemapVideoChannelUrls () {
48 const rows = await VideoChannelModel.listLocalsForSitemap('createdAt')
49
50 return rows.map(channel => ({
6dd9de95 51 url: WEBSERVER.URL + '/video-channels/' + channel.Actor.preferredUsername
2feebf3e
C
52 }))
53}
54
55async function getSitemapAccountUrls () {
56 const rows = await AccountModel.listLocalsForSitemap('createdAt')
57
58 return rows.map(channel => ({
6dd9de95 59 url: WEBSERVER.URL + '/accounts/' + channel.Actor.preferredUsername
2feebf3e
C
60 }))
61}
62
63async function getSitemapLocalVideoUrls () {
64 const resultList = await VideoModel.listForApi({
65 start: 0,
66 count: undefined,
67 sort: 'createdAt',
68 includeLocalVideos: true,
69 nsfw: buildNSFWFilter(),
70 filter: 'local',
71 withFiles: false
72 })
73
74 return resultList.data.map(v => ({
6dd9de95 75 url: WEBSERVER.URL + '/videos/watch/' + v.uuid,
2feebf3e
C
76 video: [
77 {
78 title: v.name,
79 // Sitemap description should be < 2000 characters
80 description: truncate(v.description || v.name, { length: 2000, omission: '...' }),
6dd9de95 81 player_loc: WEBSERVER.URL + '/videos/embed/' + v.uuid,
3acc5084 82 thumbnail_loc: WEBSERVER.URL + v.getMiniatureStaticPath()
2feebf3e
C
83 }
84 ]
85 }))
86}
87
88function getSitemapBasicUrls () {
89 const paths = [
90 '/about/instance',
91 '/videos/local'
92 ]
93
6dd9de95 94 return paths.map(p => ({ url: WEBSERVER.URL + p }))
2feebf3e 95}