aboutsummaryrefslogtreecommitdiffhomepage
path: root/server/helpers/requests.ts
blob: b556c392e262b321075afaa5c73d6dd79c9a7ea0 (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
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
import * as Bluebird from 'bluebird'
import { createWriteStream, remove } from 'fs-extra'
import * as request from 'request'
import { ACTIVITY_PUB, PEERTUBE_VERSION, WEBSERVER } from '../initializers/constants'
import { processImage } from './image-utils'
import { join } from 'path'
import { logger } from './logger'
import { CONFIG } from '../initializers/config'

function doRequest <T> (
  requestOptions: request.CoreOptions & request.UriOptions & { activityPub?: boolean },
  bodyKBLimit = 1000 // 1MB
): Bluebird<{ response: request.RequestResponse, body: T }> {
  if (!(requestOptions.headers)) requestOptions.headers = {}
  requestOptions.headers['User-Agent'] = getUserAgent()

  if (requestOptions.activityPub === true) {
    requestOptions.headers['accept'] = ACTIVITY_PUB.ACCEPT_HEADER
  }

  return new Bluebird<{ response: request.RequestResponse, body: T }>((res, rej) => {
    request(requestOptions, (err, response, body) => err ? rej(err) : res({ response, body }))
      .on('data', onRequestDataLengthCheck(bodyKBLimit))
  })
}

function doRequestAndSaveToFile (
  requestOptions: request.CoreOptions & request.UriOptions,
  destPath: string,
  bodyKBLimit = 10000 // 10MB
) {
  if (!requestOptions.headers) requestOptions.headers = {}
  requestOptions.headers['User-Agent'] = getUserAgent()

  return new Bluebird<void>((res, rej) => {
    const file = createWriteStream(destPath)
    file.on('finish', () => res())

    request(requestOptions)
      .on('data', onRequestDataLengthCheck(bodyKBLimit))
      .on('error', err => {
        file.close()

        remove(destPath)
          .catch(err => logger.error('Cannot remove %s after request failure.', destPath, { err }))

        return rej(err)
      })
      .pipe(file)
  })
}

async function downloadImage (url: string, destDir: string, destName: string, size: { width: number, height: number }) {
  const tmpPath = join(CONFIG.STORAGE.TMP_DIR, 'pending-' + destName)
  await doRequestAndSaveToFile({ method: 'GET', uri: url }, tmpPath)

  const destPath = join(destDir, destName)

  try {
    await processImage(tmpPath, destPath, size)
  } catch (err) {
    await remove(tmpPath)

    throw err
  }
}

function getUserAgent () {
  return `PeerTube/${PEERTUBE_VERSION} (+${WEBSERVER.URL})`
}

// ---------------------------------------------------------------------------

export {
  doRequest,
  doRequestAndSaveToFile,
  downloadImage
}

// ---------------------------------------------------------------------------

// Thanks to https://github.com/request/request/issues/2470#issuecomment-268929907 <3
function onRequestDataLengthCheck (bodyKBLimit: number) {
  let bufferLength = 0
  const bytesLimit = bodyKBLimit * 1000

  return function (chunk) {
    bufferLength += chunk.length
    if (bufferLength > bytesLimit) {
      this.abort()

      const error = new Error(`Response was too large - aborted after ${bytesLimit} bytes.`)
      this.emit('error', error)
    }
  }
}