prune-storage.ts 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. /* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */
  2. import 'mocha'
  3. import * as chai from 'chai'
  4. import { createFile, readdir } from 'fs-extra'
  5. import { join } from 'path'
  6. import { wait } from '@shared/core-utils'
  7. import { buildUUID } from '@shared/extra-utils'
  8. import { HttpStatusCode, VideoPlaylistPrivacy } from '@shared/models'
  9. import {
  10. cleanupTests,
  11. CLICommand,
  12. createMultipleServers,
  13. doubleFollow,
  14. killallServers,
  15. makeGetRequest,
  16. PeerTubeServer,
  17. setAccessTokensToServers,
  18. setDefaultVideoChannel,
  19. waitJobs
  20. } from '@shared/server-commands'
  21. const expect = chai.expect
  22. async function countFiles (server: PeerTubeServer, directory: string) {
  23. const files = await readdir(server.servers.buildDirectory(directory))
  24. return files.length
  25. }
  26. async function assertNotExists (server: PeerTubeServer, directory: string, substring: string) {
  27. const files = await readdir(server.servers.buildDirectory(directory))
  28. for (const f of files) {
  29. expect(f).to.not.contain(substring)
  30. }
  31. }
  32. async function assertCountAreOkay (servers: PeerTubeServer[]) {
  33. for (const server of servers) {
  34. const videosCount = await countFiles(server, 'videos')
  35. expect(videosCount).to.equal(8)
  36. const torrentsCount = await countFiles(server, 'torrents')
  37. expect(torrentsCount).to.equal(16)
  38. const previewsCount = await countFiles(server, 'previews')
  39. expect(previewsCount).to.equal(2)
  40. const thumbnailsCount = await countFiles(server, 'thumbnails')
  41. expect(thumbnailsCount).to.equal(6)
  42. const avatarsCount = await countFiles(server, 'avatars')
  43. expect(avatarsCount).to.equal(4)
  44. const hlsRootCount = await countFiles(server, 'streaming-playlists/hls')
  45. expect(hlsRootCount).to.equal(2)
  46. }
  47. }
  48. describe('Test prune storage scripts', function () {
  49. let servers: PeerTubeServer[]
  50. const badNames: { [directory: string]: string[] } = {}
  51. before(async function () {
  52. this.timeout(120000)
  53. servers = await createMultipleServers(2, { transcoding: { enabled: true } })
  54. await setAccessTokensToServers(servers)
  55. await setDefaultVideoChannel(servers)
  56. for (const server of servers) {
  57. await server.videos.upload({ attributes: { name: 'video 1' } })
  58. await server.videos.upload({ attributes: { name: 'video 2' } })
  59. await server.users.updateMyAvatar({ fixture: 'avatar.png' })
  60. await server.playlists.create({
  61. attributes: {
  62. displayName: 'playlist',
  63. privacy: VideoPlaylistPrivacy.PUBLIC,
  64. videoChannelId: server.store.channel.id,
  65. thumbnailfile: 'thumbnail.jpg'
  66. }
  67. })
  68. }
  69. await doubleFollow(servers[0], servers[1])
  70. // Lazy load the remote avatars
  71. {
  72. const account = await servers[0].accounts.get({ accountName: 'root@localhost:' + servers[1].port })
  73. for (const avatar of account.avatars) {
  74. await makeGetRequest({
  75. url: servers[0].url,
  76. path: avatar.path,
  77. expectedStatus: HttpStatusCode.OK_200
  78. })
  79. }
  80. }
  81. {
  82. const account = await servers[1].accounts.get({ accountName: 'root@localhost:' + servers[0].port })
  83. for (const avatar of account.avatars) {
  84. await makeGetRequest({
  85. url: servers[1].url,
  86. path: avatar.path,
  87. expectedStatus: HttpStatusCode.OK_200
  88. })
  89. }
  90. }
  91. await wait(1000)
  92. await waitJobs(servers)
  93. await killallServers(servers)
  94. await wait(1000)
  95. })
  96. it('Should have the files on the disk', async function () {
  97. await assertCountAreOkay(servers)
  98. })
  99. it('Should create some dirty files', async function () {
  100. for (let i = 0; i < 2; i++) {
  101. {
  102. const base = servers[0].servers.buildDirectory('videos')
  103. const n1 = buildUUID() + '.mp4'
  104. const n2 = buildUUID() + '.webm'
  105. await createFile(join(base, n1))
  106. await createFile(join(base, n2))
  107. badNames['videos'] = [ n1, n2 ]
  108. }
  109. {
  110. const base = servers[0].servers.buildDirectory('torrents')
  111. const n1 = buildUUID() + '-240.torrent'
  112. const n2 = buildUUID() + '-480.torrent'
  113. await createFile(join(base, n1))
  114. await createFile(join(base, n2))
  115. badNames['torrents'] = [ n1, n2 ]
  116. }
  117. {
  118. const base = servers[0].servers.buildDirectory('thumbnails')
  119. const n1 = buildUUID() + '.jpg'
  120. const n2 = buildUUID() + '.jpg'
  121. await createFile(join(base, n1))
  122. await createFile(join(base, n2))
  123. badNames['thumbnails'] = [ n1, n2 ]
  124. }
  125. {
  126. const base = servers[0].servers.buildDirectory('previews')
  127. const n1 = buildUUID() + '.jpg'
  128. const n2 = buildUUID() + '.jpg'
  129. await createFile(join(base, n1))
  130. await createFile(join(base, n2))
  131. badNames['previews'] = [ n1, n2 ]
  132. }
  133. {
  134. const base = servers[0].servers.buildDirectory('avatars')
  135. const n1 = buildUUID() + '.png'
  136. const n2 = buildUUID() + '.jpg'
  137. await createFile(join(base, n1))
  138. await createFile(join(base, n2))
  139. badNames['avatars'] = [ n1, n2 ]
  140. }
  141. {
  142. const directory = join('streaming-playlists', 'hls')
  143. const base = servers[0].servers.buildDirectory(directory)
  144. const n1 = buildUUID()
  145. await createFile(join(base, n1))
  146. badNames[directory] = [ n1 ]
  147. }
  148. }
  149. })
  150. it('Should run prune storage', async function () {
  151. this.timeout(30000)
  152. const env = servers[0].cli.getEnv()
  153. await CLICommand.exec(`echo y | ${env} npm run prune-storage`)
  154. })
  155. it('Should have removed files', async function () {
  156. await assertCountAreOkay(servers)
  157. for (const directory of Object.keys(badNames)) {
  158. for (const name of badNames[directory]) {
  159. await assertNotExists(servers[0], directory, name)
  160. }
  161. }
  162. })
  163. after(async function () {
  164. await cleanupTests(servers)
  165. })
  166. })