UTMDownloadSupportToolsTask.swift 3.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. //
  2. // Copyright © 2022 osy. All rights reserved.
  3. //
  4. // Licensed under the Apache License, Version 2.0 (the "License");
  5. // you may not use this file except in compliance with the License.
  6. // You may obtain a copy of the License at
  7. //
  8. // http://www.apache.org/licenses/LICENSE-2.0
  9. //
  10. // Unless required by applicable law or agreed to in writing, software
  11. // distributed under the License is distributed on an "AS IS" BASIS,
  12. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. // See the License for the specific language governing permissions and
  14. // limitations under the License.
  15. //
  16. import Foundation
  17. /// Downloads support tools ISO
  18. class UTMDownloadSupportToolsTask: UTMDownloadTask {
  19. private let vm: any UTMSpiceVirtualMachine
  20. private static let supportToolsDownloadUrl = URL(string: "https://getutm.app/downloads/utm-guest-tools-latest.iso")!
  21. private var toolsUrl: URL {
  22. fileManager.urls(for: .applicationSupportDirectory, in: .userDomainMask).first!.appendingPathComponent("GuestSupportTools")
  23. }
  24. private var supportToolsLocalUrl: URL {
  25. toolsUrl.appendingPathComponent(Self.supportToolsDownloadUrl.lastPathComponent)
  26. }
  27. @Setting("LastDownloadedGuestTools")
  28. private var lastDownloadGuestTools: Int = 0
  29. var hasExistingSupportTools: Bool {
  30. get async {
  31. guard fileManager.fileExists(atPath: supportToolsLocalUrl.path) else {
  32. return false
  33. }
  34. return await lastModifiedTimestamp <= lastDownloadGuestTools
  35. }
  36. }
  37. init(for vm: any UTMSpiceVirtualMachine) {
  38. self.vm = vm
  39. let name = NSLocalizedString("Windows Guest Support Tools", comment: "UTMDownloadSupportToolsTask")
  40. super.init(for: Self.supportToolsDownloadUrl, named: name)
  41. }
  42. override func processCompletedDownload(at location: URL, response: URLResponse?) async throws -> any UTMVirtualMachine {
  43. if !fileManager.fileExists(atPath: toolsUrl.path) {
  44. try fileManager.createDirectory(at: toolsUrl, withIntermediateDirectories: true)
  45. }
  46. if fileManager.fileExists(atPath: supportToolsLocalUrl.path) {
  47. try fileManager.removeItem(at: supportToolsLocalUrl)
  48. }
  49. try fileManager.moveItem(at: location, to: supportToolsLocalUrl)
  50. lastDownloadGuestTools = lastModifiedTimestamp(for: response) ?? 0
  51. return try await mountTools()
  52. }
  53. func mountTools() async throws -> any UTMVirtualMachine {
  54. for file in await vm.registryEntry.externalDrives.values {
  55. if file.path == supportToolsLocalUrl.path {
  56. throw UTMDownloadSupportToolsTaskError.alreadyMounted
  57. }
  58. }
  59. guard let drive = await vm.config.drives.last(where: { $0.isExternal && $0.imageURL == nil }) else {
  60. throw UTMDownloadSupportToolsTaskError.driveUnavailable
  61. }
  62. try await vm.changeMedium(drive, to: supportToolsLocalUrl)
  63. return vm
  64. }
  65. }
  66. enum UTMDownloadSupportToolsTaskError: Error {
  67. case driveUnavailable
  68. case alreadyMounted
  69. }
  70. extension UTMDownloadSupportToolsTaskError: LocalizedError {
  71. var errorDescription: String? {
  72. switch self {
  73. case .driveUnavailable: return NSLocalizedString("No empty removable drive found. Make sure you have at least one removable drive that is not in use.", comment: "UTMDownloadSupportToolsTaskError")
  74. case .alreadyMounted: return NSLocalizedString("The guest support tools have already been mounted.", comment: "UTMDownloadSupportToolsTaskError")
  75. }
  76. }
  77. }