mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			179 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			179 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Swift
		
	
| // Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | |
| 
 | |
| import Foundation
 | |
| import GRDB
 | |
| import Sodium
 | |
| import Curve25519Kit
 | |
| 
 | |
| public struct Identity: Codable, Identifiable, FetchableRecord, PersistableRecord, TableRecord, ColumnExpressible {
 | |
|     public static var databaseTableName: String { "identity" }
 | |
|     
 | |
|     public typealias Columns = CodingKeys
 | |
|     public enum CodingKeys: String, CodingKey, ColumnExpression {
 | |
|         case variant
 | |
|         case data
 | |
|     }
 | |
|     
 | |
|     public enum Variant: String, Codable, CaseIterable, DatabaseValueConvertible {
 | |
|         case seed
 | |
|         case ed25519SecretKey
 | |
|         case ed25519PublicKey
 | |
|         case x25519PrivateKey
 | |
|         case x25519PublicKey
 | |
|     }
 | |
|     
 | |
|     public var id: Variant { variant }
 | |
|     
 | |
|     let variant: Variant
 | |
|     let data: Data
 | |
|     
 | |
|     // MARK: - Initialization
 | |
|     
 | |
|     public init(
 | |
|         variant: Variant,
 | |
|         data: Data
 | |
|     ) {
 | |
|         self.variant = variant
 | |
|         self.data = data
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: - GRDB Interactions
 | |
| 
 | |
| public extension Identity {
 | |
|     static func generate(from seed: Data) throws -> (ed25519KeyPair: KeyPair, x25519KeyPair: KeyPair) {
 | |
|         guard (seed.count == 16) else { throw GeneralError.invalidSeed }
 | |
| 
 | |
|         let padding = Data(repeating: 0, count: 16)
 | |
|         
 | |
|         guard
 | |
|             let ed25519KeyPair = Sodium().sign.keyPair(seed: (seed + padding).bytes),
 | |
|             let x25519PublicKey = Sodium().sign.toX25519(ed25519PublicKey: ed25519KeyPair.publicKey),
 | |
|             let x25519SecretKey = Sodium().sign.toX25519(ed25519SecretKey: ed25519KeyPair.secretKey)
 | |
|         else {
 | |
|             throw GeneralError.keyGenerationFailed
 | |
|         }
 | |
|         
 | |
|         return (
 | |
|             ed25519KeyPair: KeyPair(
 | |
|                 publicKey: ed25519KeyPair.publicKey,
 | |
|                 secretKey: ed25519KeyPair.secretKey
 | |
|             ),
 | |
|             x25519KeyPair: KeyPair(
 | |
|                 publicKey: x25519PublicKey,
 | |
|                 secretKey: x25519SecretKey
 | |
|             )
 | |
|         )
 | |
|     }
 | |
| 
 | |
|     static func store(_ db: Database, seed: Data, ed25519KeyPair: KeyPair, x25519KeyPair: KeyPair) throws {
 | |
|         try Identity(variant: .seed, data: seed).save(db)
 | |
|         try Identity(variant: .ed25519SecretKey, data: Data(ed25519KeyPair.secretKey)).save(db)
 | |
|         try Identity(variant: .ed25519PublicKey, data: Data(ed25519KeyPair.publicKey)).save(db)
 | |
|         try Identity(variant: .x25519PrivateKey, data: Data(x25519KeyPair.secretKey)).save(db)
 | |
|         try Identity(variant: .x25519PublicKey, data: Data(x25519KeyPair.publicKey)).save(db)
 | |
|     }
 | |
|     
 | |
|     static func userExists(_ db: Database? = nil) -> Bool {
 | |
|         return (fetchUserKeyPair(db) != nil)
 | |
|     }
 | |
|     
 | |
|     static func fetchUserPublicKey(_ db: Database? = nil) -> Data? {
 | |
|         guard let db: Database = db else {
 | |
|             return Storage.shared.read { db in fetchUserPublicKey(db) }
 | |
|         }
 | |
|         
 | |
|         return try? Identity.fetchOne(db, id: .x25519PublicKey)?.data
 | |
|     }
 | |
|     
 | |
|     static func fetchUserPrivateKey(_ db: Database? = nil) -> Data? {
 | |
|         guard let db: Database = db else {
 | |
|             return Storage.shared.read { db in fetchUserPrivateKey(db) }
 | |
|         }
 | |
|         
 | |
|         return try? Identity.fetchOne(db, id: .x25519PrivateKey)?.data
 | |
|     }
 | |
|     
 | |
|     static func fetchUserKeyPair(_ db: Database? = nil) -> KeyPair? {
 | |
|         guard let db: Database = db else {
 | |
|             return Storage.shared.read { db in fetchUserKeyPair(db) }
 | |
|         }
 | |
|         guard
 | |
|             let publicKey: Data = fetchUserPublicKey(db),
 | |
|             let privateKey: Data = fetchUserPrivateKey(db)
 | |
|         else { return nil }
 | |
|         
 | |
|         return KeyPair(
 | |
|             publicKey: publicKey.bytes,
 | |
|             secretKey: privateKey.bytes
 | |
|         )
 | |
|     }
 | |
|     
 | |
|     static func fetchUserEd25519KeyPair(_ db: Database? = nil) -> KeyPair? {
 | |
|         guard let db: Database = db else {
 | |
|             return Storage.shared.read { db in fetchUserEd25519KeyPair(db) }
 | |
|         }
 | |
|         guard
 | |
|             let publicKey: Data = try? Identity.fetchOne(db, id: .ed25519PublicKey)?.data,
 | |
|             let secretKey: Data = try? Identity.fetchOne(db, id: .ed25519SecretKey)?.data
 | |
|         else { return nil }
 | |
|         
 | |
|         return KeyPair(
 | |
|             publicKey: publicKey.bytes,
 | |
|             secretKey: secretKey.bytes
 | |
|         )
 | |
|     }
 | |
|     
 | |
|     static func fetchHexEncodedSeed(_ db: Database? = nil) -> String? {
 | |
|         guard let db: Database = db else {
 | |
|             return Storage.shared.read { db in fetchHexEncodedSeed(db) }
 | |
|         }
 | |
|         
 | |
|         guard let data: Data = try? Identity.fetchOne(db, id: .seed)?.data else {
 | |
|             return nil
 | |
|         }
 | |
|         
 | |
|         return data.toHexString()
 | |
|     }
 | |
|     
 | |
|     static func mnemonic() throws -> String {
 | |
|         let dbIsValid: Bool = Storage.shared.isValid
 | |
|         let dbIsSuspendedUnsafe: Bool = Storage.shared.isSuspendedUnsafe
 | |
|         
 | |
|         if let hexEncodedSeed: String = Identity.fetchHexEncodedSeed() {
 | |
|             return Mnemonic.encode(hexEncodedString: hexEncodedSeed)
 | |
|         }
 | |
|         
 | |
|         guard let legacyPrivateKey: String = Identity.fetchUserPrivateKey()?.toHexString() else {
 | |
|             let hasStoredPublicKey: Bool = (Identity.fetchUserPublicKey() != nil)
 | |
|             let hasStoredEdKeyPair: Bool = (Identity.fetchUserEd25519KeyPair() != nil)
 | |
|             let dbStates: [String] = [
 | |
|                 "dbIsValid: \(dbIsValid)",
 | |
|                 "dbIsSuspendedUnsafe: \(dbIsSuspendedUnsafe)",
 | |
|                 "storedSeed: false",
 | |
|                 "userPublicKey: \(hasStoredPublicKey)",
 | |
|                 "userPrivateKey: false",
 | |
|                 "userEdKeyPair: \(hasStoredEdKeyPair)"
 | |
|             ]
 | |
|             
 | |
|             SNLog("Failed to retrieve keys for mnemonic generation (\(dbStates.joined(separator: ", ")))")
 | |
|             throw StorageError.objectNotFound
 | |
|         }
 | |
|                 
 | |
|         // Legacy account
 | |
|         return Mnemonic.encode(hexEncodedString: legacyPrivateKey)
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: - Convenience
 | |
| 
 | |
| public extension Notification.Name {
 | |
|     static let registrationStateDidChange = Notification.Name("registrationStateDidChange")
 | |
| }
 | |
| 
 | |
| public extension Identity {
 | |
|     static func didRegister() {
 | |
|         NotificationCenter.default.post(name: .registrationStateDidChange, object: nil, userInfo: nil)
 | |
|     }
 | |
| }
 |