Skip to content

Commit 5fc9e8b

Browse files
authored
feat: add visionOS support (#622)
* feat: add visionOS support * docs: add visionOS docs
1 parent adf379d commit 5fc9e8b

File tree

4 files changed

+38
-25
lines changed

4 files changed

+38
-25
lines changed

README.md

+23-18
Original file line numberDiff line numberDiff line change
@@ -22,9 +22,9 @@
2222
- [`hasInternetCredentials(server)`](#hasinternetcredentialsserver)
2323
- [`getInternetCredentials(server, [{ authenticationPrompt }])`](#getinternetcredentialsserver--authenticationprompt-)
2424
- [`resetInternetCredentials(server)`](#resetinternetcredentialsserver)
25-
- [`requestSharedWebCredentials()` (iOS only)](#requestsharedwebcredentials-ios-only)
26-
- [`setSharedWebCredentials(server, username, password)` (iOS only)](#setsharedwebcredentialsserver-username-password-ios-only)
27-
- [`canImplyAuthentication([{ authenticationType }])` (iOS only)](#canimplyauthentication-authenticationtype--ios-only)
25+
- [`requestSharedWebCredentials()` (iOS and visionOS only)](#requestsharedwebcredentials-ios-and-visionos-only)
26+
- [`setSharedWebCredentials(server, username, password)` (iOS and visionOS only)](#setsharedwebcredentialsserver-username-password-ios-and-visionos-only)
27+
- [`canImplyAuthentication([{ authenticationType }])` (iOS and visionOS only)](#canimplyauthentication-authenticationtype--ios-and-visionos-only)
2828
- [`getSupportedBiometryType()`](#getsupportedbiometrytype)
2929
- [`getSecurityLevel([{ accessControl }])` (Android only)](#getsecuritylevel-accesscontrol--android-only)
3030
- [Options](#options)
@@ -139,21 +139,21 @@ Will retrieve the server/username/password combination from the secure storage.
139139

140140
Will remove the server/username/password combination from the secure storage.
141141

142-
### `requestSharedWebCredentials()` (iOS only)
142+
### `requestSharedWebCredentials()` (iOS and visionOS only)
143143

144144
Asks the user for a shared web credential. Requires additional setup both in the app and server side, see [Apple documentation](https://developer.apple.com/documentation/security/shared_web_credentials). Resolves to `{ server, username, password }` if approved and `false` if denied and throws an error if not supported on platform or there's no shared credentials.
145145

146-
### `setSharedWebCredentials(server, username, password)` (iOS only)
146+
### `setSharedWebCredentials(server, username, password)` (iOS and visionOS only)
147147

148148
Sets a shared web credential. Resolves to `true` when successful.
149149

150-
### `canImplyAuthentication([{ authenticationType }])` (iOS only)
150+
### `canImplyAuthentication([{ authenticationType }])` (iOS and visionOS only)
151151

152152
Inquire if the type of local authentication policy is supported on this device with the device settings the user chose. Should be used in combination with `accessControl` option in the setter functions. Resolves to `true` if supported.
153153

154154
### `getSupportedBiometryType()`
155155

156-
**On iOS:** Get what type of hardware biometry support the device can use for biometric encryption. Resolves to a `Keychain.BIOMETRY_TYPE` value when supported and enrolled, otherwise `null`.
156+
**On iOS and visionOS:** Get what type of hardware biometry support the device can use for biometric encryption. Resolves to a `Keychain.BIOMETRY_TYPE` value when supported and enrolled, otherwise `null`.
157157

158158
**On Android:** Get what type of Class 3 (strong) biometry support the device has. Resolves to a `Keychain.BIOMETRY_TYPE` value when supported, otherwise `null`. In most devices this will return `FINGERPRINT` (except for Pixel 4 or similar where fingerprint sensor is not present).
159159

@@ -167,16 +167,16 @@ Get security level that is supported on the current device with the current OS.
167167

168168
#### Data Structure Properties/Fields
169169

170-
| Key | Platform | Description | Default |
171-
| -------------------------- | ------------ | ------------------------------------------------------------------------------------------------ | ------------------------------------------------------------------------- |
172-
| **`accessControl`** | All | This dictates how a keychain item may be used, see possible values in `Keychain.ACCESS_CONTROL`. | _None_ |
173-
| **`accessible`** | iOS only | This dictates when a keychain item is accessible, see possible values in `Keychain.ACCESSIBLE`. | _`Keychain.ACCESSIBLE.WHEN_UNLOCKED`_ |
174-
| **`accessGroup`** | iOS only | In which App Group to share the keychain. Requires additional setup with entitlements. | _None_ |
175-
| **`authenticationPrompt`** | All | What to prompt the user when unlocking the keychain with biometry or device password. | See [`authenticationPrompt` Properties](#authenticationprompt-properties) |
176-
| **`authenticationType`** | iOS only | Policies specifying which forms of authentication are acceptable. | `Keychain.AUTHENTICATION_TYPE.DEVICE_PASSCODE_OR_BIOMETRICS` |
177-
| **`service`** | All | Reverse domain name qualifier for the service associated with password. | _App bundle ID_ |
178-
| **`storage`** | Android only | Force specific cipher storage usage during saving the password | Select best available storage |
179-
| **`rules`** | Android only | Force following to a specific security rules | `Keychain.RULES.AUTOMATIC_UPGRADE` |
170+
| Key | Platform | Description | Default |
171+
| -------------------------- |---------------| ------------------------------------------------------------------------------------------------ | ------------------------------------------------------------------------- |
172+
| **`accessControl`** | All | This dictates how a keychain item may be used, see possible values in `Keychain.ACCESS_CONTROL`. | _None_ |
173+
| **`accessible`** | iOS, visionOS | This dictates when a keychain item is accessible, see possible values in `Keychain.ACCESSIBLE`. | _`Keychain.ACCESSIBLE.WHEN_UNLOCKED`_ |
174+
| **`accessGroup`** | iOS, visionOS | In which App Group to share the keychain. Requires additional setup with entitlements. | _None_ |
175+
| **`authenticationPrompt`** | All | What to prompt the user when unlocking the keychain with biometry or device password. | See [`authenticationPrompt` Properties](#authenticationprompt-properties) |
176+
| **`authenticationType`** | iOS, visionOS | Policies specifying which forms of authentication are acceptable. | `Keychain.AUTHENTICATION_TYPE.DEVICE_PASSCODE_OR_BIOMETRICS` |
177+
| **`service`** | All | Reverse domain name qualifier for the service associated with password. | _App bundle ID_ |
178+
| **`storage`** | Android only | Force specific cipher storage usage during saving the password | Select best available storage |
179+
| **`rules`** | Android only | Force following to a specific security rules | `Keychain.RULES.AUTOMATIC_UPGRADE` |
180180

181181
##### `authenticationPrompt` Properties
182182

@@ -236,9 +236,10 @@ Refs:
236236
#### `Keychain.BIOMETRY_TYPE` enum
237237

238238
| Key | Description |
239-
| ----------------- | -------------------------------------------------------------------- |
239+
|-------------------|----------------------------------------------------------------------|
240240
| **`TOUCH_ID`** | Device supports authentication with Touch ID. (iOS only) |
241241
| **`FACE_ID`** | Device supports authentication with Face ID. (iOS only) |
242+
| **`OPTIC_ID`** | Device supports authentication with Optic ID. (visionOS only) |
242243
| **`FINGERPRINT`** | Device supports authentication with Fingerprint. (Android only) |
243244
| **`FACE`** | Device supports authentication with Face Recognition. (Android only) |
244245
| **`IRIS`** | Device supports authentication with Iris Recognition. (Android only) |
@@ -503,6 +504,10 @@ Refs:
503504

504505
This package supports macOS Catalyst.
505506

507+
### visionOS
508+
509+
This package supports visionOS.
510+
506511
### Security
507512

508513
On API levels that do not support Android keystore, Facebook Conceal is used to en/decrypt stored data. The encrypted data is then stored in SharedPreferences. Since Conceal itself stores its encryption key in SharedPreferences, it follows that if the device is rooted (or if an attacker can somehow access the filesystem), the key can be obtained and the stored data can be decrypted. Therefore, on such a device, the conceal encryption is only an obscurity. On API level 23+ the key is stored in the Android Keystore, which makes the key non-exportable and therefore makes the entire process more secure. Follow best practices and do not store user credentials on a device. Instead use tokens or other forms of authentication and re-ask for user credentials before performing sensitive operations.

RNKeychain.podspec

+1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ Pod::Spec.new do |s|
1212
s.ios.deployment_target = '9.0'
1313
s.tvos.deployment_target = '9.0'
1414
s.osx.deployment_target = '10.13'
15+
s.visionos.deployment_target = '1.0'
1516
s.source = { :git => "https://github.com/oblador/react-native-keychain.git", :tag => "v#{s.version}" }
1617
s.source_files = 'RNKeychainManager/**/*.{h,m}'
1718
s.preserve_paths = "**/*.js"

RNKeychainManager/RNKeychainManager.m

+12-6
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@
1212
#import <React/RCTBridge.h>
1313
#import <React/RCTUtils.h>
1414

15-
#if TARGET_OS_IOS
15+
#if TARGET_OS_IOS || TARGET_OS_VISION
1616
#import <LocalAuthentication/LAContext.h>
1717
#endif
1818

@@ -151,8 +151,9 @@ CFStringRef accessibleValue(NSDictionary *options)
151151

152152
#define kBiometryTypeTouchID @"TouchID"
153153
#define kBiometryTypeFaceID @"FaceID"
154+
#define kBiometryTypeOpticID @"OpticID"
154155

155-
#if TARGET_OS_IOS
156+
#if TARGET_OS_IOS || TARGET_OS_VISION
156157
LAPolicy authPolicy(NSDictionary *options)
157158
{
158159
if (options && options[kAuthenticationType]) {
@@ -209,7 +210,7 @@ - (void)insertKeychainEntry:(NSDictionary *)attributes
209210

210211
if (accessControl) {
211212
NSError *aerr = nil;
212-
#if TARGET_OS_IOS
213+
#if TARGET_OS_IOS || TARGET_OS_VISION
213214
BOOL canAuthenticate = [[LAContext new] canEvaluatePolicy:LAPolicyDeviceOwnerAuthentication error:&aerr];
214215
if (aerr || !canAuthenticate) {
215216
return rejectWithError(reject, aerr);
@@ -305,7 +306,7 @@ - (OSStatus)deleteCredentialsForServer:(NSString *)server
305306

306307
#pragma mark - RNKeychain
307308

308-
#if TARGET_OS_IOS
309+
#if TARGET_OS_IOS || TARGET_OS_VISION
309310
RCT_EXPORT_METHOD(canCheckAuthentication:(NSDictionary * __nullable)options
310311
resolver:(RCTPromiseResolveBlock)resolve
311312
rejecter:(RCTPromiseRejectBlock)reject)
@@ -323,7 +324,7 @@ - (OSStatus)deleteCredentialsForServer:(NSString *)server
323324
}
324325
#endif
325326

326-
#if TARGET_OS_IOS
327+
#if TARGET_OS_IOS || TARGET_OS_VISION
327328
RCT_EXPORT_METHOD(getSupportedBiometryType:(RCTPromiseResolveBlock)resolve
328329
rejecter:(RCTPromiseRejectBlock)reject)
329330
{
@@ -332,6 +333,11 @@ - (OSStatus)deleteCredentialsForServer:(NSString *)server
332333
BOOL canBeProtected = [context canEvaluatePolicy:LAPolicyDeviceOwnerAuthenticationWithBiometrics error:&aerr];
333334

334335
if (!aerr && canBeProtected) {
336+
if (@available(visionOS 1, *)) {
337+
if (context.biometryType == LABiometryTypeOpticID) {
338+
return resolve(kBiometryTypeOpticID);
339+
}
340+
}
335341
if (@available(iOS 11, *)) {
336342
if (context.biometryType == LABiometryTypeFaceID) {
337343
return resolve(kBiometryTypeFaceID);
@@ -538,7 +544,7 @@ - (OSStatus)deleteCredentialsForServer:(NSString *)server
538544
return resolve(@(YES));
539545
}
540546

541-
#if TARGET_OS_IOS && !TARGET_OS_UIKITFORMAC
547+
#if (TARGET_OS_IOS || TARGET_OS_VISION) && !TARGET_OS_UIKITFORMAC
542548
RCT_EXPORT_METHOD(requestSharedWebCredentials:(RCTPromiseResolveBlock)resolve rejecter:(RCTPromiseRejectBlock)reject)
543549
{
544550
SecRequestSharedWebCredential(NULL, NULL, ^(CFArrayRef credentials, CFErrorRef error) {

typings/react-native-keychain.d.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@ declare module 'react-native-keychain' {
4646
export enum BIOMETRY_TYPE {
4747
TOUCH_ID = 'TouchID',
4848
FACE_ID = 'FaceID',
49+
OPTIC_ID = 'OpticID',
4950
FINGERPRINT = 'Fingerprint',
5051
FACE = 'Face',
5152
IRIS = 'Iris',
@@ -119,7 +120,7 @@ declare module 'react-native-keychain' {
119120
options?: Options
120121
): Promise<null | BIOMETRY_TYPE>;
121122

122-
/** IOS ONLY */
123+
/** IOS AND VISIONOS ONLY */
123124

124125
function requestSharedWebCredentials(): Promise<false | SharedWebCredentials>;
125126

0 commit comments

Comments
 (0)