helper.js 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. const fs = require('node:fs')
  2. const merge = require('lodash.merge')
  3. const stripIndent = require('common-tags/lib/stripIndent')
  4. const crypto = require('node:crypto')
  5. const utils = require('../server/helpers/utils')
  6. const logger = require('../server/logger')
  7. // @ts-ignore
  8. const { version } = require('../../package.json')
  9. /**
  10. * Reads all companion configuration set via environment variables
  11. * and via the config file path
  12. *
  13. * @returns {object}
  14. */
  15. exports.getCompanionOptions = (options = {}) => {
  16. return merge({}, getConfigFromEnv(), getConfigFromFile(), options)
  17. }
  18. /**
  19. * Loads the config from environment variables
  20. *
  21. * @returns {object}
  22. */
  23. const getConfigFromEnv = () => {
  24. const uploadUrls = process.env.COMPANION_UPLOAD_URLS
  25. const domains = process.env.COMPANION_DOMAINS || process.env.COMPANION_DOMAIN || null
  26. const validHosts = domains ? domains.split(',') : []
  27. return {
  28. providerOptions: {
  29. drive: {
  30. key: process.env.COMPANION_GOOGLE_KEY,
  31. secret: getSecret('COMPANION_GOOGLE_SECRET'),
  32. credentialsURL: process.env.COMPANION_GOOGLE_KEYS_ENDPOINT,
  33. },
  34. dropbox: {
  35. key: process.env.COMPANION_DROPBOX_KEY,
  36. secret: getSecret('COMPANION_DROPBOX_SECRET'),
  37. credentialsURL: process.env.COMPANION_DROPBOX_KEYS_ENDPOINT,
  38. },
  39. box: {
  40. key: process.env.COMPANION_BOX_KEY,
  41. secret: getSecret('COMPANION_BOX_SECRET'),
  42. },
  43. instagram: {
  44. key: process.env.COMPANION_INSTAGRAM_KEY,
  45. secret: getSecret('COMPANION_INSTAGRAM_SECRET'),
  46. credentialsURL: process.env.COMPANION_INSTAGRAM_KEYS_ENDPOINT,
  47. },
  48. facebook: {
  49. key: process.env.COMPANION_FACEBOOK_KEY,
  50. secret: getSecret('COMPANION_FACEBOOK_SECRET'),
  51. credentialsURL: process.env.COMPANION_FACEBOOK_KEYS_ENDPOINT,
  52. },
  53. onedrive: {
  54. key: process.env.COMPANION_ONEDRIVE_KEY,
  55. secret: getSecret('COMPANION_ONEDRIVE_SECRET'),
  56. credentialsURL: process.env.COMPANION_ONEDRIVE_KEYS_ENDPOINT,
  57. },
  58. zoom: {
  59. key: process.env.COMPANION_ZOOM_KEY,
  60. secret: getSecret('COMPANION_ZOOM_SECRET'),
  61. verificationToken: getSecret('COMPANION_ZOOM_VERIFICATION_TOKEN'),
  62. credentialsURL: process.env.COMPANION_ZOOM_KEYS_ENDPOINT,
  63. },
  64. unsplash: {
  65. key: process.env.COMPANION_UNSPLASH_KEY,
  66. secret: process.env.COMPANION_UNSPLASH_SECRET,
  67. },
  68. },
  69. s3: {
  70. key: process.env.COMPANION_AWS_KEY,
  71. getKey: utils.defaultGetKey,
  72. secret: getSecret('COMPANION_AWS_SECRET'),
  73. bucket: process.env.COMPANION_AWS_BUCKET,
  74. endpoint: process.env.COMPANION_AWS_ENDPOINT,
  75. region: process.env.COMPANION_AWS_REGION,
  76. useAccelerateEndpoint:
  77. process.env.COMPANION_AWS_USE_ACCELERATE_ENDPOINT === 'true',
  78. expires: parseInt(process.env.COMPANION_AWS_EXPIRES || '800', 10),
  79. acl: process.env.COMPANION_AWS_ACL,
  80. },
  81. server: {
  82. host: process.env.COMPANION_DOMAIN,
  83. protocol: process.env.COMPANION_PROTOCOL,
  84. path: process.env.COMPANION_PATH,
  85. implicitPath: process.env.COMPANION_IMPLICIT_PATH,
  86. oauthDomain: process.env.COMPANION_OAUTH_DOMAIN,
  87. validHosts,
  88. },
  89. periodicPingUrls: process.env.COMPANION_PERIODIC_PING_URLS ? process.env.COMPANION_PERIODIC_PING_URLS.split(',') : [],
  90. periodicPingInterval: process.env.COMPANION_PERIODIC_PING_INTERVAL
  91. ? parseInt(process.env.COMPANION_PERIODIC_PING_INTERVAL, 10) : undefined,
  92. periodicPingStaticPayload: process.env.COMPANION_PERIODIC_PING_STATIC_JSON_PAYLOAD
  93. ? JSON.parse(process.env.COMPANION_PERIODIC_PING_STATIC_JSON_PAYLOAD) : undefined,
  94. periodicPingCount: process.env.COMPANION_PERIODIC_PING_COUNT
  95. ? parseInt(process.env.COMPANION_PERIODIC_PING_COUNT, 10) : undefined,
  96. filePath: process.env.COMPANION_DATADIR,
  97. redisUrl: process.env.COMPANION_REDIS_URL,
  98. // adding redisOptions to keep all companion options easily visible
  99. // redisOptions refers to https://www.npmjs.com/package/redis#options-object-properties
  100. redisOptions: {},
  101. sendSelfEndpoint: process.env.COMPANION_SELF_ENDPOINT,
  102. uploadUrls: uploadUrls ? uploadUrls.split(',') : null,
  103. secret: getSecret('COMPANION_SECRET') || generateSecret(),
  104. preAuthSecret: getSecret('COMPANION_PREAUTH_SECRET') || generateSecret(),
  105. allowLocalUrls: process.env.COMPANION_ALLOW_LOCAL_URLS === 'true',
  106. // cookieDomain is kind of a hack to support distributed systems. This should be improved but we never got so far.
  107. cookieDomain: process.env.COMPANION_COOKIE_DOMAIN,
  108. streamingUpload: process.env.COMPANION_STREAMING_UPLOAD === 'true',
  109. maxFileSize: process.env.COMPANION_MAX_FILE_SIZE ? parseInt(process.env.COMPANION_MAX_FILE_SIZE, 10) : undefined,
  110. chunkSize: process.env.COMPANION_CHUNK_SIZE ? parseInt(process.env.COMPANION_CHUNK_SIZE, 10) : undefined,
  111. clientSocketConnectTimeout: process.env.COMPANION_CLIENT_SOCKET_CONNECT_TIMEOUT
  112. ? parseInt(process.env.COMPANION_CLIENT_SOCKET_CONNECT_TIMEOUT, 10) : undefined,
  113. metrics: process.env.COMPANION_HIDE_METRICS !== 'true',
  114. }
  115. }
  116. /**
  117. * Tries to read the secret from a file if the according environment variable is set.
  118. * Otherwise it falls back to the standard secret environment variable.
  119. *
  120. * @param {string} baseEnvVar
  121. *
  122. * @returns {string}
  123. */
  124. const getSecret = (baseEnvVar) => {
  125. const secretFile = process.env[`${baseEnvVar}_FILE`]
  126. return secretFile
  127. ? fs.readFileSync(secretFile).toString()
  128. : process.env[baseEnvVar]
  129. }
  130. /**
  131. * Auto-generates server secret
  132. *
  133. * @returns {string}
  134. */
  135. const generateSecret = () => {
  136. logger.warn('auto-generating server secret because none was specified', 'startup.secret')
  137. return crypto.randomBytes(64).toString('hex')
  138. }
  139. /**
  140. * Loads the config from a file and returns it as an object
  141. *
  142. * @returns {object}
  143. */
  144. const getConfigFromFile = () => {
  145. const path = getConfigPath()
  146. if (!path) return {}
  147. const rawdata = fs.readFileSync(getConfigPath())
  148. // @ts-ignore
  149. return JSON.parse(rawdata)
  150. }
  151. /**
  152. * Returns the config path specified via cli arguments
  153. *
  154. * @returns {string}
  155. */
  156. const getConfigPath = () => {
  157. let configPath
  158. for (let i = process.argv.length - 1; i >= 0; i--) {
  159. const isConfigFlag = process.argv[i] === '-c' || process.argv[i] === '--config'
  160. const flagHasValue = i + 1 <= process.argv.length
  161. if (isConfigFlag && flagHasValue) {
  162. configPath = process.argv[i + 1]
  163. break
  164. }
  165. }
  166. return configPath
  167. }
  168. /**
  169. *
  170. * @param {string} url
  171. */
  172. exports.hasProtocol = (url) => {
  173. return url.startsWith('http://') || url.startsWith('https://')
  174. }
  175. exports.buildHelpfulStartupMessage = (companionOptions) => {
  176. const buildURL = utils.getURLBuilder(companionOptions)
  177. const callbackURLs = []
  178. Object.keys(companionOptions.providerOptions).forEach((providerName) => {
  179. callbackURLs.push(buildURL(`/connect/${providerName}/redirect`, true))
  180. })
  181. return stripIndent`
  182. Welcome to Companion v${version}
  183. ===================================
  184. Congratulations on setting up Companion! Thanks for joining our cause, you have taken
  185. the first step towards the future of file uploading! We
  186. hope you are as excited about this as we are!
  187. While you did an awesome job on getting Companion running, this is just the welcome
  188. message, so let's talk about the places that really matter:
  189. - Be sure to add ${callbackURLs.join(', ')} as your Oauth redirect uris on their corresponding developer interfaces.
  190. - The URL ${buildURL('/metrics', true)} is available for statistics to keep Companion running smoothly
  191. - https://github.com/transloadit/uppy/issues - report your bugs here
  192. So quit lollygagging, start uploading and experience the future!
  193. `
  194. }