Compare commits

..

3 Commits

Author SHA1 Message Date
f72c509854 Remove CF bridges 2024-02-13 21:15:20 -08:00
6f4226f97a Standardize newline handling (#522)
* Standardize newline handling

* Fix some unterminated bolds in other languages

* Set language back
2024-01-25 02:14:34 +00:00
3315a4bfbc Add Finnish localization (#521) 2024-01-23 00:58:36 +00:00
7 changed files with 286 additions and 41 deletions

View File

@ -44,7 +44,7 @@ let package = Package(
.target(
name: "SecureEnclaveSecretKit",
dependencies: ["SecretKit"],
swiftSettings: [.unsafeFlags(["-warnings-as-errors"])]
swiftSettings: [.enableExperimentalFeature("StrictConcurrency"), .unsafeFlags(["-warnings-as-errors"])]
),
.target(
name: "SmartCardSecretKit",

View File

@ -17,7 +17,7 @@ public protocol Secret: Identifiable, Hashable {
}
/// The type of algorithm the Secret uses. Currently, only elliptic curve algorithms are supported.
public enum Algorithm: Hashable, Sendable {
public enum Algorithm: Hashable {
case ellipticCurve
case rsa

View File

@ -5,7 +5,7 @@ import SecretKit
extension SecureEnclave {
/// An implementation of Secret backed by the Secure Enclave.
public struct Secret: SecretKit.Secret, Sendable {
public struct Secret: SecretKit.Secret {
public let id: Data
public let name: String

View File

@ -75,7 +75,7 @@ extension SecureEnclave {
public func delete(secret: Secret) throws {
let deleteAttributes = KeychainDictionary([
kSecClass: kSecClassKey,
kSecAttrApplicationLabel: secret.id as CFData
kSecAttrApplicationLabel: secret.id
])
let status = SecItemDelete(deleteAttributes)
if status != errSecSuccess {
@ -87,7 +87,7 @@ extension SecureEnclave {
public func update(secret: Secret, name: String) throws {
let updateQuery = KeychainDictionary([
kSecClass: kSecClassKey,
kSecAttrApplicationLabel: secret.id as CFData
kSecAttrApplicationLabel: secret.id
])
let updatedAttributes = KeychainDictionary([
@ -114,7 +114,7 @@ extension SecureEnclave {
let attributes = KeychainDictionary([
kSecClass: kSecClassKey,
kSecAttrKeyClass: kSecAttrKeyClassPrivate,
kSecAttrApplicationLabel: secret.id as CFData,
kSecAttrApplicationLabel: secret.id,
kSecAttrKeyType: Constants.keyType,
kSecAttrTokenID: kSecAttrTokenIDSecureEnclave,
kSecAttrApplicationTag: Constants.keyTag,
@ -145,7 +145,7 @@ extension SecureEnclave {
let attributes = KeychainDictionary([
kSecClass: kSecClassKey,
kSecAttrKeyClass: kSecAttrKeyClassPrivate,
kSecAttrApplicationLabel: secret.id as CFData,
kSecAttrApplicationLabel: secret.id,
kSecAttrKeyType: Constants.keyType,
kSecAttrTokenID: kSecAttrTokenIDSecureEnclave,
kSecAttrApplicationTag: Constants.keyTag,
@ -180,7 +180,7 @@ extension SecureEnclave {
public func persistAuthentication(secret: Secret, forDuration duration: TimeInterval) throws {
let newContext = LAContext()
newContext.touchIDAuthenticationAllowableReuseDuration = max(duration, LATouchIDAuthenticationMaximumAllowableReuseDuration)
newContext.touchIDAuthenticationAllowableReuseDuration = duration
newContext.localizedCancelTitle = String(localized: "auth_context_request_deny_button")
let formatter = DateComponentsFormatter()
@ -196,23 +196,6 @@ extension SecureEnclave {
guard success else { return }
let context = PersistentAuthenticationContext(secret: secret, context: newContext, duration: duration)
self?.persistedAuthenticationContexts[secret] = context
// Contexts will expire within LATouchIDAuthenticationMaximumAllowableReuseDuration unless we periodically refresh them
if duration > LATouchIDAuthenticationMaximumAllowableReuseDuration {
DispatchQueue.main.async {
Timer.scheduledTimer(withTimeInterval: LATouchIDAuthenticationMaximumAllowableReuseDuration - 10, repeats: true) { [weak self] timer in
print("Refreshing context")
guard let refreshContext = self?.persistedAuthenticationContexts[secret] else { return }
guard refreshContext.valid else {
timer.invalidate()
return
}
refreshContext.context.evaluatePolicy(.deviceOwnerAuthentication, localizedReason: "Refresh") { success, _ in
guard success else { return }
print("Refreshed")
}
}
}
}
}
}
@ -321,8 +304,8 @@ extension SecureEnclave.Store {
extension SecureEnclave {
enum Constants {
static let keyTag = "com.maxgoedjen.secretive.secureenclave.key".data(using: .utf8)! as CFData
static let keyType = kSecAttrKeyTypeECSECPrimeRandom
static let keyTag = Data("com.maxgoedjen.secretive.secureenclave.key".utf8)
static let keyType = kSecAttrKeyTypeECSECPrimeRandom as String
static let unauthenticatedThreshold: TimeInterval = 0.05
}

View File

@ -55,7 +55,7 @@ extension SmartCard {
let attributes = KeychainDictionary([
kSecClass: kSecClassKey,
kSecAttrKeyClass: kSecAttrKeyClassPrivate,
kSecAttrApplicationLabel: secret.id as CFData,
kSecAttrApplicationLabel: secret.id,
kSecAttrTokenID: tokenID,
kSecUseAuthenticationContext: context,
kSecReturnRef: true
@ -217,7 +217,7 @@ extension SmartCard.Store {
let attributes = KeychainDictionary([
kSecClass: kSecClassKey,
kSecAttrKeyClass: kSecAttrKeyClassPrivate,
kSecAttrApplicationLabel: secret.id as CFData,
kSecAttrApplicationLabel: secret.id,
kSecAttrTokenID: tokenID,
kSecUseAuthenticationContext: context,
kSecReturnRef: true

View File

@ -433,6 +433,7 @@
fr,
de,
"pt-BR",
fi,
);
mainGroup = 50617D7623FCE48D0099B055;
productRefGroup = 50617D8023FCE48E0099B055 /* Products */;
@ -611,13 +612,14 @@
MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE;
MTL_FAST_MATH = YES;
ONLY_ACTIVE_ARCH = YES;
OTHER_SWIFT_FLAGS = "-Xfrontend -warn-concurrency -Xfrontend -enable-actor-data-race-checks";
OTHER_SWIFT_FLAGS = "";
SDKROOT = macosx;
STRIP_INSTALLED_PRODUCT = NO;
STRIP_SWIFT_SYMBOLS = NO;
SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG;
SWIFT_EMIT_LOC_STRINGS = YES;
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
SWIFT_STRICT_CONCURRENCY = complete;
};
name = Debug;
};
@ -670,13 +672,14 @@
MACOSX_DEPLOYMENT_TARGET = 11.0;
MTL_ENABLE_DEBUG_INFO = NO;
MTL_FAST_MATH = YES;
OTHER_SWIFT_FLAGS = "-Xfrontend -warn-concurrency -Xfrontend -enable-actor-data-race-checks";
OTHER_SWIFT_FLAGS = "";
SDKROOT = macosx;
STRIP_INSTALLED_PRODUCT = NO;
STRIP_SWIFT_SYMBOLS = NO;
SWIFT_COMPILATION_MODE = wholemodule;
SWIFT_EMIT_LOC_STRINGS = YES;
SWIFT_OPTIMIZATION_LEVEL = "-O";
SWIFT_STRICT_CONCURRENCY = complete;
};
name = Release;
};
@ -704,7 +707,6 @@
PRODUCT_BUNDLE_IDENTIFIER = com.maxgoedjen.Secretive.Host;
PRODUCT_NAME = "$(TARGET_NAME)";
PROVISIONING_PROFILE_SPECIFIER = "";
SWIFT_STRICT_CONCURRENCY = complete;
SWIFT_VERSION = 5.0;
};
name = Debug;
@ -733,7 +735,6 @@
PRODUCT_BUNDLE_IDENTIFIER = com.maxgoedjen.Secretive.Host;
PRODUCT_NAME = "$(TARGET_NAME)";
PROVISIONING_PROFILE_SPECIFIER = "Secretive - Host";
SWIFT_STRICT_CONCURRENCY = complete;
SWIFT_VERSION = 5.0;
};
name = Release;
@ -836,13 +837,14 @@
MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE;
MTL_FAST_MATH = YES;
ONLY_ACTIVE_ARCH = YES;
OTHER_SWIFT_FLAGS = "-Xfrontend -warn-concurrency -Xfrontend -enable-actor-data-race-checks";
OTHER_SWIFT_FLAGS = "";
SDKROOT = macosx;
STRIP_INSTALLED_PRODUCT = NO;
STRIP_SWIFT_SYMBOLS = NO;
SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG;
SWIFT_EMIT_LOC_STRINGS = YES;
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
SWIFT_STRICT_CONCURRENCY = complete;
};
name = Test;
};
@ -866,7 +868,6 @@
MARKETING_VERSION = 1;
PRODUCT_BUNDLE_IDENTIFIER = com.maxgoedjen.Secretive.Host;
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_STRICT_CONCURRENCY = complete;
SWIFT_VERSION = 5.0;
};
name = Test;
@ -911,7 +912,6 @@
MARKETING_VERSION = 1;
PRODUCT_BUNDLE_IDENTIFIER = com.maxgoedjen.Secretive.SecretAgent;
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_STRICT_CONCURRENCY = complete;
SWIFT_VERSION = 5.0;
};
name = Test;
@ -936,7 +936,6 @@
MARKETING_VERSION = 1;
PRODUCT_BUNDLE_IDENTIFIER = com.maxgoedjen.Secretive.SecretAgent;
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_STRICT_CONCURRENCY = complete;
SWIFT_VERSION = 5.0;
};
name = Debug;
@ -963,7 +962,6 @@
PRODUCT_BUNDLE_IDENTIFIER = com.maxgoedjen.Secretive.SecretAgent;
PRODUCT_NAME = "$(TARGET_NAME)";
PROVISIONING_PROFILE_SPECIFIER = "Secretive - Secret Agent";
SWIFT_STRICT_CONCURRENCY = complete;
SWIFT_VERSION = 5.0;
};
name = Release;

View File

@ -15,6 +15,12 @@
"value" : "Agent Is Not Running"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Agentti ei ole käynnissä"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -55,22 +61,28 @@
"value" : "SecretAgent is a process that runs in the background to sign requests, so you don't need to keep Secretive open all the time.\n\n**You can close Secretive, and everything will still keep working.**"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "SecretAgent on taustaprosessi joka allekirjoittaa pyyntöjä, joten Secretiveä ei tarvitse pitää käynnissä koko aikaa.**Voit sulkea Secretiven, ja kaikki toimii yhä.**"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
"value" : "SecretAgent est un processus qui s'exécute en arrière-plan pour signer les demandes, de sorte que vous n'ayez pas besoin de garder Secretive ouvert en permanence.\n\n**Vous pouvez fermer Secretive, et tout continuera à fonctionner."
"value" : "SecretAgent est un processus qui s'exécute en arrière-plan pour signer les demandes, de sorte que vous n'ayez pas besoin de garder Secretive ouvert en permanence.\n\n**Vous pouvez fermer Secretive, et tout continuera à fonctionner.**"
}
},
"it" : {
"stringUnit" : {
"state" : "translated",
"value" : "SecretAgent è un processo che viene eseguito in background per firmare le richieste, così che non devi tenere aperto Secretive tutto il tempo.\\n\\n**Puoi chiudere Secretive, e continuerà a funzionare.**"
"value" : "SecretAgent è un processo che viene eseguito in background per firmare le richieste, così che non devi tenere aperto Secretive tutto il tempo.\n\n**Puoi chiudere Secretive, e continuerà a funzionare.**"
}
},
"pt-BR" : {
"stringUnit" : {
"state" : "translated",
"value" : "SecretAgent é um processo que roda em background para assinar requisições para que você não precise manter o Secretive aberto a todo momento.\n\n**Você pode fechar o Secretive e tudo continuará funcionando."
"value" : "SecretAgent é um processo que roda em background para assinar requisições para que você não precise manter o Secretive aberto a todo momento.\n\n**Você pode fechar o Secretive e tudo continuará funcionando.**"
}
},
"zh-Hans" : {
@ -95,6 +107,12 @@
"value" : "Secret Agent is Running"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "SecretAgent on käynnissä"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -135,6 +153,12 @@
"value" : "Agent is Running"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Agentti on käynnissä"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -175,6 +199,12 @@
"value" : "Setup Secretive"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Asenna Secretive"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -215,6 +245,12 @@
"value" : "Help"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Apua"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -255,6 +291,12 @@
"value" : "New Secret"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Uusi salaisuus"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -295,6 +337,12 @@
"value" : "Setup Secretive"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Asenna Secretive"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -335,6 +383,12 @@
"value" : "Secretive needs to be in your Applications folder to work properly. Please move it and relaunch."
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Secretiven tulee olla Ohjelmat-kansiossa toimiakseen oikein. Ole hyvä ja käynnistä se uudelleen siirrettyäsi sen."
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -375,6 +429,12 @@
"value" : "Secretive Is Not in Applications Folder"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Secretive ei ole Ohjelmat-kansiossa"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -417,6 +477,12 @@
"value" : "unlock secret \"%1$@\" for %2$@"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "avaa salaisuuden \"%1$@\" lukitus ajaksi %2$@"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -459,6 +525,12 @@
"value" : "unlock secret \"%1$@\""
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "avaa salaisuuden \"%1$@\" lukitus"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -501,6 +573,12 @@
"value" : "decrypt data using secret \"%1$@\""
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "pura salaus käyttäen salaisuutta \"%1$@\""
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -543,6 +621,12 @@
"value" : "Deny"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Kiellä"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -585,6 +669,12 @@
"value" : "encrypt data using secret \"%1$@\""
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "salaa käyttäen salaisuutta \"%1$@\""
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -627,6 +717,12 @@
"value" : "sign a request from \"%1$@\" using secret \"%2$@\""
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "allekirjoita pyyntö lähteestä \"%1$@\" käyttäen salaisuutta \"%2$@\""
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -669,6 +765,12 @@
"value" : "verify a signature using secret \"%1$@\""
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "varmista allekirjoitus käyttäen salaisuutta \"%1$@\""
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -709,6 +811,12 @@
"value" : "Click to Copy"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Klikkaa kopioidaksesi"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -749,6 +857,12 @@
"value" : "Copied"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Kopioitu"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -789,6 +903,12 @@
"value" : "Cancel"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Peru"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -829,6 +949,12 @@
"value" : "Create"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Luo"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -869,6 +995,12 @@
"value" : "Name:"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Nimi:"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -909,6 +1041,12 @@
"value" : "Shhhhh"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Hysssss"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -949,6 +1087,12 @@
"value" : "No authentication is required while your Mac is unlocked, but you will be notified when a secret is used."
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Todennusta ei vaadita Macisi ollessa lukitsematta, mutta saat ilmoituksen kun salaisuutta käytetään."
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -989,6 +1133,12 @@
"value" : "Notify"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Ilmoita"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1029,6 +1179,12 @@
"value" : "You will be required to authenticate using Touch ID, Apple Watch, or password before each use."
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Touch ID-, Apple Watch- tai salasanatodennus vaaditaan joka käytöllä."
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1069,6 +1225,12 @@
"value" : "Require Authentication"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Vaadi todennus"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1109,6 +1271,12 @@
"value" : "Create a New Secret"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Luo uusi salaisuus"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1149,6 +1317,12 @@
"value" : "Don't Delete"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Älä poista"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1189,6 +1363,12 @@
"value" : "Confirm Name:"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Vahvista nimi:"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1229,6 +1409,12 @@
"value" : "Delete"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Poista"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1269,6 +1455,12 @@
"value" : "If you delete %1$@, you will not be able to recover it. Type \"%2$@\" to confirm."
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Jos poistat kohteen %1$@, sitä ei pysty palauttamaan. Kirjoita \"%2$@\" vahvistaaksesi poiston."
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1309,6 +1501,12 @@
"value" : "Delete %1$@?"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Poista %1$@?"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1349,6 +1547,12 @@
"value" : "Create a new one by clicking here."
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Luo uusi klikkaamalla tästä."
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1389,6 +1593,12 @@
"value" : "No Secrets"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Ei salaisuuksia"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1429,6 +1639,12 @@
"value" : "No Secrets"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Ei salaisuuksia"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1469,6 +1685,12 @@
"value" : "Use your Smart Card's management tool to create a secret."
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Käytä älykorttisi hallintatyökalua luodaksesi salaisuuden."
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1509,6 +1731,12 @@
"value" : "Secretive supports EC256, EC384, RSA1024, and RSA2048 keys."
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Secretive tukee EC256-, EC384-, RSA1024- ja RSA2048-avaimia."
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1549,6 +1777,12 @@
"value" : "No Secrets"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Ei salaisuuksia"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1589,6 +1823,12 @@
"value" : "Your Mac doesn't have a Secure Enclave, and there's not a compatible Smart Card inserted."
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Macissasi ei ole Secure Enclavea, eikä tuettu älykortti ole syötettynä."
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1629,6 +1869,12 @@
"value" : "No Secure Storage Available"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Turvasäilö ei saatavilla"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1669,6 +1915,12 @@
"value" : "If you're looking to add one to your Mac, the YubiKey 5 Series are great."
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Jos haluat lisätä Maciisi sellaisen, YubiKey 5 -sarja on mainio."
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1711,6 +1963,12 @@
"value" : "Leave Unlocked"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Jätä lukitsematta"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",
@ -1753,6 +2011,12 @@
"value" : "Do Not Unlock"
}
},
"fi" : {
"stringUnit" : {
"state" : "translated",
"value" : "Älä avaa lukitusta"
}
},
"fr" : {
"stringUnit" : {
"state" : "translated",