installer.ts 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. import { ensureDir, readdir, remove } from 'fs-extra'
  2. import passwordGenerator from 'password-generator'
  3. import { join } from 'path'
  4. import { isTestOrDevInstance } from '@server/helpers/core-utils'
  5. import { getNodeABIVersion } from '@server/helpers/version'
  6. import { UserRole } from '@shared/models'
  7. import { logger } from '../helpers/logger'
  8. import { buildUser, createApplicationActor, createUserAccountAndChannelAndPlaylist } from '../lib/user'
  9. import { ApplicationModel } from '../models/application/application'
  10. import { OAuthClientModel } from '../models/oauth/oauth-client'
  11. import { applicationExist, clientsExist, usersExist } from './checker-after-init'
  12. import { CONFIG } from './config'
  13. import { DIRECTORIES, FILES_CACHE, LAST_MIGRATION_VERSION } from './constants'
  14. import { sequelizeTypescript } from './database'
  15. async function installApplication () {
  16. try {
  17. await Promise.all([
  18. // Database related
  19. sequelizeTypescript.sync()
  20. .then(() => {
  21. return Promise.all([
  22. createApplicationIfNotExist(),
  23. createOAuthClientIfNotExist(),
  24. createOAuthAdminIfNotExist()
  25. ])
  26. }),
  27. // Directories
  28. removeCacheAndTmpDirectories()
  29. .then(() => createDirectoriesIfNotExist())
  30. ])
  31. } catch (err) {
  32. logger.error('Cannot install application.', { err })
  33. process.exit(-1)
  34. }
  35. }
  36. // ---------------------------------------------------------------------------
  37. export {
  38. installApplication
  39. }
  40. // ---------------------------------------------------------------------------
  41. function removeCacheAndTmpDirectories () {
  42. const cacheDirectories = Object.keys(FILES_CACHE)
  43. .map(k => FILES_CACHE[k].DIRECTORY)
  44. const tasks: Promise<any>[] = []
  45. // Cache directories
  46. for (const key of Object.keys(cacheDirectories)) {
  47. const dir = cacheDirectories[key]
  48. tasks.push(removeDirectoryOrContent(dir))
  49. }
  50. tasks.push(removeDirectoryOrContent(CONFIG.STORAGE.TMP_DIR))
  51. return Promise.all(tasks)
  52. }
  53. async function removeDirectoryOrContent (dir: string) {
  54. try {
  55. await remove(dir)
  56. } catch (err) {
  57. logger.debug('Cannot remove directory %s. Removing content instead.', dir, { err })
  58. const files = await readdir(dir)
  59. for (const file of files) {
  60. await remove(join(dir, file))
  61. }
  62. }
  63. }
  64. function createDirectoriesIfNotExist () {
  65. const storage = CONFIG.STORAGE
  66. const cacheDirectories = Object.keys(FILES_CACHE)
  67. .map(k => FILES_CACHE[k].DIRECTORY)
  68. const tasks: Promise<void>[] = []
  69. for (const key of Object.keys(storage)) {
  70. const dir = storage[key]
  71. tasks.push(ensureDir(dir))
  72. }
  73. // Cache directories
  74. for (const key of Object.keys(cacheDirectories)) {
  75. const dir = cacheDirectories[key]
  76. tasks.push(ensureDir(dir))
  77. }
  78. tasks.push(ensureDir(DIRECTORIES.HLS_STREAMING_PLAYLIST.PRIVATE))
  79. tasks.push(ensureDir(DIRECTORIES.HLS_STREAMING_PLAYLIST.PUBLIC))
  80. tasks.push(ensureDir(DIRECTORIES.VIDEOS.PUBLIC))
  81. tasks.push(ensureDir(DIRECTORIES.VIDEOS.PRIVATE))
  82. // Resumable upload directory
  83. tasks.push(ensureDir(DIRECTORIES.RESUMABLE_UPLOAD))
  84. return Promise.all(tasks)
  85. }
  86. async function createOAuthClientIfNotExist () {
  87. const exist = await clientsExist()
  88. // Nothing to do, clients already exist
  89. if (exist === true) return undefined
  90. logger.info('Creating a default OAuth Client.')
  91. const id = passwordGenerator(32, false, /[a-z0-9]/)
  92. const secret = passwordGenerator(32, false, /[a-zA-Z0-9]/)
  93. const client = new OAuthClientModel({
  94. clientId: id,
  95. clientSecret: secret,
  96. grants: [ 'password', 'refresh_token' ],
  97. redirectUris: null
  98. })
  99. const createdClient = await client.save()
  100. logger.info('Client id: ' + createdClient.clientId)
  101. logger.info('Client secret: ' + createdClient.clientSecret)
  102. return undefined
  103. }
  104. async function createOAuthAdminIfNotExist () {
  105. const exist = await usersExist()
  106. // Nothing to do, users already exist
  107. if (exist === true) return undefined
  108. logger.info('Creating the administrator.')
  109. const username = 'root'
  110. const role = UserRole.ADMINISTRATOR
  111. const email = CONFIG.ADMIN.EMAIL
  112. let validatePassword = true
  113. let password = ''
  114. // Do not generate a random password for test and dev environments
  115. if (isTestOrDevInstance()) {
  116. password = 'test'
  117. if (process.env.NODE_APP_INSTANCE) {
  118. password += process.env.NODE_APP_INSTANCE
  119. }
  120. // Our password is weak so do not validate it
  121. validatePassword = false
  122. } else if (process.env.PT_INITIAL_ROOT_PASSWORD) {
  123. password = process.env.PT_INITIAL_ROOT_PASSWORD
  124. } else {
  125. password = passwordGenerator(16, true)
  126. }
  127. const user = buildUser({
  128. username,
  129. email,
  130. password,
  131. role,
  132. emailVerified: true,
  133. videoQuota: -1,
  134. videoQuotaDaily: -1
  135. })
  136. await createUserAccountAndChannelAndPlaylist({ userToCreate: user, channelNames: undefined, validateUser: validatePassword })
  137. logger.info('Username: ' + username)
  138. logger.info('User password: ' + password)
  139. }
  140. async function createApplicationIfNotExist () {
  141. const exist = await applicationExist()
  142. // Nothing to do, application already exist
  143. if (exist === true) return undefined
  144. logger.info('Creating application account.')
  145. const application = await ApplicationModel.create({
  146. migrationVersion: LAST_MIGRATION_VERSION,
  147. nodeVersion: process.version,
  148. nodeABIVersion: getNodeABIVersion()
  149. })
  150. return createApplicationActor(application.id)
  151. }