// // ConnectionService+Migration.swift // Passepartout // // Created by Davide De Rosa on 10/25/18. // Copyright (c) 2018 Davide De Rosa. All rights reserved. // // https://github.com/keeshux // // This file is part of Passepartout. // // Passepartout is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // Passepartout is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with Passepartout. If not, see . // import Foundation import SwiftyBeaver private let log = SwiftyBeaver.self extension ConnectionService { static func migrateJSON(at from: URL, to: URL) { do { let newData = try migrateJSON(at: from) // log.verbose(String(data: newData, encoding: .utf8)!) try newData.write(to: to) } catch let e { log.error("Could not migrate service: \(e)") } } static func migrateJSON(at from: URL) throws -> Data { let data = try Data(contentsOf: from) guard var json = try JSONSerialization.jsonObject(with: data, options: []) as? [String: Any] else { throw ApplicationError.migration } // put migration logic here // TODO: remove this code after 1.0 release let build = json["build"] as? Int ?? 0 if build <= 1084 { try migrateToWrappedSessionConfiguration(&json) try migrateToBaseConfiguration(&json) try migrateToBuildNumber(&json) try migrateHostProfileConfigurations() try migrateSplitProfileSerialization(&json) } if build <= 1107 { let fm = FileManager.default let inbox = fm.userURL(for: .documentDirectory, appending: "Inbox") try? fm.removeItem(at: inbox) } return try JSONSerialization.data(withJSONObject: json, options: []) } // MARK: Atomic migrations static func migrateToWrappedSessionConfiguration(_ json: inout [String: Any]) throws { guard let profiles = json["profiles"] as? [[String: Any]] else { // migrated return } var newProfiles: [[String: Any]] = [] for var container in profiles { guard var hostProfile = container["host"] as? [String: Any] else { newProfiles.append(container) continue } guard var parameters = hostProfile["parameters"] as? [String: Any] else { throw ApplicationError.migration } guard parameters["sessionConfiguration"] == nil else { newProfiles.append(container) continue } migrateSessionConfiguration(in: ¶meters) hostProfile["parameters"] = parameters container["host"] = hostProfile newProfiles.append(container) } json["profiles"] = newProfiles } static func migrateToBaseConfiguration(_ json: inout [String: Any]) throws { guard var baseConfiguration = json["tunnelConfiguration"] as? [String: Any] else { // migrated return } migrateSessionConfiguration(in: &baseConfiguration) json["baseConfiguration"] = baseConfiguration json.removeValue(forKey: "tunnelConfiguration") } static func migrateToBuildNumber(_ json: inout [String: Any]) throws { json["build"] = GroupConstants.App.buildNumber } static func migrateHostProfileConfigurations() throws { let fm = FileManager.default let oldDirectory = fm.userURL(for: .documentDirectory, appending: "Configurations") guard fm.fileExists(atPath: oldDirectory.path) else { return } let newDirectory = fm.userURL(for: .documentDirectory, appending: AppConstants.Store.hostsDirectory) try fm.moveItem(at: oldDirectory, to: newDirectory) let list = try fm.contentsOfDirectory(at: newDirectory, includingPropertiesForKeys: nil, options: []) let prefix = "host." for url in list { let filename = url.lastPathComponent guard filename.hasPrefix(prefix) else { continue } let postPrefixIndex = filename.index(filename.startIndex, offsetBy: prefix.count) let newFilename = String(filename[postPrefixIndex..