aboutsummaryrefslogtreecommitdiffhomepage
path: root/server/lib/files-cache/abstract-video-static-file-cache.ts
blob: 61837e0f8e6c101a34f93c97e637e23fadb692f7 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
import { createWriteStream, remove } from 'fs-extra'
import { logger } from '../../helpers/logger'
import { VideoModel } from '../../models/video/video'
import { fetchRemoteVideoStaticFile } from '../activitypub'
import * as memoizee from 'memoizee'

export abstract class AbstractVideoStaticFileCache <T> {

  getFilePath: (params: T) => Promise<string>

  abstract getFilePathImpl (params: T): Promise<string>

  // Load and save the remote file, then return the local path from filesystem
  protected abstract loadRemoteFile (key: string): Promise<string>

  init (max: number, maxAge: number) {
    this.getFilePath = memoizee(this.getFilePathImpl, {
      maxAge,
      max,
      promise: true,
      dispose: (value: string) => {
        remove(value)
          .then(() => logger.debug('%s evicted from %s', value, this.constructor.name))
          .catch(err => logger.error('Cannot remove %s from cache %s.', value, this.constructor.name, { err }))
      }
    })
  }

  protected saveRemoteVideoFileAndReturnPath (video: VideoModel, remoteStaticPath: string, destPath: string) {
    return new Promise<string>((res, rej) => {
      const req = fetchRemoteVideoStaticFile(video, remoteStaticPath, rej)

      const stream = createWriteStream(destPath)

      req.pipe(stream)
         .on('error', (err) => rej(err))
         .on('finish', () => res(destPath))
    })
  }
}