Skip to content

Commit 8342de0

Browse files
committed
🎉 Add initial implementation
0 parents  commit 8342de0

14 files changed

+543
-0
lines changed

.gitattributes

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
*.pbxproj -text

.gitignore

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
2+
# OSX
3+
#
4+
.DS_Store
5+
6+
# node.js
7+
#
8+
node_modules/
9+
npm-debug.log
10+
yarn-error.log
11+
12+
13+
# Xcode
14+
#
15+
build/
16+
*.pbxuser
17+
!default.pbxuser
18+
*.mode1v3
19+
!default.mode1v3
20+
*.mode2v3
21+
!default.mode2v3
22+
*.perspectivev3
23+
!default.perspectivev3
24+
xcuserdata
25+
*.xccheckout
26+
*.moved-aside
27+
DerivedData
28+
*.hmap
29+
*.ipa
30+
*.xcuserstate
31+
project.xcworkspace
32+
33+
34+
# Android/IntelliJ
35+
#
36+
build/
37+
.idea
38+
.gradle
39+
local.properties
40+
*.iml
41+
42+
# BUCK
43+
buck-out/
44+
\.buckd/
45+
*.keystore
46+

android/build.gradle

+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
2+
buildscript {
3+
repositories {
4+
jcenter()
5+
}
6+
7+
dependencies {
8+
classpath 'com.android.tools.build:gradle:1.3.1'
9+
}
10+
}
11+
12+
apply plugin: 'com.android.library'
13+
14+
def safeExtGet(prop, fallback) {
15+
rootProject.ext.has(prop) ? rootProject.ext.get(prop) : fallback
16+
}
17+
18+
android {
19+
compileSdkVersion safeExtGet('compileSdkVersion', 26)
20+
buildToolsVersion safeExtGet('buildToolsVersion', '26.0.3')
21+
22+
defaultConfig {
23+
minSdkVersion safeExtGet('minSdkVersion', 16)
24+
targetSdkVersion safeExtGet('targetSdkVersion', 26)
25+
versionCode 1
26+
versionName "1.0"
27+
}
28+
lintOptions {
29+
abortOnError false
30+
}
31+
}
32+
33+
repositories {
34+
mavenCentral()
35+
}
36+
37+
dependencies {
38+
compile 'com.facebook.react:react-native:+'
39+
}

android/src/main/AndroidManifest.xml

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
2+
package="org.linusu">
3+
4+
</manifest>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
package org.linusu;
2+
3+
import java.security.NoSuchAlgorithmException;
4+
import java.security.SecureRandom;
5+
6+
import android.util.Base64;
7+
8+
import com.facebook.react.bridge.ReactApplicationContext;
9+
import com.facebook.react.bridge.ReactContextBaseJavaModule;
10+
import com.facebook.react.bridge.ReactMethod;
11+
import com.facebook.react.bridge.Callback;
12+
13+
public class RNGetRandomValuesModule extends ReactContextBaseJavaModule {
14+
15+
private final ReactApplicationContext reactContext;
16+
17+
public RNGetRandomValuesModule(ReactApplicationContext reactContext) {
18+
super(reactContext);
19+
this.reactContext = reactContext;
20+
}
21+
22+
@Override
23+
public String getName() {
24+
return "RNGetRandomValues";
25+
}
26+
27+
@ReactMethod(isBlockingSynchronousMethod = true)
28+
public String getRandomBase64(int byteLength) throws NoSuchAlgorithmException {
29+
byte[] data = new byte[byteLength];
30+
31+
SecureRandom.getInstanceStrong().nextBytes(data);
32+
33+
return Base64.encodeToString(data, Base64.DEFAULT);
34+
}
35+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
package org.linusu;
2+
3+
import java.util.Arrays;
4+
import java.util.Collections;
5+
import java.util.List;
6+
7+
import com.facebook.react.ReactPackage;
8+
import com.facebook.react.bridge.NativeModule;
9+
import com.facebook.react.bridge.ReactApplicationContext;
10+
import com.facebook.react.uimanager.ViewManager;
11+
import com.facebook.react.bridge.JavaScriptModule;
12+
13+
public class RNGetRandomValuesPackage implements ReactPackage {
14+
@Override
15+
public List<NativeModule> createNativeModules(ReactApplicationContext reactContext) {
16+
return Arrays.<NativeModule>asList(new RNGetRandomValuesModule(reactContext));
17+
}
18+
19+
// Deprecated from RN 0.47
20+
public List<Class<? extends JavaScriptModule>> createJSModules() {
21+
return Collections.emptyList();
22+
}
23+
24+
@Override
25+
public List<ViewManager> createViewManagers(ReactApplicationContext reactContext) {
26+
return Collections.emptyList();
27+
}
28+
}

index.js

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
const RNGetRandomValues = require('react-native').NativeModules.RNGetRandomValues
2+
const base64Decode = require('fast-base64-decode')
3+
4+
class TypeMismatchError extends Error {}
5+
class QuotaExceededError extends Error {}
6+
7+
/**
8+
* @param {Int8Array|Uint8Array|Int16Array|Uint16Array|Int32Array|Uint32Array|Uint8ClampedArray} array
9+
*/
10+
function getRandomValues (array) {
11+
if (!(array instanceof Int8Array || array instanceof Uint8Array || array instanceof Int16Array || array instanceof Uint16Array || array instanceof Int32Array || array instanceof Uint32Array || array instanceof Uint8ClampedArray)) {
12+
throw new TypeMismatchError('Expected an integer array')
13+
}
14+
15+
if (array.byteLength > 65536) {
16+
throw new QuotaExceededError('Can only request a maximum of 65536 bytes')
17+
}
18+
19+
base64Decode(RNGetRandomValues.getRandomBase64(array.byteLength), new Uint8Array(array.buffer, array.byteOffset, array.byteLength))
20+
21+
return array
22+
}
23+
24+
if (typeof global.crypto !== 'object') {
25+
global.crypto = {}
26+
}
27+
28+
if (typeof global.crypto.getRandomValues !== 'function') {
29+
global.crypto.getRandomValues = getRandomValues
30+
}

ios/RNGetRandomValues.h

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
#if __has_include("RCTBridgeModule.h")
2+
#import "RCTBridgeModule.h"
3+
#else
4+
#import <React/RCTBridgeModule.h>
5+
#endif
6+
7+
@interface RNGetRandomValues : NSObject <RCTBridgeModule>
8+
-(NSString*)getRandomBase64:(NSUInteger)byteLength;
9+
@end

ios/RNGetRandomValues.m

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
#import "RNGetRandomValues.h"
2+
3+
@implementation RNGetRandomValues
4+
5+
- (dispatch_queue_t)methodQueue
6+
{
7+
return dispatch_get_main_queue();
8+
}
9+
RCT_EXPORT_MODULE()
10+
11+
RCT_EXPORT_SYNCHRONOUS_TYPED_METHOD(NSString*, getRandomBase64:(NSUInteger)byteLength) {
12+
NSMutableData *data = [NSMutableData dataWithLength:byteLength];
13+
int result = SecRandomCopyBytes(kSecRandomDefault, byteLength, data.mutableBytes);
14+
if (result != errSecSuccess) {
15+
@throw([NSException exceptionWithName:@"NO_RANDOM_BYTES" reason:@"Failed to aquire secure random bytes" userInfo:nil]);
16+
}
17+
return [data base64EncodedStringWithOptions:0];
18+
}
19+
20+
@end

0 commit comments

Comments
 (0)