MediaLibraryService.swift 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584
  1. /*****************************************************************************
  2. * MediaLibraryService.swift
  3. * VLC for iOS
  4. *****************************************************************************
  5. * Copyright © 2018 VideoLAN. All rights reserved.
  6. * Copyright © 2018 Videolabs
  7. *
  8. * Authors: Soomin Lee <bubu # mikan.io>
  9. *
  10. * Refer to the COPYING file of the official project for license.
  11. *****************************************************************************/
  12. // MARK: - Notification names
  13. extension Notification.Name {
  14. static let VLCNewFileAddedNotification = Notification.Name("NewFileAddedNotification")
  15. }
  16. // For objc
  17. extension NSNotification {
  18. @objc static let VLCNewFileAddedNotification = Notification.Name.VLCNewFileAddedNotification
  19. }
  20. // MARK: -
  21. @objc protocol MediaLibraryObserver: class {
  22. // Video
  23. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  24. didModifyVideos videos: [VLCMLMedia])
  25. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  26. didDeleteMediaWithIds ids: [NSNumber])
  27. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  28. didAddVideos videos: [VLCMLMedia])
  29. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  30. didAddShowEpisodes showEpisodes: [VLCMLMedia])
  31. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  32. thumbnailReady media: VLCMLMedia)
  33. // Audio
  34. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  35. didAddTracks tracks: [VLCMLMedia])
  36. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  37. didModifyTracks tracks: [VLCMLMedia])
  38. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  39. didAddArtists artists: [VLCMLArtist])
  40. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  41. didModifyArtists artists: [VLCMLArtist])
  42. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  43. didDeleteArtistsWithIds artistsIds: [NSNumber])
  44. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  45. didAddAlbums albums: [VLCMLAlbum])
  46. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  47. didModifyAlbums albums: [VLCMLAlbum])
  48. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  49. didDeleteAlbumsWithIds albumsIds: [NSNumber])
  50. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  51. didAddAlbumTracks albumTracks: [VLCMLMedia])
  52. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  53. didAddGenres genres: [VLCMLGenre])
  54. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  55. didModifyGenres genres: [VLCMLGenre])
  56. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  57. didDeleteGenresWithIds genreIds: [NSNumber])
  58. // Playlist
  59. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  60. didAddPlaylists playlists: [VLCMLPlaylist])
  61. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  62. didModifyPlaylists playlists: [VLCMLPlaylist])
  63. @objc optional func medialibrary(_ medialibrary: MediaLibraryService,
  64. didDeletePlaylistsWithIds playlistsIds: [NSNumber])
  65. }
  66. // MARK: -
  67. protocol MediaLibraryMigrationDelegate: class {
  68. func medialibraryDidStartMigration(_ medialibrary: MediaLibraryService)
  69. func medialibraryDidFinishMigration(_ medialibrary: MediaLibraryService)
  70. func medialibraryDidStopMigration(_ medialibrary: MediaLibraryService)
  71. }
  72. // MARK: -
  73. class MediaLibraryService: NSObject {
  74. private static let databaseName: String = "medialibrary.db"
  75. private static let migrationKey: String = "MigratedToVLCMediaLibraryKit"
  76. private var didMigrate = UserDefaults.standard.bool(forKey: MediaLibraryService.migrationKey)
  77. private var didFinishDiscovery = false
  78. // Using ObjectIdentifier to avoid duplication and facilitate
  79. // identification of observing object
  80. private var observers = [ObjectIdentifier: Observer]()
  81. private var medialib: VLCMediaLibrary!
  82. weak var migrationDelegate: MediaLibraryMigrationDelegate?
  83. override init() {
  84. super.init()
  85. medialib = VLCMediaLibrary()
  86. medialib.delegate = self
  87. setupMediaLibrary()
  88. NotificationCenter.default.addObserver(self, selector: #selector(reload),
  89. name: .VLCNewFileAddedNotification, object: nil)
  90. }
  91. }
  92. // MARK: - Private initializers
  93. private extension MediaLibraryService {
  94. private func setupMediaDiscovery(at path: String) {
  95. let mediaFileDiscoverer = VLCMediaFileDiscoverer.sharedInstance()
  96. mediaFileDiscoverer?.directoryPath = path
  97. mediaFileDiscoverer?.addObserver(self)
  98. mediaFileDiscoverer?.startDiscovering()
  99. }
  100. private func setupMediaLibrary() {
  101. guard let documentPath = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first,
  102. let libraryPath = NSSearchPathForDirectoriesInDomains(.libraryDirectory, .userDomainMask, true).first else {
  103. preconditionFailure("MediaLibraryService: Unable to init medialibrary.")
  104. }
  105. setupMediaDiscovery(at: documentPath)
  106. let databasePath = libraryPath + "/MediaLibrary/" + MediaLibraryService.databaseName
  107. let thumbnailPath = libraryPath + "/MediaLibrary/Thumbnails"
  108. do {
  109. try FileManager.default.createDirectory(atPath: thumbnailPath,
  110. withIntermediateDirectories: true)
  111. } catch let error as NSError {
  112. assertionFailure("Failed to create directory: \(error.localizedDescription)")
  113. }
  114. let medialibraryStatus = medialib.setupMediaLibrary(databasePath: databasePath,
  115. thumbnailPath: thumbnailPath)
  116. switch medialibraryStatus {
  117. case .success:
  118. guard medialib.start() else {
  119. assertionFailure("MediaLibraryService: Medialibrary failed to start.")
  120. return
  121. }
  122. medialib.reload()
  123. medialib.discover(onEntryPoint: "file://" + documentPath)
  124. case .alreadyInitialized:
  125. assertionFailure("MediaLibraryService: Medialibrary already initialized.")
  126. case .failed:
  127. preconditionFailure("MediaLibraryService: Failed to setup medialibrary.")
  128. case .dbReset:
  129. // should still start and discover but warn the user that the db has been wipped
  130. assertionFailure("MediaLibraryService: The database was resetted, please re-configure.")
  131. @unknown default:
  132. assertionFailure("MediaLibraryService: unhandled case")
  133. }
  134. }
  135. }
  136. // MARK: - Migration
  137. private extension MediaLibraryService {
  138. func startMigrationIfNeeded() {
  139. guard !didMigrate else {
  140. return
  141. }
  142. migrationDelegate?.medialibraryDidStartMigration(self)
  143. guard migrateToNewMediaLibrary() else {
  144. migrationDelegate?.medialibraryDidStopMigration(self)
  145. return
  146. }
  147. migrationDelegate?.medialibraryDidFinishMigration(self)
  148. }
  149. func migrateMedia(_ oldMedialibrary: MLMediaLibrary) -> Bool {
  150. guard let allFiles = MLFile.allFiles() as? [MLFile] else {
  151. assertionFailure("MediaLibraryService: Migration: Unable to retrieve all files")
  152. return false
  153. }
  154. for media in allFiles {
  155. if let newMedia = fetchMedia(with: media.url) {
  156. newMedia.updateTitle(media.title)
  157. newMedia.setPlayCount(media.playCount.uint32Value)
  158. newMedia.setMetadataOf(.progress, intValue: media.lastPosition.int64Value)
  159. newMedia.setMetadataOf(.seen, intValue: media.unread.int64Value)
  160. // Only delete files that are not in playlist
  161. if media.labels.isEmpty {
  162. oldMedialibrary.remove(media)
  163. }
  164. }
  165. }
  166. oldMedialibrary.save()
  167. return true
  168. }
  169. // This private method migrates old playlist and removes file and playlist
  170. // from the old medialibrary.
  171. // Note: This removes **only** files that are in a playlist
  172. func migratePlaylists(_ oldMedialibrary: MLMediaLibrary) -> Bool {
  173. guard let allLabels = MLLabel.allLabels() as? [MLLabel] else {
  174. assertionFailure("MediaLibraryService: Migration: Unable to retrieve all labels")
  175. return false
  176. }
  177. for label in allLabels {
  178. guard let newPlaylist = createPlaylist(with: label.name) else {
  179. assertionFailure("MediaLibraryService: Migration: Unable to create playlist.")
  180. continue
  181. }
  182. guard let files = label.files as? Set<MLFile> else {
  183. assertionFailure("MediaLibraryService: Migration: Unable to retrieve files from label")
  184. oldMedialibrary.remove(label)
  185. continue
  186. }
  187. for file in files {
  188. if let newMedia = fetchMedia(with: file.url) {
  189. if newPlaylist.appendMedia(withIdentifier: newMedia.identifier()) {
  190. oldMedialibrary.remove(file)
  191. }
  192. }
  193. }
  194. oldMedialibrary.remove(label)
  195. }
  196. oldMedialibrary.save()
  197. return true
  198. }
  199. func migrateToNewMediaLibrary() -> Bool {
  200. guard let oldMedialibrary = MLMediaLibrary.sharedMediaLibrary() as? MLMediaLibrary else {
  201. assertionFailure("MediaLibraryService: Migration: Unable to retrieve old medialibrary")
  202. return false
  203. }
  204. if migrateMedia(oldMedialibrary) && migratePlaylists(oldMedialibrary) {
  205. UserDefaults.standard.set(true, forKey: MediaLibraryService.migrationKey)
  206. didMigrate = true
  207. return true
  208. }
  209. return false
  210. }
  211. }
  212. // MARK: - Observer
  213. private extension MediaLibraryService {
  214. struct Observer {
  215. weak var observer: MediaLibraryObserver?
  216. }
  217. }
  218. extension MediaLibraryService {
  219. func addObserver(_ observer: MediaLibraryObserver) {
  220. let identifier = ObjectIdentifier(observer)
  221. observers[identifier] = Observer(observer: observer)
  222. }
  223. func removeObserver(_ observer: MediaLibraryObserver) {
  224. let identifier = ObjectIdentifier(observer)
  225. observers.removeValue(forKey: identifier)
  226. }
  227. }
  228. // MARK: - Helpers
  229. @objc extension MediaLibraryService {
  230. @objc func reload() {
  231. medialib.reload()
  232. }
  233. /// Returns number of *ALL* files(audio and video) present in the medialibrary database
  234. func numberOfFiles() -> Int {
  235. return (medialib.audioFiles()?.count ?? 0) + (medialib.videoFiles()?.count ?? 0)
  236. }
  237. /// Returns *ALL* file found for a specified VLCMLMediaType
  238. ///
  239. /// - Parameter type: Type of the media
  240. /// - Returns: Array of VLCMLMedia
  241. func media(ofType type: VLCMLMediaType,
  242. sortingCriteria sort: VLCMLSortingCriteria = .alpha,
  243. desc: Bool = false) -> [VLCMLMedia] {
  244. return type == .video ? medialib.videoFiles(with: sort, desc: desc) ?? []
  245. : medialib.audioFiles(with: sort, desc: desc) ?? []
  246. }
  247. @objc func fetchMedia(with mrl: URL?) -> VLCMLMedia? {
  248. guard let mrl = mrl else {
  249. assertionFailure("MedialibraryService: no mrl")
  250. return nil
  251. }
  252. return medialib.media(withMrl: mrl)
  253. }
  254. @objc func media(for identifier: VLCMLIdentifier) -> VLCMLMedia? {
  255. return medialib.media(withIdentifier: identifier)
  256. }
  257. func savePlaybackState(from player: VLCPlaybackController) {
  258. let media: VLCMedia? = player.currentlyPlayingMedia
  259. guard let mlMedia = fetchMedia(with: media?.url.absoluteURL) else {
  260. // we opened a url and not a local file
  261. return
  262. }
  263. mlMedia.isNew = false
  264. mlMedia.progress = player.playbackPosition
  265. mlMedia.audioTrackIndex = Int64(player.indexOfCurrentAudioTrack)
  266. mlMedia.subtitleTrackIndex = Int64(player.indexOfCurrentSubtitleTrack)
  267. mlMedia.chapterIndex = Int64(player.indexOfCurrentChapter)
  268. mlMedia.titleIndex = Int64(player.indexOfCurrentTitle)
  269. //create a new thumbnail
  270. }
  271. }
  272. // MARK: - Audio methods
  273. @objc extension MediaLibraryService {
  274. func artists(sortingCriteria sort: VLCMLSortingCriteria = .alpha,
  275. desc: Bool = false, listAll all: Bool = false) -> [VLCMLArtist] {
  276. return medialib.artists(with: sort, desc: desc, all: all) ?? []
  277. }
  278. func albums(sortingCriteria sort: VLCMLSortingCriteria = .alpha,
  279. desc: Bool = false) -> [VLCMLAlbum] {
  280. return medialib.albums(with: sort, desc: desc) ?? []
  281. }
  282. }
  283. // MARK: - Video methods
  284. extension MediaLibraryService {
  285. func requestThumbnail(for media: [VLCMLMedia]) {
  286. media.forEach() {
  287. guard !$0.isThumbnailGenerated() else { return }
  288. if !medialib.requestThumbnail(for: $0) {
  289. assertionFailure("MediaLibraryService: Failed to generate thumbnail for: \($0.identifier())")
  290. }
  291. }
  292. }
  293. }
  294. // MARK: - Playlist methods
  295. @objc extension MediaLibraryService {
  296. func createPlaylist(with name: String) -> VLCMLPlaylist? {
  297. return medialib.createPlaylist(withName: name)
  298. }
  299. func deletePlaylist(with identifier: VLCMLIdentifier) -> Bool {
  300. return medialib.deletePlaylist(withIdentifier: identifier)
  301. }
  302. func playlists(sortingCriteria sort: VLCMLSortingCriteria = .default,
  303. desc: Bool = false) -> [VLCMLPlaylist] {
  304. return medialib.playlists(with: sort, desc: desc) ?? []
  305. }
  306. }
  307. // MARK: - Genre methods
  308. extension MediaLibraryService {
  309. func genres(sortingCriteria sort: VLCMLSortingCriteria = .alpha,
  310. desc: Bool = false) -> [VLCMLGenre] {
  311. return medialib.genres(with: sort, desc: desc) ?? []
  312. }
  313. }
  314. // MARK: - VLCMediaFileDiscovererDelegate
  315. extension MediaLibraryService: VLCMediaFileDiscovererDelegate {
  316. func mediaFileAdded(_ filePath: String!, loading isLoading: Bool) {
  317. guard !isLoading else {
  318. return
  319. }
  320. /* exclude media files from backup (QA1719) */
  321. var excludeURL = URL(fileURLWithPath: filePath)
  322. var resourceValue = URLResourceValues()
  323. resourceValue.isExcludedFromBackup = true
  324. do {
  325. try excludeURL.setResourceValues(resourceValue)
  326. } catch let error {
  327. assertionFailure("MediaLibraryService: VLCMediaFileDiscovererDelegate: \(error.localizedDescription)")
  328. }
  329. reload()
  330. }
  331. func mediaFileDeleted(_ filePath: String!) {
  332. reload()
  333. }
  334. }
  335. // MARK: - VLCMediaLibraryDelegate - Media
  336. extension MediaLibraryService: VLCMediaLibraryDelegate {
  337. func medialibrary(_ medialibrary: VLCMediaLibrary, didAddMedia media: [VLCMLMedia]) {
  338. media.forEach { $0.updateCoreSpotlightEntry() }
  339. let videos = media.filter {( $0.type() == .video )}
  340. let tracks = media.filter {( $0.type() == .audio )}
  341. // thumbnails only for videos
  342. requestThumbnail(for: videos)
  343. for observer in observers {
  344. observer.value.observer?.medialibrary?(self, didAddVideos: videos)
  345. observer.value.observer?.medialibrary?(self, didAddTracks: tracks)
  346. }
  347. }
  348. func medialibrary(_ medialibrary: VLCMediaLibrary, didModifyMedia media: [VLCMLMedia]) {
  349. media.forEach { $0.updateCoreSpotlightEntry() }
  350. let showEpisodes = media.filter {( $0.subtype() == .showEpisode )}
  351. let albumTrack = media.filter {( $0.subtype() == .albumTrack )}
  352. let videos = media.filter {( $0.type() == .video)}
  353. let tracks = media.filter {( $0.type() == .audio)}
  354. // Shows and albumtracks are known only after when the medialibrary calls didModifyMedia
  355. for observer in observers {
  356. observer.value.observer?.medialibrary?(self, didAddShowEpisodes: showEpisodes)
  357. observer.value.observer?.medialibrary?(self, didAddAlbumTracks: albumTrack)
  358. observer.value.observer?.medialibrary?(self, didModifyVideos: videos)
  359. observer.value.observer?.medialibrary?(self, didModifyTracks: tracks)
  360. }
  361. }
  362. func medialibrary(_ medialibrary: VLCMediaLibrary, didDeleteMediaWithIds mediaIds: [NSNumber]) {
  363. var stringIds = [String]()
  364. mediaIds.forEach { stringIds.append("\($0)") }
  365. CSSearchableIndex.default().deleteSearchableItems(withIdentifiers: stringIds, completionHandler: nil)
  366. for observer in observers {
  367. observer.value.observer?.medialibrary?(self, didDeleteMediaWithIds: mediaIds)
  368. }
  369. }
  370. func medialibrary(_ medialibrary: VLCMediaLibrary, thumbnailReadyFor media: VLCMLMedia, withSuccess success: Bool) {
  371. for observer in observers {
  372. observer.value.observer?.medialibrary?(self, thumbnailReady: media)
  373. }
  374. }
  375. }
  376. // MARK: - VLCMediaLibraryDelegate - Artists
  377. extension MediaLibraryService {
  378. func medialibrary(_ medialibrary: VLCMediaLibrary, didAdd artists: [VLCMLArtist]) {
  379. for observer in observers {
  380. observer.value.observer?.medialibrary?(self, didAddArtists: artists)
  381. }
  382. }
  383. func medialibrary(_ medialibrary: VLCMediaLibrary, didModifyArtists artists: [VLCMLArtist]) {
  384. for observer in observers {
  385. observer.value.observer?.medialibrary?(self, didModifyArtists: artists)
  386. }
  387. }
  388. func medialibrary(_ medialibrary: VLCMediaLibrary, didDeleteArtistsWithIds artistsIds: [NSNumber]) {
  389. for observer in observers {
  390. observer.value.observer?.medialibrary?(self, didDeleteArtistsWithIds: artistsIds)
  391. }
  392. }
  393. }
  394. // MARK: - VLCMediaLibraryDelegate - Albums
  395. extension MediaLibraryService {
  396. func medialibrary(_ medialibrary: VLCMediaLibrary, didAdd albums: [VLCMLAlbum]) {
  397. for observer in observers {
  398. observer.value.observer?.medialibrary?(self, didAddAlbums: albums)
  399. }
  400. }
  401. func medialibrary(_ medialibrary: VLCMediaLibrary, didModify albums: [VLCMLAlbum]) {
  402. for observer in observers {
  403. observer.value.observer?.medialibrary?(self, didAddAlbums: albums)
  404. }
  405. }
  406. func medialibrary(_ medialibrary: VLCMediaLibrary, didDeleteAlbumsWithIds albumsIds: [NSNumber]) {
  407. for observer in observers {
  408. observer.value.observer?.medialibrary?(self, didDeleteAlbumsWithIds: albumsIds)
  409. }
  410. }
  411. }
  412. // MARK: - VLCMediaLibraryDelegate - Genres
  413. extension MediaLibraryService {
  414. func medialibrary(_ medialibrary: VLCMediaLibrary, didAdd genres: [VLCMLGenre]) {
  415. for observer in observers {
  416. observer.value.observer?.medialibrary?(self, didAddGenres: genres)
  417. }
  418. }
  419. func medialibrary(_ medialibrary: VLCMediaLibrary, didModifyGenres genres: [VLCMLGenre]) {
  420. for observer in observers {
  421. observer.value.observer?.medialibrary?(self, didModifyGenres: genres)
  422. }
  423. }
  424. func medialibrary(_ medialibrary: VLCMediaLibrary,
  425. didDeleteGenresWithIds genresIds: [NSNumber]) {
  426. for observer in observers {
  427. observer.value.observer?.medialibrary?(self, didDeleteGenresWithIds: genresIds)
  428. }
  429. }
  430. }
  431. // MARK: - VLCMediaLibraryDelegate - Playlists
  432. extension MediaLibraryService {
  433. func medialibrary(_ medialibrary: VLCMediaLibrary, didAdd playlists: [VLCMLPlaylist]) {
  434. for observer in observers {
  435. observer.value.observer?.medialibrary?(self, didAddPlaylists: playlists)
  436. }
  437. }
  438. func medialibrary(_ medialibrary: VLCMediaLibrary, didModifyPlaylists playlists: [VLCMLPlaylist]) {
  439. for observer in observers {
  440. observer.value.observer?.medialibrary?(self, didModifyPlaylists: playlists)
  441. }
  442. }
  443. func medialibrary(_ medialibrary: VLCMediaLibrary, didDeletePlaylistsWithIds playlistsIds: [NSNumber]) {
  444. for observer in observers {
  445. observer.value.observer?.medialibrary?(self, didDeletePlaylistsWithIds: playlistsIds)
  446. }
  447. }
  448. }
  449. // MARK: - VLCMediaLibraryDelegate - Discovery
  450. extension MediaLibraryService {
  451. func medialibrary(_ medialibrary: VLCMediaLibrary, didStartDiscovery entryPoint: String) {
  452. }
  453. func medialibrary(_ medialibrary: VLCMediaLibrary, didCompleteDiscovery entryPoint: String) {
  454. didFinishDiscovery = true
  455. }
  456. func medialibrary(_ medialibrary: VLCMediaLibrary, didProgressDiscovery entryPoint: String) {
  457. }
  458. func medialibrary(_ medialibrary: VLCMediaLibrary, didUpdateParsingStatsWithPercent percent: UInt32) {
  459. if didFinishDiscovery && percent == 100 {
  460. startMigrationIfNeeded()
  461. }
  462. }
  463. }