mirror of
https://github.com/maxgoedjen/secretive.git
synced 2025-04-06 07:37:07 +00:00
move certificate check into its own function and neaten up a few things
This commit is contained in:
parent
2cce241afe
commit
4adeabc294
@ -7,6 +7,7 @@ import AppKit
|
|||||||
enum OpenSSHCertificateError: Error {
|
enum OpenSSHCertificateError: Error {
|
||||||
case unsupportedType
|
case unsupportedType
|
||||||
case parsingFailed
|
case parsingFailed
|
||||||
|
case doesNotExist
|
||||||
}
|
}
|
||||||
|
|
||||||
extension OpenSSHCertificateError: CustomStringConvertible {
|
extension OpenSSHCertificateError: CustomStringConvertible {
|
||||||
@ -16,6 +17,8 @@ extension OpenSSHCertificateError: CustomStringConvertible {
|
|||||||
return "The key type was unsupported"
|
return "The key type was unsupported"
|
||||||
case .parsingFailed:
|
case .parsingFailed:
|
||||||
return "Failed to properly parse the SSH certificate"
|
return "Failed to properly parse the SSH certificate"
|
||||||
|
case .doesNotExist:
|
||||||
|
return "Certificate does not exist"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -102,35 +105,12 @@ extension Agent {
|
|||||||
var keyData = Data()
|
var keyData = Data()
|
||||||
|
|
||||||
for secret in secrets {
|
for secret in secrets {
|
||||||
let minimalHex = writer.openSSHMD5Fingerprint(secret: secret).replacingOccurrences(of: ":", with: "")
|
|
||||||
let certificatePath = certsPath.appending("/").appending("\(minimalHex)-cert.pub")
|
|
||||||
var keyBlob = writer.data(secret: secret)
|
var keyBlob = writer.data(secret: secret)
|
||||||
var curveData = writer.curveType(for: secret.algorithm, length: secret.keySize).data(using: .utf8)!
|
var curveData = writer.curveType(for: secret.algorithm, length: secret.keySize).data(using: .utf8)!
|
||||||
|
|
||||||
if FileManager.default.fileExists(atPath: certificatePath) {
|
if let (certBlob, certName) = try? checkForCert(secret: secret) {
|
||||||
Logger().debug("Found certificate for \(secret.name)")
|
keyBlob = certBlob
|
||||||
do {
|
curveData = certName
|
||||||
let certContent = try String(contentsOfFile:certificatePath, encoding: .utf8)
|
|
||||||
let certElements = certContent.trimmingCharacters(in: .whitespacesAndNewlines).components(separatedBy: " ")
|
|
||||||
|
|
||||||
if certElements.count >= 2 {
|
|
||||||
if let certDecoded = Data(base64Encoded: certElements[1] as String) {
|
|
||||||
keyBlob = certDecoded
|
|
||||||
|
|
||||||
if certElements.count >= 3 {
|
|
||||||
if let certName = certElements[2].data(using: .utf8) {
|
|
||||||
curveData = certName
|
|
||||||
} else {
|
|
||||||
Logger().info("Certificate for \(secret.name) does not have a name tag, using curveData instead")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Logger().warning("Certificate found for \(secret.name) but failed to decode base64 key, using public key instead")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch {
|
|
||||||
Logger().warning("Certificate found for \(secret.name) but failed to load, using public key instead")
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
keyData.append(writer.lengthAndData(of: keyBlob))
|
keyData.append(writer.lengthAndData(of: keyBlob))
|
||||||
keyData.append(writer.lengthAndData(of: curveData))
|
keyData.append(writer.lengthAndData(of: curveData))
|
||||||
@ -214,7 +194,9 @@ extension Agent {
|
|||||||
return signedData
|
return signedData
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Reconstructs a public key from a ``Data`` object that contains an OpenSSH certificate. Currently only ecdsa certificates are supported
|
||||||
|
/// - Parameter certBlock: The openssh certificate to extract the public key from
|
||||||
|
/// - Returns: A ``Data`` object containing the public key in OpenSSH wire format
|
||||||
func getPublicKeyFromCert(certBlob: Data) throws -> Data {
|
func getPublicKeyFromCert(certBlob: Data) throws -> Data {
|
||||||
let reader = OpenSSHReader(data: certBlob)
|
let reader = OpenSSHReader(data: certBlob)
|
||||||
let certType = String(decoding: reader.readNextChunk(), as: UTF8.self)
|
let certType = String(decoding: reader.readNextChunk(), as: UTF8.self)
|
||||||
@ -239,6 +221,46 @@ extension Agent {
|
|||||||
throw OpenSSHCertificateError.unsupportedType
|
throw OpenSSHCertificateError.unsupportedType
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// Attempts to find an OpenSSH Certificate that corresponds to a ``Secret``
|
||||||
|
/// - Parameter secret: The secret to search for a certificate with
|
||||||
|
/// - Returns: Two ``Data`` objects containing the certificate and certificate name respectively
|
||||||
|
func checkForCert(secret: AnySecret) throws -> (Data, Data) {
|
||||||
|
let minimalHex = writer.openSSHMD5Fingerprint(secret: secret).replacingOccurrences(of: ":", with: "")
|
||||||
|
let certificatePath = certsPath.appending("/").appending("\(minimalHex)-cert.pub")
|
||||||
|
|
||||||
|
if FileManager.default.fileExists(atPath: certificatePath) {
|
||||||
|
Logger().debug("Found certificate for \(secret.name)")
|
||||||
|
do {
|
||||||
|
let certContent = try String(contentsOfFile:certificatePath, encoding: .utf8)
|
||||||
|
let certElements = certContent.trimmingCharacters(in: .whitespacesAndNewlines).components(separatedBy: " ")
|
||||||
|
|
||||||
|
if certElements.count >= 2 {
|
||||||
|
if let certDecoded = Data(base64Encoded: certElements[1] as String) {
|
||||||
|
if certElements.count >= 3 {
|
||||||
|
if let certName = certElements[2].data(using: .utf8) {
|
||||||
|
return (certDecoded, certName)
|
||||||
|
} else if let certName = secret.name.data(using: .utf8) {
|
||||||
|
Logger().info("Certificate for \(secret.name) does not have a name tag, using secret name instead")
|
||||||
|
return (certDecoded, certName)
|
||||||
|
} else {
|
||||||
|
throw OpenSSHCertificateError.parsingFailed
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Logger().warning("Certificate found for \(secret.name) but failed to decode base64 key")
|
||||||
|
throw OpenSSHCertificateError.parsingFailed
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
Logger().warning("Certificate found for \(secret.name) but failed to load")
|
||||||
|
throw OpenSSHCertificateError.parsingFailed
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw OpenSSHCertificateError.doesNotExist
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user