@l.x/hashcash-native 1.0.2 → 1.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +122 -0
- package/NitroHashcashNative.podspec +56 -0
- package/README.md +24 -0
- package/android/CMakeLists.txt +29 -0
- package/android/build.gradle +141 -0
- package/android/fix-prefab.gradle +51 -0
- package/android/gradle.properties +5 -0
- package/android/src/main/AndroidManifest.xml +2 -0
- package/android/src/main/cpp/cpp-adapter.cpp +6 -0
- package/android/src/main/java/com/margelo/nitro/hashcashnative/HybridHashcash.kt +113 -0
- package/android/src/main/java/com/margelo/nitro/hashcashnative/NitroHashcashNativePackage.kt +18 -0
- package/ios/Bridge.h +8 -0
- package/ios/HybridHashcash.swift +133 -0
- package/lib/benchmark.d.ts +74 -0
- package/lib/benchmark.d.ts.map +1 -0
- package/lib/benchmark.js +92 -0
- package/lib/benchmark.js.map +1 -0
- package/lib/index.d.ts +11 -0
- package/lib/index.d.ts.map +1 -0
- package/lib/index.js +10 -0
- package/lib/index.js.map +1 -0
- package/lib/specs/Hashcash.nitro.d.ts +59 -0
- package/lib/specs/Hashcash.nitro.d.ts.map +1 -0
- package/lib/specs/Hashcash.nitro.js +2 -0
- package/lib/specs/Hashcash.nitro.js.map +1 -0
- package/nitro.json +18 -0
- package/nitrogen/generated/.gitattributes +1 -0
- package/nitrogen/generated/android/NitroHashcashNative+autolinking.cmake +81 -0
- package/nitrogen/generated/android/NitroHashcashNative+autolinking.gradle +27 -0
- package/nitrogen/generated/android/NitroHashcashNativeOnLoad.cpp +44 -0
- package/nitrogen/generated/android/NitroHashcashNativeOnLoad.hpp +25 -0
- package/nitrogen/generated/android/c++/JFindProofParams.hpp +68 -0
- package/nitrogen/generated/android/c++/JHashcashChallenge.hpp +69 -0
- package/nitrogen/generated/android/c++/JHashcashProofResult.hpp +69 -0
- package/nitrogen/generated/android/c++/JHybridHashcashSpec.cpp +81 -0
- package/nitrogen/generated/android/c++/JHybridHashcashSpec.hpp +66 -0
- package/nitrogen/generated/android/kotlin/com/margelo/nitro/hashcashnative/FindProofParams.kt +44 -0
- package/nitrogen/generated/android/kotlin/com/margelo/nitro/hashcashnative/HashcashChallenge.kt +47 -0
- package/nitrogen/generated/android/kotlin/com/margelo/nitro/hashcashnative/HashcashProofResult.kt +47 -0
- package/nitrogen/generated/android/kotlin/com/margelo/nitro/hashcashnative/HybridHashcashSpec.kt +62 -0
- package/nitrogen/generated/android/kotlin/com/margelo/nitro/hashcashnative/NitroHashcashNativeOnLoad.kt +35 -0
- package/nitrogen/generated/ios/NitroHashcashNative+autolinking.rb +60 -0
- package/nitrogen/generated/ios/NitroHashcashNative-Swift-Cxx-Bridge.cpp +49 -0
- package/nitrogen/generated/ios/NitroHashcashNative-Swift-Cxx-Bridge.hpp +154 -0
- package/nitrogen/generated/ios/NitroHashcashNative-Swift-Cxx-Umbrella.hpp +55 -0
- package/nitrogen/generated/ios/NitroHashcashNativeAutolinking.mm +33 -0
- package/nitrogen/generated/ios/NitroHashcashNativeAutolinking.swift +25 -0
- package/nitrogen/generated/ios/c++/HybridHashcashSpecSwift.cpp +11 -0
- package/nitrogen/generated/ios/c++/HybridHashcashSpecSwift.hpp +92 -0
- package/nitrogen/generated/ios/swift/FindProofParams.swift +82 -0
- package/nitrogen/generated/ios/swift/Func_void_std__exception_ptr.swift +47 -0
- package/nitrogen/generated/ios/swift/Func_void_std__optional_HashcashProofResult_.swift +47 -0
- package/nitrogen/generated/ios/swift/HashcashChallenge.swift +69 -0
- package/nitrogen/generated/ios/swift/HashcashProofResult.swift +69 -0
- package/nitrogen/generated/ios/swift/HybridHashcashSpec.swift +57 -0
- package/nitrogen/generated/ios/swift/HybridHashcashSpec_cxx.swift +155 -0
- package/nitrogen/generated/shared/c++/FindProofParams.hpp +85 -0
- package/nitrogen/generated/shared/c++/HashcashChallenge.hpp +87 -0
- package/nitrogen/generated/shared/c++/HashcashProofResult.hpp +87 -0
- package/nitrogen/generated/shared/c++/HybridHashcashSpec.cpp +22 -0
- package/nitrogen/generated/shared/c++/HybridHashcashSpec.hpp +69 -0
- package/package.json +73 -1
- package/react-native.config.js +16 -0
- package/src/benchmark.ts +152 -0
- package/src/index.ts +23 -0
- package/src/specs/Hashcash.nitro.ts +64 -0
- package/index.d.ts +0 -1
- package/index.js +0 -1
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
import Foundation
|
|
2
|
+
import CommonCrypto
|
|
3
|
+
import NitroModules
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Native hashcash proof-of-work solver for iOS.
|
|
7
|
+
*
|
|
8
|
+
* Uses CommonCrypto for SHA256 hashing. The computation runs on a
|
|
9
|
+
* background thread via Nitro's Promise.async to avoid blocking
|
|
10
|
+
* the main/JS thread.
|
|
11
|
+
*
|
|
12
|
+
* Performance notes:
|
|
13
|
+
* - Cancellation is checked every 1000 iterations to balance responsiveness
|
|
14
|
+
* with cancellation check overhead
|
|
15
|
+
* - NSLock provides thread-safe access to the cancellation flag
|
|
16
|
+
* (Swift's standard pattern for thread-safe booleans, equivalent to
|
|
17
|
+
* Java's AtomicBoolean pattern)
|
|
18
|
+
* - Native CommonCrypto is significantly faster than JS crypto
|
|
19
|
+
*/
|
|
20
|
+
class HybridHashcash: HybridHashcashSpec {
|
|
21
|
+
/// Thread-safe cancellation flag. Checked every 1000 iterations
|
|
22
|
+
/// to allow responsive cancellation without excessive overhead.
|
|
23
|
+
private var cancelled = false
|
|
24
|
+
|
|
25
|
+
/// NSLock for thread-safe cancellation flag access.
|
|
26
|
+
/// Swift's standard pattern for synchronizing access to shared state.
|
|
27
|
+
private let lock = NSLock()
|
|
28
|
+
|
|
29
|
+
private var isCancelled: Bool {
|
|
30
|
+
lock.lock()
|
|
31
|
+
defer { lock.unlock() }
|
|
32
|
+
return cancelled
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
private func setCancelled(_ value: Bool) {
|
|
36
|
+
lock.lock()
|
|
37
|
+
defer { lock.unlock() }
|
|
38
|
+
cancelled = value
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Find a proof-of-work solution for the given challenge.
|
|
43
|
+
*
|
|
44
|
+
* Iterates through counter values computing SHA256 hashes until
|
|
45
|
+
* one is found with the required number of leading zero bytes.
|
|
46
|
+
*/
|
|
47
|
+
func findProof(params: FindProofParams) throws -> Promise<HashcashProofResult?> {
|
|
48
|
+
return Promise.async { [weak self] in
|
|
49
|
+
guard let self = self else { return nil }
|
|
50
|
+
|
|
51
|
+
self.setCancelled(false)
|
|
52
|
+
|
|
53
|
+
let challenge = params.challenge
|
|
54
|
+
let rangeStart = Int(params.rangeStart ?? 0)
|
|
55
|
+
let rangeSize = Int(params.rangeSize ?? challenge.maxProofLength)
|
|
56
|
+
let rangeEnd = rangeStart + rangeSize
|
|
57
|
+
|
|
58
|
+
let startTime = CFAbsoluteTimeGetCurrent()
|
|
59
|
+
|
|
60
|
+
for counter in rangeStart..<rangeEnd {
|
|
61
|
+
// Check cancellation every 1000 iterations
|
|
62
|
+
if counter % 1000 == 0 && self.isCancelled {
|
|
63
|
+
return nil
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
// Compute hash for this counter
|
|
67
|
+
let hash = self.computeHash(
|
|
68
|
+
subject: challenge.subject,
|
|
69
|
+
nonce: challenge.nonce,
|
|
70
|
+
counter: counter
|
|
71
|
+
)
|
|
72
|
+
|
|
73
|
+
// Check if hash meets difficulty requirement
|
|
74
|
+
if self.checkDifficulty(hash: hash, difficulty: Int(challenge.difficulty)) {
|
|
75
|
+
let timeMs = (CFAbsoluteTimeGetCurrent() - startTime) * 1000
|
|
76
|
+
|
|
77
|
+
return HashcashProofResult(
|
|
78
|
+
counter: String(counter),
|
|
79
|
+
hashBase64: hash.base64EncodedString(),
|
|
80
|
+
attempts: Double(counter - rangeStart + 1),
|
|
81
|
+
timeMs: timeMs
|
|
82
|
+
)
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
return nil
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
/**
|
|
91
|
+
* Cancel any in-progress proof search.
|
|
92
|
+
*/
|
|
93
|
+
func cancel() throws {
|
|
94
|
+
setCancelled(true)
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
// MARK: - Private Helpers
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* Compute SHA256 hash of "{subject}:{nonce}:{counter}"
|
|
101
|
+
*/
|
|
102
|
+
private func computeHash(subject: String, nonce: String, counter: Int) -> Data {
|
|
103
|
+
let solutionString = "\(subject):\(nonce):\(counter)"
|
|
104
|
+
guard let data = solutionString.data(using: .utf8) else {
|
|
105
|
+
return Data()
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
var hash = [UInt8](repeating: 0, count: Int(CC_SHA256_DIGEST_LENGTH))
|
|
109
|
+
data.withUnsafeBytes { bytes in
|
|
110
|
+
_ = CC_SHA256(bytes.baseAddress, CC_LONG(data.count), &hash)
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
return Data(hash)
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
/**
|
|
117
|
+
* Check if hash has required number of leading zero bytes.
|
|
118
|
+
*
|
|
119
|
+
* difficulty=1 means first byte must be 0x00
|
|
120
|
+
* difficulty=2 means first two bytes must be 0x00, etc.
|
|
121
|
+
*/
|
|
122
|
+
private func checkDifficulty(hash: Data, difficulty: Int) -> Bool {
|
|
123
|
+
guard hash.count >= difficulty else { return false }
|
|
124
|
+
|
|
125
|
+
for i in 0..<difficulty {
|
|
126
|
+
if hash[i] != 0 {
|
|
127
|
+
return false
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
return true
|
|
132
|
+
}
|
|
133
|
+
}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Hashcash Benchmark Utility
|
|
3
|
+
*
|
|
4
|
+
* Compares native vs JS hashcash implementation performance.
|
|
5
|
+
* Run this in your app to measure the speedup from native code.
|
|
6
|
+
*/
|
|
7
|
+
/** biome-ignore-all lint/suspicious/noConsole: dev only file */
|
|
8
|
+
interface BenchmarkResult {
|
|
9
|
+
implementation: 'native' | 'js';
|
|
10
|
+
difficulty: number;
|
|
11
|
+
counter: string | null;
|
|
12
|
+
attempts: number;
|
|
13
|
+
timeMs: number;
|
|
14
|
+
}
|
|
15
|
+
interface BenchmarkSummary {
|
|
16
|
+
native: BenchmarkResult;
|
|
17
|
+
js: BenchmarkResult;
|
|
18
|
+
speedup: number;
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Run a single benchmark with the native implementation
|
|
22
|
+
*/
|
|
23
|
+
export declare function benchmarkNative(difficulty: number): Promise<BenchmarkResult>;
|
|
24
|
+
/**
|
|
25
|
+
* Run a single benchmark with the JS implementation
|
|
26
|
+
* Requires passing in the findProof function from @l.x/sessions
|
|
27
|
+
*/
|
|
28
|
+
export declare function benchmarkJS(difficulty: number, findProof: (params: {
|
|
29
|
+
challenge: {
|
|
30
|
+
difficulty: number;
|
|
31
|
+
subject: string;
|
|
32
|
+
algorithm: 'sha256';
|
|
33
|
+
nonce: string;
|
|
34
|
+
max_proof_length: number;
|
|
35
|
+
};
|
|
36
|
+
}) => {
|
|
37
|
+
counter: string;
|
|
38
|
+
attempts: number;
|
|
39
|
+
timeMs: number;
|
|
40
|
+
} | null): Promise<BenchmarkResult>;
|
|
41
|
+
/**
|
|
42
|
+
* Run a comparison benchmark between native and JS implementations
|
|
43
|
+
*/
|
|
44
|
+
export declare function runBenchmark(difficulty: number, jsFindProof: (params: {
|
|
45
|
+
challenge: {
|
|
46
|
+
difficulty: number;
|
|
47
|
+
subject: string;
|
|
48
|
+
algorithm: 'sha256';
|
|
49
|
+
nonce: string;
|
|
50
|
+
max_proof_length: number;
|
|
51
|
+
};
|
|
52
|
+
}) => {
|
|
53
|
+
counter: string;
|
|
54
|
+
attempts: number;
|
|
55
|
+
timeMs: number;
|
|
56
|
+
} | null): Promise<BenchmarkSummary>;
|
|
57
|
+
/**
|
|
58
|
+
* Run benchmarks at multiple difficulty levels
|
|
59
|
+
*/
|
|
60
|
+
export declare function runFullBenchmark(jsFindProof: (params: {
|
|
61
|
+
challenge: {
|
|
62
|
+
difficulty: number;
|
|
63
|
+
subject: string;
|
|
64
|
+
algorithm: 'sha256';
|
|
65
|
+
nonce: string;
|
|
66
|
+
max_proof_length: number;
|
|
67
|
+
};
|
|
68
|
+
}) => {
|
|
69
|
+
counter: string;
|
|
70
|
+
attempts: number;
|
|
71
|
+
timeMs: number;
|
|
72
|
+
} | null): Promise<void>;
|
|
73
|
+
export {};
|
|
74
|
+
//# sourceMappingURL=benchmark.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"benchmark.d.ts","sourceRoot":"","sources":["../src/benchmark.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AACH,gEAAgE;AAIhE,UAAU,eAAe;IACvB,cAAc,EAAE,QAAQ,GAAG,IAAI,CAAA;IAC/B,UAAU,EAAE,MAAM,CAAA;IAClB,OAAO,EAAE,MAAM,GAAG,IAAI,CAAA;IACtB,QAAQ,EAAE,MAAM,CAAA;IAChB,MAAM,EAAE,MAAM,CAAA;CACf;AAED,UAAU,gBAAgB;IACxB,MAAM,EAAE,eAAe,CAAA;IACvB,EAAE,EAAE,eAAe,CAAA;IACnB,OAAO,EAAE,MAAM,CAAA;CAChB;AAED;;GAEG;AACH,wBAAsB,eAAe,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC,CAmBlF;AAED;;;GAGG;AACH,wBAAsB,WAAW,CAC/B,UAAU,EAAE,MAAM,EAClB,SAAS,EAAE,CAAC,MAAM,EAAE;IAClB,SAAS,EAAE;QACT,UAAU,EAAE,MAAM,CAAA;QAClB,OAAO,EAAE,MAAM,CAAA;QACf,SAAS,EAAE,QAAQ,CAAA;QACnB,KAAK,EAAE,MAAM,CAAA;QACb,gBAAgB,EAAE,MAAM,CAAA;KACzB,CAAA;CACF,KAAK;IAAE,OAAO,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GAAG,IAAI,GACjE,OAAO,CAAC,eAAe,CAAC,CAkB1B;AAED;;GAEG;AACH,wBAAsB,YAAY,CAChC,UAAU,EAAE,MAAM,EAClB,WAAW,EAAE,CAAC,MAAM,EAAE;IACpB,SAAS,EAAE;QACT,UAAU,EAAE,MAAM,CAAA;QAClB,OAAO,EAAE,MAAM,CAAA;QACf,SAAS,EAAE,QAAQ,CAAA;QACnB,KAAK,EAAE,MAAM,CAAA;QACb,gBAAgB,EAAE,MAAM,CAAA;KACzB,CAAA;CACF,KAAK;IAAE,OAAO,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GAAG,IAAI,GACjE,OAAO,CAAC,gBAAgB,CAAC,CAyB3B;AAED;;GAEG;AACH,wBAAsB,gBAAgB,CACpC,WAAW,EAAE,CAAC,MAAM,EAAE;IACpB,SAAS,EAAE;QACT,UAAU,EAAE,MAAM,CAAA;QAClB,OAAO,EAAE,MAAM,CAAA;QACf,SAAS,EAAE,QAAQ,CAAA;QACnB,KAAK,EAAE,MAAM,CAAA;QACb,gBAAgB,EAAE,MAAM,CAAA;KACzB,CAAA;CACF,KAAK;IAAE,OAAO,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GAAG,IAAI,GACjE,OAAO,CAAC,IAAI,CAAC,CAcf"}
|
package/lib/benchmark.js
ADDED
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Hashcash Benchmark Utility
|
|
3
|
+
*
|
|
4
|
+
* Compares native vs JS hashcash implementation performance.
|
|
5
|
+
* Run this in your app to measure the speedup from native code.
|
|
6
|
+
*/
|
|
7
|
+
/** biome-ignore-all lint/suspicious/noConsole: dev only file */
|
|
8
|
+
import { HashcashNative } from './index';
|
|
9
|
+
/**
|
|
10
|
+
* Run a single benchmark with the native implementation
|
|
11
|
+
*/
|
|
12
|
+
export async function benchmarkNative(difficulty) {
|
|
13
|
+
var _a, _b;
|
|
14
|
+
const challenge = {
|
|
15
|
+
difficulty,
|
|
16
|
+
subject: 'Benchmark',
|
|
17
|
+
nonce: 'dGVzdC1ub25jZS1iZW5jaG1hcms=', // "test-nonce-benchmark" in base64
|
|
18
|
+
maxProofLength: 10000000,
|
|
19
|
+
};
|
|
20
|
+
const startTime = performance.now();
|
|
21
|
+
const result = await HashcashNative.findProof({ challenge });
|
|
22
|
+
const endTime = performance.now();
|
|
23
|
+
return {
|
|
24
|
+
implementation: 'native',
|
|
25
|
+
difficulty,
|
|
26
|
+
counter: (_a = result === null || result === void 0 ? void 0 : result.counter) !== null && _a !== void 0 ? _a : null,
|
|
27
|
+
attempts: (_b = result === null || result === void 0 ? void 0 : result.attempts) !== null && _b !== void 0 ? _b : 0,
|
|
28
|
+
timeMs: endTime - startTime,
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Run a single benchmark with the JS implementation
|
|
33
|
+
* Requires passing in the findProof function from @l.x/sessions
|
|
34
|
+
*/
|
|
35
|
+
export async function benchmarkJS(difficulty, findProof) {
|
|
36
|
+
var _a, _b, _c;
|
|
37
|
+
const challenge = {
|
|
38
|
+
difficulty,
|
|
39
|
+
subject: 'Benchmark',
|
|
40
|
+
algorithm: 'sha256',
|
|
41
|
+
nonce: 'dGVzdC1ub25jZS1iZW5jaG1hcms=',
|
|
42
|
+
max_proof_length: 10000000,
|
|
43
|
+
};
|
|
44
|
+
const result = findProof({ challenge });
|
|
45
|
+
return {
|
|
46
|
+
implementation: 'js',
|
|
47
|
+
difficulty,
|
|
48
|
+
counter: (_a = result === null || result === void 0 ? void 0 : result.counter) !== null && _a !== void 0 ? _a : null,
|
|
49
|
+
attempts: (_b = result === null || result === void 0 ? void 0 : result.attempts) !== null && _b !== void 0 ? _b : 0,
|
|
50
|
+
timeMs: (_c = result === null || result === void 0 ? void 0 : result.timeMs) !== null && _c !== void 0 ? _c : 0,
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Run a comparison benchmark between native and JS implementations
|
|
55
|
+
*/
|
|
56
|
+
export async function runBenchmark(difficulty, jsFindProof) {
|
|
57
|
+
console.log(`\n🏁 Starting hashcash benchmark (difficulty=${difficulty})...\n`);
|
|
58
|
+
// Run native first
|
|
59
|
+
console.log('⏱️ Running native implementation...');
|
|
60
|
+
const nativeResult = await benchmarkNative(difficulty);
|
|
61
|
+
console.log(` Native: ${nativeResult.timeMs.toFixed(2)}ms (${nativeResult.attempts} attempts)`);
|
|
62
|
+
// Run JS
|
|
63
|
+
console.log('⏱️ Running JS implementation...');
|
|
64
|
+
const jsResult = await benchmarkJS(difficulty, jsFindProof);
|
|
65
|
+
console.log(` JS: ${jsResult.timeMs.toFixed(2)}ms (${jsResult.attempts} attempts)`);
|
|
66
|
+
const speedup = jsResult.timeMs / nativeResult.timeMs;
|
|
67
|
+
console.log(`\n📊 Results:`);
|
|
68
|
+
console.log(` Native: ${nativeResult.timeMs.toFixed(2)}ms`);
|
|
69
|
+
console.log(` JS: ${jsResult.timeMs.toFixed(2)}ms`);
|
|
70
|
+
console.log(` Speedup: ${speedup.toFixed(2)}x faster with native\n`);
|
|
71
|
+
return {
|
|
72
|
+
native: nativeResult,
|
|
73
|
+
js: jsResult,
|
|
74
|
+
speedup,
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Run benchmarks at multiple difficulty levels
|
|
79
|
+
*/
|
|
80
|
+
export async function runFullBenchmark(jsFindProof) {
|
|
81
|
+
console.log('═══════════════════════════════════════════');
|
|
82
|
+
console.log(' HASHCASH NATIVE vs JS BENCHMARK');
|
|
83
|
+
console.log('═══════════════════════════════════════════');
|
|
84
|
+
const difficulties = [1, 2];
|
|
85
|
+
for (const difficulty of difficulties) {
|
|
86
|
+
await runBenchmark(difficulty, jsFindProof);
|
|
87
|
+
}
|
|
88
|
+
console.log('═══════════════════════════════════════════');
|
|
89
|
+
console.log(' BENCHMARK COMPLETE');
|
|
90
|
+
console.log('═══════════════════════════════════════════\n');
|
|
91
|
+
}
|
|
92
|
+
//# sourceMappingURL=benchmark.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"benchmark.js","sourceRoot":"","sources":["../src/benchmark.ts"],"names":[],"mappings":"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"}
|
package/lib/index.d.ts
ADDED
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import type { Hashcash } from './specs/Hashcash.nitro';
|
|
2
|
+
/**
|
|
3
|
+
* Native hashcash proof-of-work solver.
|
|
4
|
+
*
|
|
5
|
+
* Uses platform-native SHA256 implementation for better performance
|
|
6
|
+
* than JavaScript-based hashing.
|
|
7
|
+
*/
|
|
8
|
+
export declare const HashcashNative: Hashcash;
|
|
9
|
+
export { benchmarkJS, benchmarkNative, runBenchmark, runFullBenchmark, } from './benchmark';
|
|
10
|
+
export type { FindProofParams, Hashcash, HashcashChallenge, HashcashProofResult, } from './specs/Hashcash.nitro';
|
|
11
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,wBAAwB,CAAA;AAEtD;;;;;GAKG;AACH,eAAO,MAAM,cAAc,UAAwD,CAAA;AAEnF,OAAO,EACL,WAAW,EACX,eAAe,EACf,YAAY,EACZ,gBAAgB,GACjB,MAAM,aAAa,CAAA;AACpB,YAAY,EACV,eAAe,EACf,QAAQ,EACR,iBAAiB,EACjB,mBAAmB,GACpB,MAAM,wBAAwB,CAAA"}
|
package/lib/index.js
ADDED
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { NitroModules } from 'react-native-nitro-modules';
|
|
2
|
+
/**
|
|
3
|
+
* Native hashcash proof-of-work solver.
|
|
4
|
+
*
|
|
5
|
+
* Uses platform-native SHA256 implementation for better performance
|
|
6
|
+
* than JavaScript-based hashing.
|
|
7
|
+
*/
|
|
8
|
+
export const HashcashNative = NitroModules.createHybridObject('Hashcash');
|
|
9
|
+
export { benchmarkJS, benchmarkNative, runBenchmark, runFullBenchmark, } from './benchmark';
|
|
10
|
+
//# sourceMappingURL=index.js.map
|
package/lib/index.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,4BAA4B,CAAA;AAGzD;;;;;GAKG;AACH,MAAM,CAAC,MAAM,cAAc,GAAG,YAAY,CAAC,kBAAkB,CAAW,UAAU,CAAC,CAAA;AAEnF,OAAO,EACL,WAAW,EACX,eAAe,EACf,YAAY,EACZ,gBAAgB,GACjB,MAAM,aAAa,CAAA"}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import type { HybridObject } from 'react-native-nitro-modules';
|
|
2
|
+
/**
|
|
3
|
+
* Challenge parameters for hashcash proof-of-work
|
|
4
|
+
*/
|
|
5
|
+
export interface HashcashChallenge {
|
|
6
|
+
/** Number of zero bytes required at start of hash */
|
|
7
|
+
difficulty: number;
|
|
8
|
+
/** Subject string (e.g., "Lx") */
|
|
9
|
+
subject: string;
|
|
10
|
+
/** Base64-encoded nonce from backend */
|
|
11
|
+
nonce: string;
|
|
12
|
+
/** Maximum counter value to try */
|
|
13
|
+
maxProofLength: number;
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Result of a successful proof-of-work computation
|
|
17
|
+
*/
|
|
18
|
+
export interface HashcashProofResult {
|
|
19
|
+
/** The counter value that produced a valid hash */
|
|
20
|
+
counter: string;
|
|
21
|
+
/** Base64-encoded SHA256 hash */
|
|
22
|
+
hashBase64: string;
|
|
23
|
+
/** Number of hash attempts made */
|
|
24
|
+
attempts: number;
|
|
25
|
+
/** Time taken in milliseconds */
|
|
26
|
+
timeMs: number;
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Parameters for findProof
|
|
30
|
+
*/
|
|
31
|
+
export interface FindProofParams {
|
|
32
|
+
challenge: HashcashChallenge;
|
|
33
|
+
rangeStart?: number;
|
|
34
|
+
rangeSize?: number;
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Native hashcash proof-of-work solver.
|
|
38
|
+
*
|
|
39
|
+
* Computes SHA256 hashes to find a counter value that produces
|
|
40
|
+
* a hash with the required number of leading zero bytes.
|
|
41
|
+
*/
|
|
42
|
+
export interface Hashcash extends HybridObject<{
|
|
43
|
+
ios: 'swift';
|
|
44
|
+
android: 'kotlin';
|
|
45
|
+
}> {
|
|
46
|
+
/**
|
|
47
|
+
* Find a proof-of-work solution for the given challenge.
|
|
48
|
+
*
|
|
49
|
+
* @param params - Challenge parameters and optional range limits
|
|
50
|
+
* @returns ProofResult if found, undefined if cancelled or no solution in range
|
|
51
|
+
*/
|
|
52
|
+
findProof(params: FindProofParams): Promise<HashcashProofResult | undefined>;
|
|
53
|
+
/**
|
|
54
|
+
* Cancel any in-progress proof search.
|
|
55
|
+
* The current findProof call will return undefined.
|
|
56
|
+
*/
|
|
57
|
+
cancel(): void;
|
|
58
|
+
}
|
|
59
|
+
//# sourceMappingURL=Hashcash.nitro.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Hashcash.nitro.d.ts","sourceRoot":"","sources":["../../src/specs/Hashcash.nitro.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,4BAA4B,CAAA;AAE9D;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC,qDAAqD;IACrD,UAAU,EAAE,MAAM,CAAA;IAClB,kCAAkC;IAClC,OAAO,EAAE,MAAM,CAAA;IACf,wCAAwC;IACxC,KAAK,EAAE,MAAM,CAAA;IACb,mCAAmC;IACnC,cAAc,EAAE,MAAM,CAAA;CACvB;AAED;;GAEG;AACH,MAAM,WAAW,mBAAmB;IAClC,mDAAmD;IACnD,OAAO,EAAE,MAAM,CAAA;IACf,iCAAiC;IACjC,UAAU,EAAE,MAAM,CAAA;IAClB,mCAAmC;IACnC,QAAQ,EAAE,MAAM,CAAA;IAChB,iCAAiC;IACjC,MAAM,EAAE,MAAM,CAAA;CACf;AAED;;GAEG;AACH,MAAM,WAAW,eAAe;IAC9B,SAAS,EAAE,iBAAiB,CAAA;IAC5B,UAAU,CAAC,EAAE,MAAM,CAAA;IACnB,SAAS,CAAC,EAAE,MAAM,CAAA;CACnB;AAED;;;;;GAKG;AACH,MAAM,WAAW,QACf,SAAQ,YAAY,CAAC;IACnB,GAAG,EAAE,OAAO,CAAA;IACZ,OAAO,EAAE,QAAQ,CAAA;CAClB,CAAC;IACF;;;;;OAKG;IACH,SAAS,CAAC,MAAM,EAAE,eAAe,GAAG,OAAO,CAAC,mBAAmB,GAAG,SAAS,CAAC,CAAA;IAE5E;;;OAGG;IACH,MAAM,IAAI,IAAI,CAAA;CACf"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Hashcash.nitro.js","sourceRoot":"","sources":["../../src/specs/Hashcash.nitro.ts"],"names":[],"mappings":""}
|
package/nitro.json
ADDED
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
{
|
|
2
|
+
"$schema": "https://nitro.margelo.com/nitro.schema.json",
|
|
3
|
+
"cxxNamespace": ["hashcashnative"],
|
|
4
|
+
"ios": {
|
|
5
|
+
"iosModuleName": "NitroHashcashNative"
|
|
6
|
+
},
|
|
7
|
+
"android": {
|
|
8
|
+
"androidNamespace": ["hashcashnative"],
|
|
9
|
+
"androidCxxLibName": "NitroHashcashNative"
|
|
10
|
+
},
|
|
11
|
+
"autolinking": {
|
|
12
|
+
"Hashcash": {
|
|
13
|
+
"swift": "HybridHashcash",
|
|
14
|
+
"kotlin": "HybridHashcash"
|
|
15
|
+
}
|
|
16
|
+
},
|
|
17
|
+
"ignorePaths": ["**/node_modules"]
|
|
18
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
** linguist-generated=true
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
#
|
|
2
|
+
# NitroHashcashNative+autolinking.cmake
|
|
3
|
+
# This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
# https://github.com/mrousavy/nitro
|
|
5
|
+
# Copyright © 2025 Marc Rousavy @ Margelo
|
|
6
|
+
#
|
|
7
|
+
|
|
8
|
+
# This is a CMake file that adds all files generated by Nitrogen
|
|
9
|
+
# to the current CMake project.
|
|
10
|
+
#
|
|
11
|
+
# To use it, add this to your CMakeLists.txt:
|
|
12
|
+
# ```cmake
|
|
13
|
+
# include(${CMAKE_SOURCE_DIR}/../nitrogen/generated/android/NitroHashcashNative+autolinking.cmake)
|
|
14
|
+
# ```
|
|
15
|
+
|
|
16
|
+
# Define a flag to check if we are building properly
|
|
17
|
+
add_definitions(-DBUILDING_NITROHASHCASHNATIVE_WITH_GENERATED_CMAKE_PROJECT)
|
|
18
|
+
|
|
19
|
+
# Enable Raw Props parsing in react-native (for Nitro Views)
|
|
20
|
+
add_definitions(-DRN_SERIALIZABLE_STATE)
|
|
21
|
+
|
|
22
|
+
# Add all headers that were generated by Nitrogen
|
|
23
|
+
include_directories(
|
|
24
|
+
"../nitrogen/generated/shared/c++"
|
|
25
|
+
"../nitrogen/generated/android/c++"
|
|
26
|
+
"../nitrogen/generated/android/"
|
|
27
|
+
)
|
|
28
|
+
|
|
29
|
+
# Add all .cpp sources that were generated by Nitrogen
|
|
30
|
+
target_sources(
|
|
31
|
+
# CMake project name (Android C++ library name)
|
|
32
|
+
NitroHashcashNative PRIVATE
|
|
33
|
+
# Autolinking Setup
|
|
34
|
+
../nitrogen/generated/android/NitroHashcashNativeOnLoad.cpp
|
|
35
|
+
# Shared Nitrogen C++ sources
|
|
36
|
+
../nitrogen/generated/shared/c++/HybridHashcashSpec.cpp
|
|
37
|
+
# Android-specific Nitrogen C++ sources
|
|
38
|
+
../nitrogen/generated/android/c++/JHybridHashcashSpec.cpp
|
|
39
|
+
)
|
|
40
|
+
|
|
41
|
+
# From node_modules/react-native/ReactAndroid/cmake-utils/folly-flags.cmake
|
|
42
|
+
# Used in node_modules/react-native/ReactAndroid/cmake-utils/ReactNative-application.cmake
|
|
43
|
+
target_compile_definitions(
|
|
44
|
+
NitroHashcashNative PRIVATE
|
|
45
|
+
-DFOLLY_NO_CONFIG=1
|
|
46
|
+
-DFOLLY_HAVE_CLOCK_GETTIME=1
|
|
47
|
+
-DFOLLY_USE_LIBCPP=1
|
|
48
|
+
-DFOLLY_CFG_NO_COROUTINES=1
|
|
49
|
+
-DFOLLY_MOBILE=1
|
|
50
|
+
-DFOLLY_HAVE_RECVMMSG=1
|
|
51
|
+
-DFOLLY_HAVE_PTHREAD=1
|
|
52
|
+
# Once we target android-23 above, we can comment
|
|
53
|
+
# the following line. NDK uses GNU style stderror_r() after API 23.
|
|
54
|
+
-DFOLLY_HAVE_XSI_STRERROR_R=1
|
|
55
|
+
)
|
|
56
|
+
|
|
57
|
+
# Add all libraries required by the generated specs
|
|
58
|
+
find_package(fbjni REQUIRED) # <-- Used for communication between Java <-> C++
|
|
59
|
+
find_package(ReactAndroid REQUIRED) # <-- Used to set up React Native bindings (e.g. CallInvoker/TurboModule)
|
|
60
|
+
find_package(react-native-nitro-modules REQUIRED) # <-- Used to create all HybridObjects and use the Nitro core library
|
|
61
|
+
|
|
62
|
+
# Link all libraries together
|
|
63
|
+
target_link_libraries(
|
|
64
|
+
NitroHashcashNative
|
|
65
|
+
fbjni::fbjni # <-- Facebook C++ JNI helpers
|
|
66
|
+
ReactAndroid::jsi # <-- RN: JSI
|
|
67
|
+
react-native-nitro-modules::NitroModules # <-- NitroModules Core :)
|
|
68
|
+
)
|
|
69
|
+
|
|
70
|
+
# Link react-native (different prefab between RN 0.75 and RN 0.76)
|
|
71
|
+
if(ReactAndroid_VERSION_MINOR GREATER_EQUAL 76)
|
|
72
|
+
target_link_libraries(
|
|
73
|
+
NitroHashcashNative
|
|
74
|
+
ReactAndroid::reactnative # <-- RN: Native Modules umbrella prefab
|
|
75
|
+
)
|
|
76
|
+
else()
|
|
77
|
+
target_link_libraries(
|
|
78
|
+
NitroHashcashNative
|
|
79
|
+
ReactAndroid::react_nativemodule_core # <-- RN: TurboModules Core
|
|
80
|
+
)
|
|
81
|
+
endif()
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
///
|
|
2
|
+
/// NitroHashcashNative+autolinking.gradle
|
|
3
|
+
/// This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
/// https://github.com/mrousavy/nitro
|
|
5
|
+
/// Copyright © 2025 Marc Rousavy @ Margelo
|
|
6
|
+
///
|
|
7
|
+
|
|
8
|
+
/// This is a Gradle file that adds all files generated by Nitrogen
|
|
9
|
+
/// to the current Gradle project.
|
|
10
|
+
///
|
|
11
|
+
/// To use it, add this to your build.gradle:
|
|
12
|
+
/// ```gradle
|
|
13
|
+
/// apply from: '../nitrogen/generated/android/NitroHashcashNative+autolinking.gradle'
|
|
14
|
+
/// ```
|
|
15
|
+
|
|
16
|
+
logger.warn("[NitroModules] 🔥 NitroHashcashNative is boosted by nitro!")
|
|
17
|
+
|
|
18
|
+
android {
|
|
19
|
+
sourceSets {
|
|
20
|
+
main {
|
|
21
|
+
java.srcDirs += [
|
|
22
|
+
// Nitrogen files
|
|
23
|
+
"${project.projectDir}/../nitrogen/generated/android/kotlin"
|
|
24
|
+
]
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
///
|
|
2
|
+
/// NitroHashcashNativeOnLoad.cpp
|
|
3
|
+
/// This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
/// https://github.com/mrousavy/nitro
|
|
5
|
+
/// Copyright © 2025 Marc Rousavy @ Margelo
|
|
6
|
+
///
|
|
7
|
+
|
|
8
|
+
#ifndef BUILDING_NITROHASHCASHNATIVE_WITH_GENERATED_CMAKE_PROJECT
|
|
9
|
+
#error NitroHashcashNativeOnLoad.cpp is not being built with the autogenerated CMakeLists.txt project. Is a different CMakeLists.txt building this?
|
|
10
|
+
#endif
|
|
11
|
+
|
|
12
|
+
#include "NitroHashcashNativeOnLoad.hpp"
|
|
13
|
+
|
|
14
|
+
#include <jni.h>
|
|
15
|
+
#include <fbjni/fbjni.h>
|
|
16
|
+
#include <NitroModules/HybridObjectRegistry.hpp>
|
|
17
|
+
|
|
18
|
+
#include "JHybridHashcashSpec.hpp"
|
|
19
|
+
#include <NitroModules/DefaultConstructableObject.hpp>
|
|
20
|
+
|
|
21
|
+
namespace margelo::nitro::hashcashnative {
|
|
22
|
+
|
|
23
|
+
int initialize(JavaVM* vm) {
|
|
24
|
+
using namespace margelo::nitro;
|
|
25
|
+
using namespace margelo::nitro::hashcashnative;
|
|
26
|
+
using namespace facebook;
|
|
27
|
+
|
|
28
|
+
return facebook::jni::initialize(vm, [] {
|
|
29
|
+
// Register native JNI methods
|
|
30
|
+
margelo::nitro::hashcashnative::JHybridHashcashSpec::registerNatives();
|
|
31
|
+
|
|
32
|
+
// Register Nitro Hybrid Objects
|
|
33
|
+
HybridObjectRegistry::registerHybridObjectConstructor(
|
|
34
|
+
"Hashcash",
|
|
35
|
+
[]() -> std::shared_ptr<HybridObject> {
|
|
36
|
+
static DefaultConstructableObject<JHybridHashcashSpec::javaobject> object("com/margelo/nitro/hashcashnative/HybridHashcash");
|
|
37
|
+
auto instance = object.create();
|
|
38
|
+
return instance->cthis()->shared();
|
|
39
|
+
}
|
|
40
|
+
);
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
} // namespace margelo::nitro::hashcashnative
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
///
|
|
2
|
+
/// NitroHashcashNativeOnLoad.hpp
|
|
3
|
+
/// This file was generated by nitrogen. DO NOT MODIFY THIS FILE.
|
|
4
|
+
/// https://github.com/mrousavy/nitro
|
|
5
|
+
/// Copyright © 2025 Marc Rousavy @ Margelo
|
|
6
|
+
///
|
|
7
|
+
|
|
8
|
+
#include <jni.h>
|
|
9
|
+
#include <NitroModules/NitroDefines.hpp>
|
|
10
|
+
|
|
11
|
+
namespace margelo::nitro::hashcashnative {
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Initializes the native (C++) part of NitroHashcashNative, and autolinks all Hybrid Objects.
|
|
15
|
+
* Call this in your `JNI_OnLoad` function (probably inside `cpp-adapter.cpp`).
|
|
16
|
+
* Example:
|
|
17
|
+
* ```cpp (cpp-adapter.cpp)
|
|
18
|
+
* JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void*) {
|
|
19
|
+
* return margelo::nitro::hashcashnative::initialize(vm);
|
|
20
|
+
* }
|
|
21
|
+
* ```
|
|
22
|
+
*/
|
|
23
|
+
int initialize(JavaVM* vm);
|
|
24
|
+
|
|
25
|
+
} // namespace margelo::nitro::hashcashnative
|