passforios/passKit/Models/PasswordStore.swift

570 lines
22 KiB
Swift
Raw Normal View History

2017-01-19 21:15:47 +08:00
//
// PasswordStore.swift
2021-08-28 07:32:31 +02:00
// passKit
2017-01-19 21:15:47 +08:00
//
// Created by Mingshen Sun on 19/1/2017.
// Copyright © 2017 Bob Sun. All rights reserved.
//
import CoreData
import Foundation
import KeychainAccess
import ObjectiveGit
import SwiftyUserDefaults
import UIKit
2017-01-19 21:15:47 +08:00
public class PasswordStore {
public static let shared = PasswordStore()
private static let dateFormatter: DateFormatter = {
let dateFormatter = DateFormatter()
dateFormatter.dateStyle = .short
dateFormatter.timeStyle = .short
return dateFormatter
}()
2020-04-11 23:23:38 -07:00
public var storeURL: URL
public var tempStoreURL: URL {
URL(fileURLWithPath: "\(storeURL.path)-temp")
2020-04-11 23:23:38 -07:00
}
public var storeRepository: GTRepository?
2019-06-09 22:18:54 -07:00
public var gitSignatureForNow: GTSignature? {
let gitSignatureName = Defaults.gitSignatureName ?? Globals.gitSignatureDefaultName
let gitSignatureEmail = Defaults.gitSignatureEmail ?? Globals.gitSignatureDefaultEmail
return GTSignature(name: gitSignatureName, email: gitSignatureEmail, time: Date())
2017-03-16 00:38:38 +08:00
}
public var gitPassword: String? {
get {
AppKeychain.shared.get(for: Globals.gitPassword)
}
set {
AppKeychain.shared.add(string: newValue, for: Globals.gitPassword)
}
}
public var gitSSHPrivateKeyPassphrase: String? {
get {
AppKeychain.shared.get(for: Globals.gitSSHPrivateKeyPassphrase)
}
set {
AppKeychain.shared.add(string: newValue, for: Globals.gitSSHPrivateKeyPassphrase)
}
}
private let fileManager = FileManager.default
2025-01-25 15:40:12 -08:00
private lazy var context: NSManagedObjectContext = PersistenceController.shared.viewContext()
public var numberOfPasswords: Int {
2025-01-25 15:40:12 -08:00
PasswordEntity.totalNumber(in: context)
}
public var sizeOfRepositoryByteCount: UInt64 {
(try? fileManager.allocatedSizeOfDirectoryAtURL(directoryURL: storeURL)) ?? 0
}
public var numberOfLocalCommits: Int {
(try? getLocalCommits()).map(\.count) ?? 0
}
public var lastSyncedTime: Date? {
Defaults.lastSyncedTime
}
2021-01-17 19:49:05 -08:00
public var lastSyncedTimeString: String {
guard let date = lastSyncedTime else {
return "SyncAgain?".localize()
}
let formatter = DateFormatter()
formatter.dateStyle = .medium
formatter.timeStyle = .short
return formatter.string(from: date)
}
public var numberOfCommits: UInt? {
storeRepository?.numberOfCommits(inCurrentBranch: nil)
}
2020-04-11 23:23:38 -07:00
2025-01-25 15:40:12 -08:00
init(url: URL = Globals.repositoryURL) {
self.storeURL = url
2020-04-11 23:23:38 -07:00
2019-07-19 01:46:56 +08:00
// Migration
importExistingKeysIntoKeychain()
2020-04-11 23:23:38 -07:00
2017-01-23 16:29:36 +08:00
do {
if fileManager.fileExists(atPath: storeURL.path) {
try self.storeRepository = GTRepository(url: storeURL)
}
2017-01-23 16:29:36 +08:00
} catch {
print(error)
2017-01-19 21:15:47 +08:00
}
}
private func importExistingKeysIntoKeychain() {
2019-07-19 01:46:56 +08:00
// App Store update: v0.5.1 -> v0.6.0
try? KeyFileManager(keyType: PGPKey.PUBLIC, keyPath: Globals.pgpPublicKeyPath).importKeyFromFileSharing()
try? KeyFileManager(keyType: PGPKey.PRIVATE, keyPath: Globals.pgpPrivateKeyPath).importKeyFromFileSharing()
try? KeyFileManager(keyType: SSHKey.PRIVATE, keyPath: Globals.gitSSHPrivateKeyPath).importKeyFromFileSharing()
Defaults.remove(\.pgpPublicKeyArmor)
Defaults.remove(\.pgpPrivateKeyArmor)
Defaults.remove(\.gitSSHPrivateKeyArmor)
}
public func repositoryExists() -> Bool {
2025-01-25 15:40:12 -08:00
fileManager.fileExists(atPath: Globals.repositoryURL.path)
}
public func passwordExisted(password: Password) -> Bool {
2025-01-25 15:40:12 -08:00
PasswordEntity.exists(password: password, in: context)
}
public func getPasswordEntity(by path: String, isDir: Bool) -> PasswordEntity? {
2025-01-25 15:40:12 -08:00
PasswordEntity.fetch(by: path, isDir: isDir, in: context)
}
public func cloneRepository(
remoteRepoURL: URL,
branchName: String,
options: [AnyHashable: Any]? = nil,
transferProgressBlock: @escaping (UnsafePointer<git_transfer_progress>, UnsafeMutablePointer<ObjCBool>) -> Void = { _, _ in },
checkoutProgressBlock: @escaping (String, UInt, UInt) -> Void = { _, _, _ in }
) throws {
try? fileManager.removeItem(at: storeURL)
try? fileManager.removeItem(at: tempStoreURL)
gitPassword = nil
gitSSHPrivateKeyPassphrase = nil
2017-02-04 14:59:55 +08:00
do {
storeRepository = try GTRepository.clone(
from: remoteRepoURL,
toWorkingDirectory: tempStoreURL,
options: options,
transferProgressBlock: transferProgressBlock
)
try fileManager.moveItem(at: tempStoreURL, to: storeURL)
2017-04-28 20:33:41 -07:00
storeRepository = try GTRepository(url: storeURL)
if (try storeRepository?.currentBranch().name) != branchName {
try checkoutAndChangeBranch(withName: branchName, progressBlock: checkoutProgressBlock)
}
2017-02-04 14:59:55 +08:00
} catch {
Defaults.lastSyncedTime = nil
DispatchQueue.main.async {
2025-01-25 15:40:12 -08:00
self.deleteCoreData()
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
}
throw (error)
2017-02-04 14:59:55 +08:00
}
Defaults.lastSyncedTime = Date()
DispatchQueue.main.async {
2025-01-25 15:40:12 -08:00
self.deleteCoreData()
self.initPasswordEntityCoreData()
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
}
2017-01-23 16:29:36 +08:00
}
private func checkoutAndChangeBranch(withName localBranchName: String, progressBlock: @escaping (String, UInt, UInt) -> Void) throws {
guard let storeRepository else {
throw AppError.repositoryNotSet
2019-01-06 20:10:47 +01:00
}
let remoteBranchName = "origin/\(localBranchName)"
let remoteBranch = try storeRepository.lookUpBranch(withName: remoteBranchName, type: .remote, success: nil)
2019-01-06 20:10:47 +01:00
guard let remoteBranchOid = remoteBranch.oid else {
throw AppError.repositoryRemoteBranchNotFound(branchName: remoteBranchName)
2019-01-06 20:10:47 +01:00
}
let localBranch = try storeRepository.createBranchNamed(localBranchName, from: remoteBranchOid, message: nil)
try localBranch.updateTrackingBranch(remoteBranch)
let checkoutOptions = GTCheckoutOptions(strategy: .force, progressBlock: progressBlock)
2019-01-06 20:10:47 +01:00
try storeRepository.checkoutReference(localBranch.reference, options: checkoutOptions)
try storeRepository.moveHEAD(to: localBranch.reference)
}
public func pullRepository(
options: [String: Any],
progressBlock: @escaping (UnsafePointer<git_transfer_progress>, UnsafeMutablePointer<ObjCBool>) -> Void = { _, _ in }
) throws {
guard let storeRepository else {
throw AppError.repositoryNotSet
}
2018-11-16 23:08:35 -08:00
let remote = try GTRemote(name: "origin", in: storeRepository)
try storeRepository.pull(storeRepository.currentBranch(), from: remote, withOptions: options, progress: progressBlock)
Defaults.lastSyncedTime = Date()
setAllSynced()
DispatchQueue.main.async {
2025-01-25 15:40:12 -08:00
self.deleteCoreData()
self.initPasswordEntityCoreData()
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
}
2017-01-19 21:15:47 +08:00
}
2025-01-25 15:40:12 -08:00
private func initPasswordEntityCoreData() {
PasswordEntity.initPasswordEntityCoreData(url: storeURL, in: context)
saveUpdatedContext()
2017-01-19 21:15:47 +08:00
}
public func getRecentCommits(count: Int) throws -> [GTCommit] {
guard let storeRepository else {
2017-03-19 10:36:59 -07:00
return []
}
2017-02-22 18:43:19 +08:00
var commits = [GTCommit]()
2017-04-30 18:29:47 -05:00
let enumerator = try GTEnumerator(repository: storeRepository)
if let targetOID = try storeRepository.headReference().targetOID {
try enumerator.pushSHA(targetOID.sha)
}
for _ in 0 ..< count {
2018-10-11 13:41:47 +08:00
if let commit = try? enumerator.nextObject(withSuccess: nil) {
commits.append(commit)
}
2017-02-22 18:43:19 +08:00
}
return commits
}
public func fetchPasswordEntityCoreData(parent: PasswordEntity?) -> [PasswordEntity] {
2025-01-25 15:40:12 -08:00
PasswordEntity.fetch(by: parent, in: context)
2017-01-19 21:15:47 +08:00
}
2025-01-25 15:40:12 -08:00
public func fetchPasswordEntityCoreData(withDir _: Bool) -> [PasswordEntity] {
PasswordEntity.fetchAllPassword(in: context)
}
public func fetchUnsyncedPasswords() -> [PasswordEntity] {
2025-01-25 15:40:12 -08:00
PasswordEntity.fetchUnsynced(in: context)
}
public func fetchPasswordEntity(with path: String) -> PasswordEntity? {
2025-01-25 15:40:12 -08:00
PasswordEntity.fetch(by: path, in: context)
}
public func setAllSynced() {
2025-01-25 15:40:12 -08:00
_ = PasswordEntity.updateAllToSynced(in: context)
saveUpdatedContext()
}
2025-01-25 15:40:12 -08:00
public func getLatestUpdateInfo(path: String) -> String {
guard let storeRepository else {
2019-01-14 20:57:45 +01:00
return "Unknown".localize()
}
2025-01-25 15:40:12 -08:00
guard let blameHunks = try? storeRepository.blame(withFile: path, options: nil).hunks else {
return "Unknown".localize()
}
guard let latestCommitTime = blameHunks.map({ $0.finalSignature?.time?.timeIntervalSince1970 ?? 0 }).max() else {
return "Unknown".localize()
}
let lastCommitDate = Date(timeIntervalSince1970: latestCommitTime)
if Date().timeIntervalSince(lastCommitDate) <= 60 {
return "JustNow".localize()
}
return Self.dateFormatter.string(from: lastCommitDate)
}
private func gitAdd(path: String) throws {
guard let storeRepository else {
throw AppError.repositoryNotSet
}
2017-04-30 18:29:47 -05:00
try storeRepository.index().addFile(path)
try storeRepository.index().write()
}
private func gitRm(path: String) throws {
guard let storeRepository else {
throw AppError.repositoryNotSet
2017-02-13 01:15:42 +08:00
}
2017-04-30 18:29:47 -05:00
let url = storeURL.appendingPathComponent(path)
if fileManager.fileExists(atPath: url.path) {
try fileManager.removeItem(at: url)
}
2017-04-30 18:29:47 -05:00
try storeRepository.index().removeFile(path)
try storeRepository.index().write()
2017-04-26 20:28:15 -07:00
}
2017-04-26 20:28:15 -07:00
private func deleteDirectoryTree(at url: URL) throws {
2025-01-25 15:40:12 -08:00
var tempURL = url.deletingLastPathComponent()
while try fileManager.contentsOfDirectory(atPath: tempURL.path).isEmpty {
try fileManager.removeItem(at: tempURL)
2017-04-26 20:28:15 -07:00
tempURL.deleteLastPathComponent()
}
}
2017-04-26 20:28:15 -07:00
private func createDirectoryTree(at url: URL) throws {
2025-01-25 15:40:12 -08:00
let tempURL = url.deletingLastPathComponent()
try fileManager.createDirectory(at: tempURL, withIntermediateDirectories: true)
2017-02-13 01:15:42 +08:00
}
private func gitMv(from: String, to: String) throws {
2017-04-30 18:29:47 -05:00
let fromURL = storeURL.appendingPathComponent(from)
let toURL = storeURL.appendingPathComponent(to)
try fileManager.moveItem(at: fromURL, to: toURL)
try gitAdd(path: to)
try gitRm(path: from)
}
private func gitCommit(message: String) throws -> GTCommit? {
guard let storeRepository else {
throw AppError.repositoryNotSet
}
2017-04-30 18:29:47 -05:00
let newTree = try storeRepository.index().writeTree()
let headReference = try storeRepository.headReference()
let commitEnum = try GTEnumerator(repository: storeRepository)
try commitEnum.pushSHA(headReference.targetOID!.sha)
let parent = commitEnum.nextObject() as! GTCommit
2019-06-09 22:18:54 -07:00
guard let signature = gitSignatureForNow else {
throw AppError.gitCreateSignature
2019-06-09 22:18:54 -07:00
}
return try storeRepository.createCommit(with: newTree, message: message, author: signature, committer: signature, parents: [parent], updatingReferenceNamed: headReference.name)
}
private func getLocalBranch(withName branchName: String) throws -> GTBranch? {
guard let storeRepository else {
throw AppError.repositoryNotSet
}
let reference = GTBranch.localNamePrefix().appending(branchName)
2017-04-30 18:29:47 -05:00
let branches = try storeRepository.branches(withPrefix: reference)
return branches.first
}
public func pushRepository(
options: [String: Any],
transferProgressBlock: @escaping (UInt32, UInt32, Int, UnsafeMutablePointer<ObjCBool>) -> Void = { _, _, _, _ in }
) throws {
guard let storeRepository else {
throw AppError.repositoryNotSet
}
if let branch = try getLocalBranch(withName: Defaults.gitBranchName) {
let remote = try GTRemote(name: "origin", in: storeRepository)
try storeRepository.push(branch, to: remote, withOptions: options, progress: transferProgressBlock)
}
if numberOfLocalCommits != 0 {
throw AppError.gitPushNotSuccessful
2017-04-28 20:33:41 -07:00
}
2017-02-10 22:15:01 +08:00
}
private func addPasswordEntities(password: Password) throws -> PasswordEntity? {
guard !passwordExisted(password: password) else {
throw AppError.passwordDuplicated
}
var paths: [String] = []
2025-01-25 15:40:12 -08:00
var path = password.path
while !path.isEmpty {
paths.append(path)
path = (path as NSString).deletingLastPathComponent
}
2025-01-25 15:40:12 -08:00
var parentPasswordEntity: PasswordEntity?
2025-01-25 15:40:12 -08:00
for (index, path) in paths.reversed().enumerated() {
if index == paths.count - 1 {
let passwordEntity = PasswordEntity.insert(name: password.name, path: path, isDir: false, into: context)
passwordEntity.parent = parentPasswordEntity
parentPasswordEntity = passwordEntity
} else {
2025-01-25 15:40:12 -08:00
if let passwordEntity = PasswordEntity.fetch(by: path, isDir: true, in: context) {
passwordEntity.isSynced = false
parentPasswordEntity = passwordEntity
} else {
2025-01-25 15:40:12 -08:00
let name = (path as NSString).lastPathComponent
let passwordEntity = PasswordEntity.insert(name: name, path: path, isDir: true, into: context)
passwordEntity.parent = parentPasswordEntity
parentPasswordEntity = passwordEntity
}
}
2017-02-10 22:15:01 +08:00
}
saveUpdatedContext()
return parentPasswordEntity
}
public func add(password: Password, keyID: String? = nil) throws -> PasswordEntity? {
2025-01-25 15:40:12 -08:00
let saveURL = storeURL.appendingPathComponent(password.path)
try createDirectoryTree(at: saveURL)
try encrypt(password: password, keyID: keyID).write(to: saveURL)
2025-01-25 15:40:12 -08:00
try gitAdd(path: password.path)
_ = try gitCommit(message: "AddPassword.".localize(password.path))
let newPasswordEntity = try addPasswordEntities(password: password)
2017-03-21 13:16:25 -07:00
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
return newPasswordEntity
}
2017-04-26 20:28:15 -07:00
public func delete(passwordEntity: PasswordEntity) throws {
2025-01-25 15:40:12 -08:00
let deletedFileURL = storeURL.appendingPathComponent(passwordEntity.path)
try gitRm(path: passwordEntity.path)
2018-04-12 00:52:10 +08:00
try deletePasswordEntities(passwordEntity: passwordEntity)
try deleteDirectoryTree(at: deletedFileURL)
2025-01-25 15:40:12 -08:00
_ = try gitCommit(message: "RemovePassword.".localize(passwordEntity.path))
2017-04-26 20:28:15 -07:00
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
}
public func edit(passwordEntity: PasswordEntity, password: Password, keyID: String? = nil) throws -> PasswordEntity? {
var newPasswordEntity: PasswordEntity? = passwordEntity
2025-01-25 15:40:12 -08:00
let url = storeURL.appendingPathComponent(passwordEntity.path)
if password.changed & PasswordChange.content.rawValue != 0 {
2025-01-25 15:40:12 -08:00
try encrypt(password: password, keyID: keyID).write(to: url)
try gitAdd(path: password.path)
_ = try gitCommit(message: "EditPassword.".localize(passwordEntity.path))
2017-04-26 20:28:15 -07:00
newPasswordEntity = passwordEntity
2025-01-25 15:40:12 -08:00
newPasswordEntity?.isSynced = false
}
if password.changed & PasswordChange.path.rawValue != 0 {
2019-06-09 22:18:54 -07:00
let deletedFileURL = url
2017-04-26 20:28:15 -07:00
// add
2025-01-25 15:40:12 -08:00
let newFileURL = storeURL.appendingPathComponent(password.path)
try createDirectoryTree(at: newFileURL)
2017-04-26 20:28:15 -07:00
newPasswordEntity = try addPasswordEntities(password: password)
2017-04-26 20:28:15 -07:00
// mv
2025-01-25 15:40:12 -08:00
try gitMv(from: passwordEntity.path, to: password.path)
2017-04-26 20:28:15 -07:00
// delete
try deleteDirectoryTree(at: deletedFileURL)
try deletePasswordEntities(passwordEntity: passwordEntity)
2025-01-25 15:40:12 -08:00
_ = try gitCommit(message: "RenamePassword.".localize(passwordEntity.path, password.path))
}
2025-01-25 15:40:12 -08:00
saveUpdatedContext()
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
return newPasswordEntity
}
private func deletePasswordEntities(passwordEntity: PasswordEntity) throws {
2025-01-25 15:40:12 -08:00
PasswordEntity.deleteRecursively(entity: passwordEntity, in: context)
saveUpdatedContext()
2017-03-21 13:16:25 -07:00
}
public func saveUpdatedContext() {
2025-01-25 15:40:12 -08:00
PersistenceController.shared.save()
2017-02-13 01:15:42 +08:00
}
2025-01-25 15:40:12 -08:00
public func deleteCoreData() {
PasswordEntity.deleteAll(in: context)
PersistenceController.shared.save()
}
public func eraseStoreData() {
// Delete files.
try? fileManager.removeItem(at: storeURL)
try? fileManager.removeItem(at: tempStoreURL)
// Delete core data.
2025-01-25 15:40:12 -08:00
deleteCoreData()
// Clean up variables inside PasswordStore.
2017-02-13 14:30:38 +08:00
storeRepository = nil
// Broadcast.
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
NotificationCenter.default.post(name: .passwordStoreErased, object: nil)
2017-02-07 16:45:14 +08:00
}
public func erase() {
eraseStoreData()
// Delete PGP key, SSH key and other secrets from the keychain.
AppKeychain.shared.removeAllContent()
// Delete default settings.
Defaults.removeAll()
// Delete cache explicitly.
PasscodeLock.shared.delete()
PGPAgent.shared.uninitKeys()
}
2018-12-09 16:59:07 -08:00
// return the number of discarded commits
public func reset() throws -> Int {
guard let storeRepository else {
throw AppError.repositoryNotSet
2017-04-30 18:29:47 -05:00
}
// get a list of local commits
let localCommits = try getLocalCommits()
if localCommits.isEmpty {
return 0 // no new commit
}
// get the oldest local commit
guard let firstLocalCommit = localCommits.last,
2020-11-07 12:06:28 +01:00
firstLocalCommit.parents.count == 1,
let newHead = firstLocalCommit.parents.first else {
throw AppError.gitReset
}
try storeRepository.reset(to: newHead, resetType: .hard)
setAllSynced()
2025-01-25 15:40:12 -08:00
deleteCoreData()
initPasswordEntityCoreData()
NotificationCenter.default.post(name: .passwordStoreUpdated, object: nil)
NotificationCenter.default.post(name: .passwordStoreChangeDiscarded, object: nil)
return localCommits.count
}
private func getLocalCommits() throws -> [GTCommit] {
guard let storeRepository else {
throw AppError.repositoryNotSet
}
2019-01-06 20:10:47 +01:00
// get the remote branch
let remoteBranchName = Defaults.gitBranchName
2019-01-06 20:10:47 +01:00
guard let remoteBranch = try storeRepository.remoteBranches().first(where: { $0.shortName == remoteBranchName }) else {
throw AppError.repositoryRemoteBranchNotFound(branchName: remoteBranchName)
}
// check oid before calling localCommitsRelative
2019-01-06 20:10:47 +01:00
guard remoteBranch.oid != nil else {
throw AppError.repositoryRemoteBranchNotFound(branchName: remoteBranchName)
}
// get a list of local commits
2019-01-06 20:10:47 +01:00
return try storeRepository.localCommitsRelative(toRemoteBranch: remoteBranch)
}
public func decrypt(passwordEntity: PasswordEntity, keyID: String? = nil, requestPGPKeyPassphrase: @escaping (String) -> String) throws -> Password {
2025-01-25 15:40:12 -08:00
let url = storeURL.appendingPathComponent(passwordEntity.path)
let encryptedData = try Data(contentsOf: url)
2021-01-07 21:58:38 -08:00
let data: Data? = try {
if Defaults.isEnableGPGIDOn {
2025-01-25 15:40:12 -08:00
let keyID = keyID ?? findGPGID(from: url)
2021-01-07 21:58:38 -08:00
return try PGPAgent.shared.decrypt(encryptedData: encryptedData, keyID: keyID, requestPGPKeyPassphrase: requestPGPKeyPassphrase)
}
2021-01-31 13:34:37 +01:00
return try PGPAgent.shared.decrypt(encryptedData: encryptedData, requestPGPKeyPassphrase: requestPGPKeyPassphrase)
2021-01-07 21:58:38 -08:00
}()
guard let decryptedData = data else {
throw AppError.decryption
}
let plainText = String(data: decryptedData, encoding: .utf8) ?? ""
2025-01-25 15:40:12 -08:00
return Password(name: passwordEntity.name, path: passwordEntity.path, plainText: plainText)
}
public func decrypt(path: String, keyID: String? = nil, requestPGPKeyPassphrase: @escaping (String) -> String) throws -> Password {
guard let passwordEntity = fetchPasswordEntity(with: path) else {
throw AppError.decryption
}
if Defaults.isEnableGPGIDOn {
2021-01-07 21:58:38 -08:00
return try decrypt(passwordEntity: passwordEntity, keyID: keyID, requestPGPKeyPassphrase: requestPGPKeyPassphrase)
}
2021-01-31 13:34:37 +01:00
return try decrypt(passwordEntity: passwordEntity, requestPGPKeyPassphrase: requestPGPKeyPassphrase)
}
public func encrypt(password: Password, keyID: String? = nil) throws -> Data {
2025-01-25 15:40:12 -08:00
let encryptedDataPath = storeURL.appendingPathComponent(password.path)
let keyID = keyID ?? findGPGID(from: encryptedDataPath)
if Defaults.isEnableGPGIDOn {
2021-01-10 15:27:50 -08:00
return try PGPAgent.shared.encrypt(plainData: password.plainData, keyID: keyID)
}
2021-01-31 13:34:37 +01:00
return try PGPAgent.shared.encrypt(plainData: password.plainData)
2017-06-03 18:12:33 -07:00
}
public func removeGitSSHKeys() {
try? fileManager.removeItem(atPath: Globals.gitSSHPrivateKeyPath)
Defaults.remove(\.gitSSHKeySource)
Defaults.remove(\.gitSSHPrivateKeyArmor)
Defaults.remove(\.gitSSHPrivateKeyURL)
AppKeychain.shared.removeContent(for: SSHKey.PRIVATE.getKeychainKey())
gitSSHPrivateKeyPassphrase = nil
}
2017-01-19 21:15:47 +08:00
}
2021-12-31 07:35:17 +01:00
func findGPGID(from url: URL) -> String {
var path = url
while !FileManager.default.fileExists(atPath: path.appendingPathComponent(".gpg-id").path),
2020-11-07 12:06:28 +01:00
path.path != "file:///" {
path = path.deletingLastPathComponent()
}
path = path.appendingPathComponent(".gpg-id")
return (try? String(contentsOf: path))?.trimmed ?? ""
}