PeerTube/server/core/lib/live/live-segment-sha-store.ts

98 lines
3.0 KiB
TypeScript
Raw Normal View History

import { writeJson } from 'fs-extra/esm'
import { rename } from 'fs/promises'
2023-05-15 04:09:16 -05:00
import PQueue from 'p-queue'
2021-06-16 08:14:41 -05:00
import { basename } from 'path'
import { mapToJSON } from '@server/helpers/core-utils.js'
import { logger, loggerTagsFactory } from '@server/helpers/logger.js'
import { MStreamingPlaylistVideo } from '@server/types/models/index.js'
import { buildSha256Segment } from '../hls.js'
import { storeHLSFileFromPath } from '../object-storage/index.js'
import { JFWriteOptions } from 'jsonfile'
2021-06-16 08:14:41 -05:00
const lTags = loggerTagsFactory('live')
class LiveSegmentShaStore {
private readonly segmentsSha256 = new Map<string, string>()
private readonly videoUUID: string
2023-05-15 04:09:16 -05:00
private readonly sha256Path: string
2023-05-15 04:09:16 -05:00
private readonly sha256PathTMP: string
private readonly streamingPlaylist: MStreamingPlaylistVideo
private readonly sendToObjectStorage: boolean
private readonly writeQueue = new PQueue({ concurrency: 1 })
constructor (options: {
videoUUID: string
sha256Path: string
streamingPlaylist: MStreamingPlaylistVideo
sendToObjectStorage: boolean
}) {
this.videoUUID = options.videoUUID
2023-05-15 04:09:16 -05:00
this.sha256Path = options.sha256Path
2023-05-15 04:09:16 -05:00
this.sha256PathTMP = options.sha256Path + '.tmp'
this.streamingPlaylist = options.streamingPlaylist
this.sendToObjectStorage = options.sendToObjectStorage
2021-06-16 08:14:41 -05:00
}
async addSegmentSha (segmentPath: string) {
logger.debug('Adding live sha segment %s.', segmentPath, lTags(this.videoUUID))
2021-06-16 08:14:41 -05:00
const shaResult = await buildSha256Segment(segmentPath)
const segmentName = basename(segmentPath)
this.segmentsSha256.set(segmentName, shaResult)
2021-06-16 08:14:41 -05:00
try {
await this.writeToDisk()
} catch (err) {
logger.error('Cannot write sha segments to disk.', { err })
}
2021-06-16 08:14:41 -05:00
}
async removeSegmentSha (segmentPath: string) {
2021-06-16 08:14:41 -05:00
const segmentName = basename(segmentPath)
logger.debug('Removing live sha segment %s.', segmentPath, lTags(this.videoUUID))
2021-06-16 08:14:41 -05:00
if (!this.segmentsSha256.has(segmentName)) {
logger.warn(
'Unknown segment in live segment hash store for video %s and segment %s.',
this.videoUUID, segmentPath, lTags(this.videoUUID)
)
2021-06-16 08:14:41 -05:00
return
}
this.segmentsSha256.delete(segmentName)
2021-06-16 08:14:41 -05:00
await this.writeToDisk()
2021-06-16 08:14:41 -05:00
}
private writeToDisk () {
return this.writeQueue.add(async () => {
2023-06-19 07:29:50 -05:00
logger.debug(`Writing segment sha JSON ${this.sha256Path} of ${this.videoUUID} on disk.`, lTags(this.videoUUID))
2023-05-22 07:31:35 -05:00
2023-05-16 03:06:07 -05:00
// Atomic write: use rename instead of move that is not atomic
await writeJson(this.sha256PathTMP, mapToJSON(this.segmentsSha256), { flush: true } as JFWriteOptions) // FIXME: jsonfile typings
2023-05-16 03:06:07 -05:00
await rename(this.sha256PathTMP, this.sha256Path)
2021-06-16 08:14:41 -05:00
if (this.sendToObjectStorage) {
const url = await storeHLSFileFromPath(this.streamingPlaylist, this.sha256Path)
if (this.streamingPlaylist.segmentsSha256Url !== url) {
this.streamingPlaylist.segmentsSha256Url = url
await this.streamingPlaylist.save()
}
}
})
2021-06-16 08:14:41 -05:00
}
}
export {
LiveSegmentShaStore
}