@magic-sdk/react-native-bare 34.0.0 → 34.1.0-canary.964.20943413682.0

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.
@@ -0,0 +1,23 @@
1
+ require "json"
2
+
3
+ package = JSON.parse(File.read(File.join(__dir__, "package.json")))
4
+
5
+ Pod::Spec.new do |s|
6
+ s.name = "MagicSdkReactNativeBare"
7
+ s.version = package["version"]
8
+ s.summary = package["description"]
9
+ s.description = <<-DESC
10
+ #{package["description"]}
11
+ DESC
12
+ s.homepage = package["homepage"]
13
+ s.license = package["license"]
14
+ s.author = package["author"]
15
+ s.platforms = { :ios => "13.4" }
16
+ s.source = { :git => package["repository"]["url"].gsub(/.git$/, ''), :tag => "#{s.version}" }
17
+
18
+ # This is a pure JavaScript package
19
+ # Native dependencies (react-native-keychain, react-native-device-crypto) will be
20
+ # automatically linked via React Native autolinking when this package is installed
21
+
22
+ s.dependency "React-Core"
23
+ end
package/dist/cjs/index.js CHANGED
@@ -1,2 +1,28 @@
1
- "use strict";var G=Object.create;var y=Object.defineProperty;var $=Object.getOwnPropertyDescriptor;var K=Object.getOwnPropertyNames;var Y=Object.getPrototypeOf,q=Object.prototype.hasOwnProperty;var z=(o,r)=>{for(var n in r)y(o,n,{get:r[n],enumerable:!0})},g=(o,r,n,s)=>{if(r&&typeof r=="object"||typeof r=="function")for(let i of K(r))!q.call(o,i)&&i!==n&&y(o,i,{get:()=>r[i],enumerable:!(s=$(r,i))||s.enumerable});return o},t=(o,r,n)=>(g(o,r,"default"),n&&g(n,r,"default")),m=(o,r,n)=>(n=o!=null?G(Y(o)):{},g(r||!o||!o.__esModule?y(n,"default",{value:o,enumerable:!0}):n,o)),J=o=>g(y({},"__esModule",{value:!0}),o);var e={};z(e,{Magic:()=>Z,useInternetConnection:()=>h});module.exports=J(e);var ce=require("regenerator-runtime/runtime"),D=require("@magic-sdk/provider"),X=m(require("process")),S=require("whatwg-url"),R=require("buffer"),T=m(require("lodash")),L=require("react-native-device-info");var a=m(require("react")),v=require("react-native"),V=require("react-native-webview"),O=require("react-native-safe-area-context"),E=require("@magic-sdk/provider"),P=require("lodash"),f=m(require("@react-native-async-storage/async-storage")),_=require("react-native-event-listeners");var u=require("react"),C=m(require("@react-native-community/netinfo")),h=()=>{let[o,r]=(0,u.useState)(!0);return(0,u.useEffect)(()=>{let n=s=>{r(!!s.isConnected)};return C.default.addEventListener(n)},[]),o};var A="MAGIC_PAYLOAD_FLAG_TYPED_ARRAY",H="open_in_device_browser",j="#FFFFFF",M="msg_posted_after_inactivity_event",w="lastMessageTime";function Q(){return v.StyleSheet.create({"magic-webview":{flex:1,backgroundColor:"transparent"},"webview-container":{flex:1,width:"100%",backgroundColor:"transparent",position:"absolute",top:0,left:0,right:0,bottom:0},show:{zIndex:1e4,elevation:1e4},hide:{zIndex:-1e4,elevation:0}})}var b=class extends E.ViewController{webView;container;styles;init(){this.webView=null,this.container=null,this.styles=Q()}Relayer=r=>{let[n,s]=(0,a.useState)(!1),[i,l]=(0,a.useState)(!0),d=h();(0,a.useEffect)(()=>{this.isConnectedToInternet=d},[d]),(0,a.useEffect)(()=>(f.default.setItem(w,""),()=>{this.isReadyForRequest=!1}),[]),(0,a.useEffect)(()=>{_.EventRegister.addEventListener(M,async c=>{this.isReadyForRequest=!1,l(!1),this.post(c.msgType,c.payload),await f.default.setItem(w,new Date().toISOString())})},[]),(0,a.useEffect)(()=>{i||setTimeout(()=>l(!0),10)},[i]);let N=(0,a.useCallback)(c=>{this.webView=c},[]),k=(0,a.useCallback)(c=>{this.container={...c,showOverlay:F,hideOverlay:U}},[]),F=(0,a.useCallback)(()=>{s(!0)},[]),U=(0,a.useCallback)(()=>{s(!1)},[]),W=(0,a.useMemo)(()=>[this.styles["webview-container"],n?{...this.styles.show,backgroundColor:r??j}:this.styles.hide],[n]),B=(0,a.useCallback)(c=>{this.handleReactNativeWebViewMessage(c)},[]);return i?a.default.createElement(O.SafeAreaView,{ref:k,style:W},a.default.createElement(V.WebView,{ref:N,source:{uri:`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`},onMessage:B,style:this.styles["magic-webview"],webviewDebuggingEnabled:!0,autoManageStatusBarEnabled:!1,onShouldStartLoadWithRequest:c=>new URLSearchParams(c.url.split("?")[1]).get(H)?(v.Linking.openURL(c.url),!1):!0})):null};handleReactNativeWebViewMessage(r){let n=new URL(`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`);if(r.nativeEvent&&typeof r.nativeEvent.data=="string"&&(r.nativeEvent.url===`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`||r.nativeEvent.url===`${this.endpoint}/send/?params=${this.parameters}`||r.nativeEvent.url===this.endpoint||r.nativeEvent.title===`${n.hostname}/send/${n.search}`)){let s=JSON.parse(r.nativeEvent.data,(i,l)=>{try{if(l&&typeof l=="object"&&l.flag&&l.flag===A)return new global[l.constructor](l.data.split(","))}catch(d){console.log("Error parsing data",d)}return l});if(s&&s.msgType&&this.messageHandlers.size){s.response=s.response??{};let i={data:s};for(let l of this.messageHandlers.values())l(i)}}}async msgPostedAfterInactivity(){let r=await f.default.getItem(w);if(r){let n=new Date(r).getTime(),s=new Date().getTime(),i=5*60*1e3;return s-n>i}return!1}hideOverlay(){this.container&&this.container.hideOverlay()}showOverlay(){this.container&&this.container.showOverlay()}async _post(r){if(await this.msgPostedAfterInactivity()){_.EventRegister.emit(M,r);return}if(this.webView&&this.webView.postMessage)this.webView.postMessage(JSON.stringify(r,(n,s)=>(0,P.isTypedArray)(s)?{constructor:s.constructor.name,data:s.toString(),flag:A}:s)),f.default.setItem(w,new Date().toISOString());else throw(0,E.createModalNotReadyError)()}checkRelayerExistsInDOM(){return Promise.resolve(!0)}reloadRelayer(){return Promise.resolve(void 0)}};var x=require("@magic-sdk/provider"),I=class extends x.SDKBase{get Relayer(){return this.overlay.Relayer}};var p=m(require("@react-native-async-storage/async-storage"));t(e,require("@magic-sdk/provider"),module.exports);t(e,require("@magic-sdk/types"),module.exports);global.process=T.default.merge(global.process,X);global.process.browser=!1;global.Buffer=R.Buffer;global.URL=S.URL;global.URLSearchParams=S.URLSearchParams;global.btoa=o=>R.Buffer.from(o,"binary").toString("base64");global.atob=o=>R.Buffer.from(o,"base64").toString("binary");var Z=(0,D.createSDK)(I,{platform:"react-native",sdkName:"@magic-sdk/react-native-bare",version:"34.0.0",bundleId:(0,L.getBundleId)(),defaultEndpoint:"https://box.magic.link/",ViewController:b,configureStorage:async()=>({ready:async()=>Promise.resolve(!0),getItem:p.default.getItem,setItem:p.default.setItem,removeItem:p.default.removeItem,clear:p.default.clear,length:()=>{},key:()=>{},keys:p.default.getAllKeys,iterate:()=>{}})});0&&(module.exports={Magic,useInternetConnection,...require("@magic-sdk/provider"),...require("@magic-sdk/types")});
1
+ "use strict";var ce=Object.create;var E=Object.defineProperty;var le=Object.getOwnPropertyDescriptor;var pe=Object.getOwnPropertyNames;var me=Object.getPrototypeOf,de=Object.prototype.hasOwnProperty;var fe=(t,e)=>{for(var n in e)E(t,n,{get:e[n],enumerable:!0})},b=(t,e,n,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let s of pe(e))!de.call(t,s)&&s!==n&&E(t,s,{get:()=>e[s],enumerable:!(r=le(e,s))||r.enumerable});return t},a=(t,e,n)=>(b(t,e,"default"),n&&b(n,e,"default")),m=(t,e,n)=>(n=t!=null?ce(me(t)):{},b(e||!t||!t.__esModule?E(n,"default",{value:t,enumerable:!0}):n,t)),ue=t=>b(E({},"__esModule",{value:!0}),t);var o={};fe(o,{Magic:()=>xe,useInternetConnection:()=>S});module.exports=ue(o);var Ze=require("regenerator-runtime/runtime"),re=require("@magic-sdk/provider"),Se=m(require("process")),P=require("whatwg-url"),D=require("buffer"),ne=m(require("lodash")),oe=require("react-native-device-info");var i=m(require("react")),T=require("react-native"),Q=require("react-native-webview"),Z=require("react-native-safe-area-context"),C=require("@magic-sdk/provider"),ee=require("lodash"),v=m(require("@react-native-async-storage/async-storage")),O=require("react-native-event-listeners");var I=require("react"),L=m(require("@react-native-community/netinfo")),S=()=>{let[t,e]=(0,I.useState)(!0);return(0,I.useEffect)(()=>{let n=r=>{e(!!r.isConnected)};return L.default.addEventListener(n)},[]),t};var u=m(require("react-native-keychain"));var U=m(require("react-native-device-info")),ye={dpop:"magic.sdk.dpop",refreshToken:"magic.sdk.rt",refreshTokenService:"magic.sdk.rt.service"};function h(t){return`${U.default.getBundleId()}.${ye[t]}`}var K=h("refreshTokenService"),ge=h("refreshToken"),y=null,V=async t=>{if(y===t)return!0;try{let e=await u.setGenericPassword(ge,t,{service:K,accessible:u.ACCESSIBLE.AFTER_FIRST_UNLOCK_THIS_DEVICE_ONLY});return y=t,e}catch(e){return console.error("Failed to set refresh token in keychain",e),!1}},F=async()=>{if(y!==null)return y;try{let t=await u.getGenericPassword({service:K});return t?(y=t.password,y):null}catch(t){return console.error("Failed to get refresh token in keychain",t),null}};var J=m(require("react-native-uuid"));var he=t=>{let e="",n=t.byteLength;for(let r=0;r<n;r++)e+=String.fromCharCode(t[r]);return btoa(e)},x=t=>{let e=atob(t),n=e.length,r=new Uint8Array(n);for(let s=0;s<n;s++)r[s]=e.charCodeAt(s);return r},we=t=>{if(typeof TextEncoder<"u")return new TextEncoder().encode(t);let e=[];for(let n=0;n<t.length;n++){let r=t.charCodeAt(n);r<128?e.push(r):r<2048?e.push(192|r>>6,128|r&63):r<55296||r>=57344?e.push(224|r>>12,128|r>>6&63,128|r&63):(n++,r=65536+((r&1023)<<10|t.charCodeAt(n)&1023),e.push(240|r>>18,128|r>>12&63,128|r>>6&63,128|r&63))}return new Uint8Array(e)},f=t=>{let e;return typeof t=="string"?e=we(t):e=t,he(e).replace(/\+/g,"-").replace(/\//g,"_").replace(/=+$/,"")};var B=t=>{let e=x(t),n=e.subarray(e.length-65);if(n[0]!==4)throw new Error("Invalid Public Key format: Expected uncompressed point");let r=n.subarray(1,33),s=n.subarray(33,65);return{kty:"EC",crv:"P-256",x:f(r),y:f(s)}};var $="ES256",W="dpop+jwt";var G=t=>{let e=x(t),n=2;if(e[n]!==2)throw new Error("Invalid DER: R tag missing");n++;let r=e[n++],s=e.subarray(n,n+r);if(n+=r,r===33&&s[0]===0&&(s=s.subarray(1)),e[n]!==2)throw new Error("Invalid DER: S tag missing");n++;let c=e[n++],l=e.subarray(n,n+c);c===33&&l[0]===0&&(l=l.subarray(1));let d=new Uint8Array(64);return d.set(s,32-s.length),d.set(l,64-l.length),f(d)};var w=m(require("react-native-device-crypto"));var Y=h("dpop"),j=async()=>{try{let t=await w.default.getOrCreateAsymmetricKey(Y,{accessLevel:w.AccessLevel.ALWAYS,invalidateOnNewBiometry:!1}),e=B(t),r={iat:Math.floor(Date.now()/1e3),jti:J.default.v4()},c=f(JSON.stringify({typ:W,alg:$,jwk:e})),l=f(JSON.stringify(r)),d=`${c}.${l}`,_=await w.default.sign(Y,d,{biometryTitle:"Sign DPoP",biometrySubTitle:"Sign DPoP",biometryDescription:"Sign DPoP"}),N=G(_);return`${d}.${N}`}catch(t){return console.error("DPoP Generation Error:",t),null}};var k=require("react-native"),ve=[{name:"react-native-keychain",nativeModuleName:"RNKeychainManager",packageName:"react-native-keychain"},{name:"react-native-device-crypto",nativeModuleName:"DeviceCrypto",packageName:"react-native-device-crypto"}],q=!1,H=()=>{if(q)return;let t=[];for(let e of ve)k.NativeModules[e.nativeModuleName]||t.push(e);if(t.length>0){q=!0;let e=k.Platform.OS,n=t.map(l=>` - ${l.packageName}`).join(`
2
+ `),r=t.map(l=>`npm install ${l.packageName}`).join(`
3
+ `);console.warn(`@magic-sdk/react-native-bare: Missing native modules detected.
4
+
5
+ The following native modules are not linked:
6
+ ${n}
7
+
8
+ The SDK will continue to work, but some security features may not function properly.
9
+
10
+ Note: If you're running in a simulator/emulator, some native modules (like react-native-device-crypto)
11
+ require hardware features (Secure Enclave) that are only available on physical devices.
12
+
13
+ If you're on a physical device and see this warning, please ensure the packages are installed and linked:
14
+
15
+ 1. Install the missing packages:
16
+ ${r}
17
+
18
+ 2. Link the native modules:
19
+ ${e==="ios"?`
20
+ For iOS, run:
21
+ cd ios && pod install && cd ..
22
+ `:""}${e==="android"?`
23
+ For Android, rebuild your app:
24
+ npx react-native run-android
25
+ `:""}
26
+ 3. Rebuild your app completely.
27
+ `)}};var z="MAGIC_PAYLOAD_FLAG_TYPED_ARRAY",be="open_in_device_browser",Ee="#FFFFFF",X="msg_posted_after_inactivity_event",R="lastMessageTime";function Ie(){return T.StyleSheet.create({"magic-webview":{flex:1,backgroundColor:"transparent"},"webview-container":{flex:1,width:"100%",backgroundColor:"transparent",position:"absolute",top:0,left:0,right:0,bottom:0},show:{zIndex:1e4,elevation:1e4},hide:{zIndex:-1e4,elevation:0}})}var A=class extends C.ViewController{webView;container;styles;init(){this.webView=null,this.container=null,this.styles=Ie()}Relayer=e=>{let[n,r]=(0,i.useState)(!1),[s,c]=(0,i.useState)(!0),l=S();(0,i.useEffect)(()=>{H()},[]),(0,i.useEffect)(()=>{this.isConnectedToInternet=l},[l]),(0,i.useEffect)(()=>(v.default.setItem(R,""),()=>{this.isReadyForRequest=!1}),[]),(0,i.useEffect)(()=>{O.EventRegister.addEventListener(X,async p=>{this.isReadyForRequest=!1,c(!1),this.post(p.msgType,p.payload),await v.default.setItem(R,new Date().toISOString())})},[]),(0,i.useEffect)(()=>{s||setTimeout(()=>c(!0),10)},[s]);let d=(0,i.useCallback)(p=>{this.webView=p},[]),_=(0,i.useCallback)(p=>{this.container={...p,showOverlay:N,hideOverlay:ae}},[]),N=(0,i.useCallback)(()=>{r(!0)},[]),ae=(0,i.useCallback)(()=>{r(!1)},[]),se=(0,i.useMemo)(()=>[this.styles["webview-container"],n?{...this.styles.show,backgroundColor:e??Ee}:this.styles.hide],[n]),ie=(0,i.useCallback)(p=>{this.handleReactNativeWebViewMessage(p)},[]);return s?i.default.createElement(Z.SafeAreaView,{ref:_,style:se},i.default.createElement(Q.WebView,{ref:d,source:{uri:`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`},onMessage:ie,style:this.styles["magic-webview"],webviewDebuggingEnabled:!0,autoManageStatusBarEnabled:!1,onShouldStartLoadWithRequest:p=>new URLSearchParams(p.url.split("?")[1]).get(be)?(T.Linking.openURL(p.url),!1):!0})):null};handleReactNativeWebViewMessage(e){let n=new URL(`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`);if(e.nativeEvent&&typeof e.nativeEvent.data=="string"&&(e.nativeEvent.url===`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`||e.nativeEvent.url===`${this.endpoint}/send/?params=${this.parameters}`||e.nativeEvent.url===this.endpoint||e.nativeEvent.title===`${n.hostname}/send/${n.search}`)){let r=JSON.parse(e.nativeEvent.data,(s,c)=>{try{if(c&&typeof c=="object"&&c.flag&&c.flag===z)return new global[c.constructor](c.data.split(","))}catch(l){console.log("Error parsing data",l)}return c});if(r&&r.msgType&&this.messageHandlers.size){r.response=r.response??{};let s={data:r};for(let c of this.messageHandlers.values())c(s)}}}async msgPostedAfterInactivity(){let e=await v.default.getItem(R);if(e){let n=new Date(e).getTime(),r=new Date().getTime(),s=5*60*1e3;return r-n>s}return!1}hideOverlay(){this.container&&this.container.hideOverlay()}showOverlay(){this.container&&this.container.showOverlay()}async _post(e){if(await this.msgPostedAfterInactivity()){O.EventRegister.emit(X,e);return}if(this.webView&&this.webView.postMessage)this.webView.postMessage(JSON.stringify(e,(n,r)=>(0,ee.isTypedArray)(r)?{constructor:r.constructor.name,data:r.toString(),flag:z}:r)),v.default.setItem(R,new Date().toISOString());else throw(0,C.createModalNotReadyError)()}async persistMagicEventRefreshToken(e){e?.data?.rt&&await V(e.data.rt)}async getRT(){return await F()}async getJWT(){try{return await j()}catch{return null}}checkRelayerExistsInDOM(){return Promise.resolve(!0)}reloadRelayer(){return Promise.resolve(void 0)}};var te=require("@magic-sdk/provider"),M=class extends te.SDKBase{get Relayer(){return this.overlay.Relayer}};var g=m(require("@react-native-async-storage/async-storage"));a(o,require("@magic-sdk/provider"),module.exports);a(o,require("@magic-sdk/types"),module.exports);global.process=ne.default.merge(global.process,Se);global.process.browser=!1;global.Buffer=D.Buffer;global.URL=P.URL;global.URLSearchParams=P.URLSearchParams;global.btoa=t=>D.Buffer.from(t,"binary").toString("base64");global.atob=t=>D.Buffer.from(t,"base64").toString("binary");var xe=(0,re.createSDK)(M,{platform:"react-native",sdkName:"@magic-sdk/react-native-bare",version:"34.0.0",bundleId:(0,oe.getBundleId)(),defaultEndpoint:"https://box.magic.link/",ViewController:A,configureStorage:async()=>({ready:async()=>Promise.resolve(!0),getItem:g.default.getItem,setItem:g.default.setItem,removeItem:g.default.removeItem,clear:g.default.clear,length:()=>{},key:()=>{},keys:g.default.getAllKeys,iterate:()=>{}})});0&&(module.exports={Magic,useInternetConnection,...require("@magic-sdk/provider"),...require("@magic-sdk/types")});
2
28
  //# sourceMappingURL=index.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
- "sources": ["../../src/index.ts", "../../src/react-native-webview-controller.tsx", "../../src/hooks.ts", "../../src/react-native-sdk-base.ts"],
4
- "sourcesContent": ["/* istanbul ignore file */\n\nimport 'regenerator-runtime/runtime';\n\nimport { createSDK, InstanceWithExtensions, MagicSDKExtensionsOption } from '@magic-sdk/provider';\nimport * as processPolyfill from 'process';\nimport { URL as URLPolyfill, URLSearchParams as URLSearchParamsPolyfill } from 'whatwg-url';\nimport { Buffer } from 'buffer';\nimport _ from 'lodash';\nimport { getBundleId } from 'react-native-device-info';\nimport { ReactNativeWebViewController } from './react-native-webview-controller';\nimport { SDKBaseReactNative } from './react-native-sdk-base';\nimport type localForage from 'localforage';\nimport AsyncStorage from '@react-native-async-storage/async-storage';\n\n// Web3 assumes a browser context, so we need\n// to provide `btoa` and `atob` shims.\n\n// We expect `global.process` to be a Node Process for web3.js usage\n// so we replace it here.\nglobal.process = _.merge(global.process, processPolyfill);\n\n(global.process as any).browser = false;\n\n// WHATWG URL requires global `Buffer` access.\nglobal.Buffer = Buffer;\n\n// Setup global WHATWG URL polyfills\nglobal.URL = URLPolyfill as any;\nglobal.URLSearchParams = URLSearchParamsPolyfill as any;\n\n/* istanbul ignore next */\nglobal.btoa = str => Buffer.from(str, 'binary').toString('base64');\n/* istanbul ignore next */\nglobal.atob = b64Encoded => Buffer.from(b64Encoded, 'base64').toString('binary');\n\nexport * from '@magic-sdk/provider';\nexport * from '@magic-sdk/types';\n\nexport const Magic = createSDK(SDKBaseReactNative, {\n platform: 'react-native',\n sdkName: '@magic-sdk/react-native-bare',\n version: process.env.BARE_REACT_NATIVE_VERSION!,\n bundleId: getBundleId(),\n defaultEndpoint: 'https://box.magic.link/',\n ViewController: ReactNativeWebViewController,\n configureStorage: /* istanbul ignore next */ async () => {\n return {\n ready: async () => Promise.resolve(true),\n getItem: AsyncStorage.getItem,\n setItem: AsyncStorage.setItem,\n removeItem: AsyncStorage.removeItem,\n clear: AsyncStorage.clear,\n length: () => {},\n key: () => {},\n keys: AsyncStorage.getAllKeys,\n iterate: () => {},\n } as unknown as typeof localForage;\n },\n});\n\nexport type Magic<T extends MagicSDKExtensionsOption<any> = MagicSDKExtensionsOption> = InstanceWithExtensions<\n SDKBaseReactNative,\n T\n>;\n\nexport { useInternetConnection } from './hooks';\n", "import React, { useState, useCallback, useMemo, useEffect } from 'react';\nimport { Linking, StyleSheet, View } from 'react-native';\nimport { WebView } from 'react-native-webview';\nimport { SafeAreaView } from 'react-native-safe-area-context';\nimport { ViewController, createModalNotReadyError } from '@magic-sdk/provider';\nimport { MagicMessageEvent } from '@magic-sdk/types';\nimport { isTypedArray } from 'lodash';\nimport AsyncStorage from '@react-native-async-storage/async-storage';\nimport { EventRegister } from 'react-native-event-listeners';\n/* global NodeJS */\nimport Global = NodeJS.Global;\nimport { useInternetConnection } from './hooks';\n\nconst MAGIC_PAYLOAD_FLAG_TYPED_ARRAY = 'MAGIC_PAYLOAD_FLAG_TYPED_ARRAY';\nconst OPEN_IN_DEVICE_BROWSER = 'open_in_device_browser';\nconst DEFAULT_BACKGROUND_COLOR = '#FFFFFF';\nconst MSG_POSTED_AFTER_INACTIVITY_EVENT = 'msg_posted_after_inactivity_event';\nconst LAST_MESSAGE_TIME = 'lastMessageTime';\n/**\n * Builds the Magic `<WebView>` overlay styles. These base styles enable\n * `<WebView>` UI to render above all other DOM content.\n */\nfunction createWebViewStyles() {\n return StyleSheet.create({\n 'magic-webview': {\n flex: 1,\n backgroundColor: 'transparent',\n },\n\n 'webview-container': {\n flex: 1,\n width: '100%',\n backgroundColor: 'transparent',\n position: 'absolute',\n top: 0,\n left: 0,\n right: 0,\n bottom: 0,\n },\n\n show: {\n zIndex: 10000,\n elevation: 10000,\n },\n\n hide: {\n zIndex: -10000,\n elevation: 0,\n },\n });\n}\n\n/**\n * Overloads React Native's `<View>` with helper methods we require to hide/show\n * the rendered `<WebView>`.\n */\ninterface ViewWrapper extends View {\n showOverlay: () => void;\n hideOverlay: () => void;\n}\n\n/**\n * View controller for the Magic `<WebView>` overlay.\n */\nexport class ReactNativeWebViewController extends ViewController {\n private webView!: WebView | null;\n private container!: ViewWrapper | null;\n private styles: any;\n\n protected init() {\n this.webView = null;\n this.container = null;\n this.styles = createWebViewStyles();\n }\n\n /**\n * Renders a React Native `<WebView>` with built-in message handling to and\n * from the Magic `<iframe>` context.\n */\n // Validating this logic requires lots of React-specific boilerplate. We will\n // revisit this method for unit testing in the future. For now, manual testing\n // is sufficient (this logic is stable right now and not expected to change in\n // the foreseeable future).\n /* istanbul ignore next */\n public Relayer: React.FC<{ backgroundColor?: string }> = backgroundColor => {\n const [show, setShow] = useState(false);\n const [mountOverlay, setMountOverlay] = useState(true);\n const isConnected = useInternetConnection();\n\n useEffect(() => {\n this.isConnectedToInternet = isConnected;\n }, [isConnected]);\n\n useEffect(() => {\n // reset lastMessage when webview is first mounted\n AsyncStorage.setItem(LAST_MESSAGE_TIME, '');\n return () => {\n this.isReadyForRequest = false;\n };\n }, []);\n\n useEffect(() => {\n EventRegister.addEventListener(MSG_POSTED_AFTER_INACTIVITY_EVENT, async message => {\n // If inactivity has been determined, the message is posted only after a brief\n // unmount and re-mount of the webview. This is to ensure the webview is accepting messages.\n // iOS kills webview processes after a certain period of inactivity, like when the app is\n // on background for long periods of time.\n this.isReadyForRequest = false;\n setMountOverlay(false);\n this.post(message.msgType, message.payload);\n await AsyncStorage.setItem(LAST_MESSAGE_TIME, new Date().toISOString());\n });\n }, []);\n\n useEffect(() => {\n if (!mountOverlay) {\n // Briefly unmount and re-mount the webview to ensure it's ready to accept messages.\n setTimeout(() => setMountOverlay(true), 10);\n }\n }, [mountOverlay]);\n\n /**\n * Saves a reference to the underlying `<WebView>` node so we can interact\n * with incoming messages.\n */\n const webViewRef = useCallback((webView: any): void => {\n this.webView = webView;\n }, []);\n\n /**\n * Saves a reference to the underlying `<View>` node so we can interact with\n * display styles.\n */\n const containerRef = useCallback((view: any): void => {\n this.container = {\n ...view,\n showOverlay,\n hideOverlay,\n };\n }, []);\n\n /**\n * Show the Magic `<WebView>` overlay.\n */\n const showOverlay = useCallback(() => {\n setShow(true);\n }, []);\n\n /**\n * Hide the Magic `<WebView>` overlay.\n */\n const hideOverlay = useCallback(() => {\n setShow(false);\n }, []);\n\n const containerStyles = useMemo(() => {\n return [\n this.styles['webview-container'],\n show\n ? {\n ...this.styles.show,\n backgroundColor: backgroundColor ?? DEFAULT_BACKGROUND_COLOR,\n }\n : this.styles.hide,\n ];\n }, [show]);\n\n const handleWebViewMessage = useCallback((event: unknown) => {\n this.handleReactNativeWebViewMessage(event);\n }, []);\n\n if (!mountOverlay) {\n return null;\n }\n\n return (\n <SafeAreaView ref={containerRef} style={containerStyles}>\n <WebView\n ref={webViewRef}\n source={{ uri: `${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}` }}\n onMessage={handleWebViewMessage}\n style={this.styles['magic-webview']}\n webviewDebuggingEnabled\n autoManageStatusBarEnabled={false}\n onShouldStartLoadWithRequest={event => {\n const queryParams = new URLSearchParams(event.url.split('?')[1]);\n const openInDeviceBrowser = queryParams.get(OPEN_IN_DEVICE_BROWSER);\n\n if (openInDeviceBrowser) {\n Linking.openURL(event.url);\n return false;\n }\n return true;\n }}\n />\n </SafeAreaView>\n );\n };\n\n /**\n * Route incoming messages from a React Native `<WebView>`.\n */\n private handleReactNativeWebViewMessage(event: any) {\n const url = new URL(`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`);\n\n if (\n event.nativeEvent &&\n typeof event.nativeEvent.data === 'string' &&\n /* Backward compatible */\n (event.nativeEvent.url === `${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}` ||\n event.nativeEvent.url === `${this.endpoint}/send/?params=${this.parameters}` ||\n event.nativeEvent.url === this.endpoint ||\n event.nativeEvent.title === `${url.hostname}/send/${url.search}`)\n ) {\n // Special parsing logic when dealing with TypedArray in the payload\n // Such change is required as JSON.stringify will manipulate the object and cause exceptions during parsing\n // The typed Array is stringified in Mgbox with a flag as notation.\n const data: any = JSON.parse(event.nativeEvent.data, (key, value) => {\n try {\n if (value && typeof value === 'object' && value.flag && value.flag === MAGIC_PAYLOAD_FLAG_TYPED_ARRAY) {\n return new (global[value.constructor as keyof Global] as any)(value.data.split(','));\n }\n\n // silently handles exception and return the original copy\n } catch (e) {\n console.log('Error parsing data', e);\n }\n return value;\n });\n if (data && data.msgType && this.messageHandlers.size) {\n // If the response object is undefined, we ensure it's at least an\n // empty object before passing to the event listener.\n\n data.response = data.response ?? {};\n\n // Reconstruct event from RN event\n const magicEvent: MagicMessageEvent = { data } as MagicMessageEvent;\n for (const handler of this.messageHandlers.values()) {\n handler(magicEvent);\n }\n }\n }\n }\n\n private async msgPostedAfterInactivity() {\n const lastPostTimestamp: string | null = await AsyncStorage.getItem(LAST_MESSAGE_TIME);\n if (lastPostTimestamp) {\n const lastPostDate = new Date(lastPostTimestamp).getTime();\n const now = new Date().getTime();\n const fiveMinutes = 5 * 60 * 1000;\n\n // there's been inactivity if the new message is posted\n // 5 min or more after the last message.\n return now - lastPostDate > fiveMinutes;\n }\n return false;\n }\n\n protected hideOverlay() {\n if (this.container) this.container.hideOverlay();\n }\n\n protected showOverlay() {\n if (this.container) this.container.showOverlay();\n }\n\n protected async _post(data: any) {\n if (await this.msgPostedAfterInactivity()) {\n EventRegister.emit(MSG_POSTED_AFTER_INACTIVITY_EVENT, data);\n return;\n }\n if (this.webView && this.webView.postMessage) {\n this.webView.postMessage(\n JSON.stringify(data, (key, value) => {\n // parse Typed Array to Stringify object\n if (isTypedArray(value)) {\n return {\n constructor: value.constructor.name,\n data: value.toString(),\n flag: MAGIC_PAYLOAD_FLAG_TYPED_ARRAY,\n };\n }\n return value;\n }),\n );\n AsyncStorage.setItem(LAST_MESSAGE_TIME, new Date().toISOString());\n } else {\n throw createModalNotReadyError();\n }\n }\n\n // Todo - implement these methods\n /* istanbul ignore next */\n protected checkRelayerExistsInDOM(): Promise<boolean> {\n return Promise.resolve(true);\n }\n\n /* istanbul ignore next */\n protected reloadRelayer(): Promise<void> {\n return Promise.resolve(undefined);\n }\n}\n", "import { useEffect, useState } from 'react';\nimport NetInfo, { NetInfoState } from '@react-native-community/netinfo';\n\nexport const useInternetConnection = () => {\n const [isConnected, setIsConnected] = useState(true);\n useEffect(() => {\n const handleConnectionChange = (connectionInfo: NetInfoState) => {\n setIsConnected(!!connectionInfo.isConnected);\n };\n\n // Subscribe to connection changes and cleanup on unmount\n return NetInfo.addEventListener(handleConnectionChange);\n }, []);\n\n return isConnected;\n};\n", "import { SDKBase } from '@magic-sdk/provider';\nimport { ReactNativeWebViewController } from './react-native-webview-controller';\n\nexport class SDKBaseReactNative extends SDKBase {\n public get Relayer() {\n return (this.overlay as ReactNativeWebViewController).Relayer;\n }\n}\n"],
5
- "mappings": "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",
6
- "names": ["index_exports", "__export", "Magic", "useInternetConnection", "__toCommonJS", "import_runtime", "import_provider", "processPolyfill", "import_whatwg_url", "import_buffer", "import_lodash", "import_react_native_device_info", "import_react", "import_react_native", "import_react_native_webview", "import_react_native_safe_area_context", "import_provider", "import_lodash", "import_async_storage", "import_react_native_event_listeners", "import_react", "import_netinfo", "useInternetConnection", "isConnected", "setIsConnected", "handleConnectionChange", "connectionInfo", "NetInfo", "MAGIC_PAYLOAD_FLAG_TYPED_ARRAY", "OPEN_IN_DEVICE_BROWSER", "DEFAULT_BACKGROUND_COLOR", "MSG_POSTED_AFTER_INACTIVITY_EVENT", "LAST_MESSAGE_TIME", "createWebViewStyles", "ReactNativeWebViewController", "backgroundColor", "show", "setShow", "mountOverlay", "setMountOverlay", "isConnected", "useInternetConnection", "AsyncStorage", "message", "webViewRef", "webView", "containerRef", "view", "showOverlay", "hideOverlay", "containerStyles", "handleWebViewMessage", "event", "React", "url", "data", "key", "value", "e", "magicEvent", "handler", "lastPostTimestamp", "lastPostDate", "now", "fiveMinutes", "import_provider", "SDKBaseReactNative", "import_async_storage", "__reExport", "index_exports", "_", "processPolyfill", "URLPolyfill", "URLSearchParamsPolyfill", "str", "b64Encoded", "Magic", "SDKBaseReactNative", "ReactNativeWebViewController", "AsyncStorage"]
3
+ "sources": ["../../src/index.ts", "../../src/react-native-webview-controller.tsx", "../../src/hooks.ts", "../../src/native-crypto/keychain.ts", "../../src/native-crypto/utils/key-alias.ts", "../../src/native-crypto/dpop.ts", "../../src/native-crypto/utils/uint8.ts", "../../src/native-crypto/utils/jwk.ts", "../../src/native-crypto/constants/index.ts", "../../src/native-crypto/utils/der.ts", "../../src/native-crypto/check-native-modules.ts", "../../src/react-native-sdk-base.ts"],
4
+ "sourcesContent": ["/* istanbul ignore file */\n\nimport 'regenerator-runtime/runtime';\n\nimport { createSDK, InstanceWithExtensions, MagicSDKExtensionsOption } from '@magic-sdk/provider';\nimport * as processPolyfill from 'process';\nimport { URL as URLPolyfill, URLSearchParams as URLSearchParamsPolyfill } from 'whatwg-url';\nimport { Buffer } from 'buffer';\nimport _ from 'lodash';\nimport { getBundleId } from 'react-native-device-info';\nimport { ReactNativeWebViewController } from './react-native-webview-controller';\nimport { SDKBaseReactNative } from './react-native-sdk-base';\nimport type localForage from 'localforage';\nimport AsyncStorage from '@react-native-async-storage/async-storage';\n\n// Web3 assumes a browser context, so we need\n// to provide `btoa` and `atob` shims.\n\n// We expect `global.process` to be a Node Process for web3.js usage\n// so we replace it here.\nglobal.process = _.merge(global.process, processPolyfill);\n\n(global.process as any).browser = false;\n\n// WHATWG URL requires global `Buffer` access.\nglobal.Buffer = Buffer;\n\n// Setup global WHATWG URL polyfills\nglobal.URL = URLPolyfill as any;\nglobal.URLSearchParams = URLSearchParamsPolyfill as any;\n\n/* istanbul ignore next */\nglobal.btoa = str => Buffer.from(str, 'binary').toString('base64');\n/* istanbul ignore next */\nglobal.atob = b64Encoded => Buffer.from(b64Encoded, 'base64').toString('binary');\n\nexport * from '@magic-sdk/provider';\nexport * from '@magic-sdk/types';\n\nexport const Magic = createSDK(SDKBaseReactNative, {\n platform: 'react-native',\n sdkName: '@magic-sdk/react-native-bare',\n version: process.env.BARE_REACT_NATIVE_VERSION!,\n bundleId: getBundleId(),\n defaultEndpoint: 'https://box.magic.link/',\n ViewController: ReactNativeWebViewController,\n configureStorage: /* istanbul ignore next */ async () => {\n return {\n ready: async () => Promise.resolve(true),\n getItem: AsyncStorage.getItem,\n setItem: AsyncStorage.setItem,\n removeItem: AsyncStorage.removeItem,\n clear: AsyncStorage.clear,\n length: () => {},\n key: () => {},\n keys: AsyncStorage.getAllKeys,\n iterate: () => {},\n } as unknown as typeof localForage;\n },\n});\n\nexport type Magic<T extends MagicSDKExtensionsOption<any> = MagicSDKExtensionsOption> = InstanceWithExtensions<\n SDKBaseReactNative,\n T\n>;\n\nexport { useInternetConnection } from './hooks';\n", "import React, { useState, useCallback, useMemo, useEffect } from 'react';\nimport { Linking, StyleSheet, View } from 'react-native';\nimport { WebView } from 'react-native-webview';\nimport { SafeAreaView } from 'react-native-safe-area-context';\nimport { ViewController, createModalNotReadyError } from '@magic-sdk/provider';\nimport { MagicMessageEvent } from '@magic-sdk/types';\nimport { isTypedArray } from 'lodash';\nimport AsyncStorage from '@react-native-async-storage/async-storage';\nimport { EventRegister } from 'react-native-event-listeners';\n/* global NodeJS */\nimport Global = NodeJS.Global;\nimport { useInternetConnection } from './hooks';\nimport { getRefreshTokenInKeychain, setRefreshTokenInKeychain } from './native-crypto/keychain';\nimport { getDpop } from './native-crypto/dpop';\nimport { checkNativeModules } from './native-crypto/check-native-modules';\n\nconst MAGIC_PAYLOAD_FLAG_TYPED_ARRAY = 'MAGIC_PAYLOAD_FLAG_TYPED_ARRAY';\nconst OPEN_IN_DEVICE_BROWSER = 'open_in_device_browser';\nconst DEFAULT_BACKGROUND_COLOR = '#FFFFFF';\nconst MSG_POSTED_AFTER_INACTIVITY_EVENT = 'msg_posted_after_inactivity_event';\nconst LAST_MESSAGE_TIME = 'lastMessageTime';\n/**\n * Builds the Magic `<WebView>` overlay styles. These base styles enable\n * `<WebView>` UI to render above all other DOM content.\n */\nfunction createWebViewStyles() {\n return StyleSheet.create({\n 'magic-webview': { flex: 1, backgroundColor: 'transparent' },\n\n 'webview-container': {\n flex: 1,\n width: '100%',\n backgroundColor: 'transparent',\n position: 'absolute',\n top: 0,\n left: 0,\n right: 0,\n bottom: 0,\n },\n\n show: { zIndex: 10000, elevation: 10000 },\n\n hide: { zIndex: -10000, elevation: 0 },\n });\n}\n\n/**\n * Overloads React Native's `<View>` with helper methods we require to hide/show\n * the rendered `<WebView>`.\n */\ninterface ViewWrapper extends View {\n showOverlay: () => void;\n hideOverlay: () => void;\n}\n\n/**\n * View controller for the Magic `<WebView>` overlay.\n */\nexport class ReactNativeWebViewController extends ViewController {\n private webView!: WebView | null;\n private container!: ViewWrapper | null;\n private styles: any;\n\n protected init() {\n this.webView = null;\n this.container = null;\n this.styles = createWebViewStyles();\n }\n\n /**\n * Renders a React Native `<WebView>` with built-in message handling to and\n * from the Magic `<iframe>` context.\n */\n // Validating this logic requires lots of React-specific boilerplate. We will\n // revisit this method for unit testing in the future. For now, manual testing\n // is sufficient (this logic is stable right now and not expected to change in\n // the foreseeable future).\n /* istanbul ignore next */\n public Relayer: React.FC<{ backgroundColor?: string }> = backgroundColor => {\n const [show, setShow] = useState(false);\n const [mountOverlay, setMountOverlay] = useState(true);\n const isConnected = useInternetConnection();\n\n useEffect(() => {\n checkNativeModules();\n }, []);\n\n useEffect(() => {\n this.isConnectedToInternet = isConnected;\n }, [isConnected]);\n\n useEffect(() => {\n // reset lastMessage when webview is first mounted\n AsyncStorage.setItem(LAST_MESSAGE_TIME, '');\n return () => {\n this.isReadyForRequest = false;\n };\n }, []);\n\n useEffect(() => {\n EventRegister.addEventListener(MSG_POSTED_AFTER_INACTIVITY_EVENT, async message => {\n // If inactivity has been determined, the message is posted only after a brief\n // unmount and re-mount of the webview. This is to ensure the webview is accepting messages.\n // iOS kills webview processes after a certain period of inactivity, like when the app is\n // on background for long periods of time.\n this.isReadyForRequest = false;\n setMountOverlay(false);\n this.post(message.msgType, message.payload);\n await AsyncStorage.setItem(LAST_MESSAGE_TIME, new Date().toISOString());\n });\n }, []);\n\n useEffect(() => {\n if (!mountOverlay) {\n // Briefly unmount and re-mount the webview to ensure it's ready to accept messages.\n setTimeout(() => setMountOverlay(true), 10);\n }\n }, [mountOverlay]);\n\n /**\n * Saves a reference to the underlying `<WebView>` node so we can interact\n * with incoming messages.\n */\n const webViewRef = useCallback((webView: any): void => {\n this.webView = webView;\n }, []);\n\n /**\n * Saves a reference to the underlying `<View>` node so we can interact with\n * display styles.\n */\n const containerRef = useCallback((view: any): void => {\n this.container = { ...view, showOverlay, hideOverlay };\n }, []);\n\n /**\n * Show the Magic `<WebView>` overlay.\n */\n const showOverlay = useCallback(() => {\n setShow(true);\n }, []);\n\n /**\n * Hide the Magic `<WebView>` overlay.\n */\n const hideOverlay = useCallback(() => {\n setShow(false);\n }, []);\n\n const containerStyles = useMemo(() => {\n return [\n this.styles['webview-container'],\n show ? { ...this.styles.show, backgroundColor: backgroundColor ?? DEFAULT_BACKGROUND_COLOR } : this.styles.hide,\n ];\n }, [show]);\n\n const handleWebViewMessage = useCallback((event: unknown) => {\n this.handleReactNativeWebViewMessage(event);\n }, []);\n\n if (!mountOverlay) {\n return null;\n }\n\n return (\n <SafeAreaView ref={containerRef} style={containerStyles}>\n <WebView\n ref={webViewRef}\n source={{ uri: `${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}` }}\n onMessage={handleWebViewMessage}\n style={this.styles['magic-webview']}\n webviewDebuggingEnabled\n autoManageStatusBarEnabled={false}\n onShouldStartLoadWithRequest={event => {\n const queryParams = new URLSearchParams(event.url.split('?')[1]);\n const openInDeviceBrowser = queryParams.get(OPEN_IN_DEVICE_BROWSER);\n\n if (openInDeviceBrowser) {\n Linking.openURL(event.url);\n return false;\n }\n return true;\n }}\n />\n </SafeAreaView>\n );\n };\n\n /**\n * Route incoming messages from a React Native `<WebView>`.\n */\n private handleReactNativeWebViewMessage(event: any) {\n const url = new URL(`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`);\n\n if (\n event.nativeEvent &&\n typeof event.nativeEvent.data === 'string' &&\n /* Backward compatible */\n (event.nativeEvent.url === `${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}` ||\n event.nativeEvent.url === `${this.endpoint}/send/?params=${this.parameters}` ||\n event.nativeEvent.url === this.endpoint ||\n event.nativeEvent.title === `${url.hostname}/send/${url.search}`)\n ) {\n // Special parsing logic when dealing with TypedArray in the payload\n // Such change is required as JSON.stringify will manipulate the object and cause exceptions during parsing\n // The typed Array is stringified in Mgbox with a flag as notation.\n const data: any = JSON.parse(event.nativeEvent.data, (key, value) => {\n try {\n if (value && typeof value === 'object' && value.flag && value.flag === MAGIC_PAYLOAD_FLAG_TYPED_ARRAY) {\n return new (global[value.constructor as keyof Global] as any)(value.data.split(','));\n }\n\n // silently handles exception and return the original copy\n } catch (e) {\n console.log('Error parsing data', e);\n }\n return value;\n });\n if (data && data.msgType && this.messageHandlers.size) {\n // If the response object is undefined, we ensure it's at least an\n // empty object before passing to the event listener.\n\n data.response = data.response ?? {};\n\n // Reconstruct event from RN event\n const magicEvent: MagicMessageEvent = { data } as MagicMessageEvent;\n for (const handler of this.messageHandlers.values()) {\n handler(magicEvent);\n }\n }\n }\n }\n\n private async msgPostedAfterInactivity() {\n const lastPostTimestamp: string | null = await AsyncStorage.getItem(LAST_MESSAGE_TIME);\n if (lastPostTimestamp) {\n const lastPostDate = new Date(lastPostTimestamp).getTime();\n const now = new Date().getTime();\n const fiveMinutes = 5 * 60 * 1000;\n\n // there's been inactivity if the new message is posted\n // 5 min or more after the last message.\n return now - lastPostDate > fiveMinutes;\n }\n return false;\n }\n\n protected hideOverlay() {\n if (this.container) this.container.hideOverlay();\n }\n\n protected showOverlay() {\n if (this.container) this.container.showOverlay();\n }\n\n protected async _post(data: any) {\n if (await this.msgPostedAfterInactivity()) {\n EventRegister.emit(MSG_POSTED_AFTER_INACTIVITY_EVENT, data);\n return;\n }\n if (this.webView && this.webView.postMessage) {\n this.webView.postMessage(\n JSON.stringify(data, (key, value) => {\n // parse Typed Array to Stringify object\n if (isTypedArray(value)) {\n return {\n constructor: value.constructor.name,\n data: value.toString(),\n flag: MAGIC_PAYLOAD_FLAG_TYPED_ARRAY,\n };\n }\n return value;\n }),\n );\n AsyncStorage.setItem(LAST_MESSAGE_TIME, new Date().toISOString());\n } else {\n throw createModalNotReadyError();\n }\n }\n\n // Overrides parent method to keep refresh token in keychain\n async persistMagicEventRefreshToken(event: MagicMessageEvent) {\n if (!event?.data?.rt) {\n return;\n }\n\n await setRefreshTokenInKeychain(event.data.rt);\n }\n\n // Overrides parent method to retrieve refresh token from keychain while creating a request\n async getRT(): Promise<string | null> {\n return await getRefreshTokenInKeychain();\n }\n\n async getJWT(): Promise<string | null | undefined> {\n try {\n return await getDpop();\n } catch (e) {\n return null;\n }\n }\n\n // Todo - implement these methods\n /* istanbul ignore next */\n protected checkRelayerExistsInDOM(): Promise<boolean> {\n return Promise.resolve(true);\n }\n\n /* istanbul ignore next */\n protected reloadRelayer(): Promise<void> {\n return Promise.resolve(undefined);\n }\n}\n", "import { useEffect, useState } from 'react';\nimport NetInfo, { NetInfoState } from '@react-native-community/netinfo';\n\nexport const useInternetConnection = () => {\n const [isConnected, setIsConnected] = useState(true);\n useEffect(() => {\n const handleConnectionChange = (connectionInfo: NetInfoState) => {\n setIsConnected(!!connectionInfo.isConnected);\n };\n\n // Subscribe to connection changes and cleanup on unmount\n return NetInfo.addEventListener(handleConnectionChange);\n }, []);\n\n return isConnected;\n};\n", "import * as Keychain from 'react-native-keychain';\nimport { getKeyAlias } from './utils/key-alias';\n\nconst SERVICE = getKeyAlias('refreshTokenService');\nconst KEY = getKeyAlias('refreshToken');\n\nlet cachedRefreshToken: string | null = null;\n\nexport const setRefreshTokenInKeychain = async (rt: string) => {\n // Skip write if token hasn't changed\n if (cachedRefreshToken === rt) {\n return true;\n }\n\n try {\n const result = await Keychain.setGenericPassword(KEY, rt, {\n service: SERVICE,\n accessible: Keychain.ACCESSIBLE.AFTER_FIRST_UNLOCK_THIS_DEVICE_ONLY,\n });\n cachedRefreshToken = rt; // Update cache on successful write\n return result;\n } catch (error) {\n console.error('Failed to set refresh token in keychain', error);\n return false;\n }\n};\n\nexport const getRefreshTokenInKeychain = async (): Promise<string | null> => {\n // Return cached value if available\n if (cachedRefreshToken !== null) {\n return cachedRefreshToken;\n }\n\n try {\n const keychainEntry = await Keychain.getGenericPassword({ service: SERVICE });\n if (!keychainEntry) return null;\n\n cachedRefreshToken = keychainEntry.password;\n return cachedRefreshToken;\n } catch (error) {\n console.error('Failed to get refresh token in keychain', error);\n return null;\n }\n};\n\nexport const removeRefreshTokenInKeychain = async () => {\n try {\n cachedRefreshToken = null; // Clear cache\n return await Keychain.resetGenericPassword({ service: SERVICE });\n } catch (error) {\n console.error('Failed to remove refresh token in keychain', error);\n return null;\n }\n};\n", "import DeviceInfo from 'react-native-device-info';\n\nconst KEY_SUFFIX_MAP = {\n dpop: 'magic.sdk.dpop',\n refreshToken: 'magic.sdk.rt',\n refreshTokenService: 'magic.sdk.rt.service',\n};\n\n/**\n * Returns the key alias for the given key.\n * Let's us to safely store the keys in the keychain and avoid conflicts with other apps using magic sdk.\n */\nexport function getKeyAlias(key: keyof typeof KEY_SUFFIX_MAP): string {\n const appId = DeviceInfo.getBundleId();\n return `${appId}.${KEY_SUFFIX_MAP[key]}`;\n}\n", "import uuid from 'react-native-uuid';\nimport { toBase64Url } from './utils/uint8';\nimport { spkiToJwk } from './utils/jwk';\nimport { ALG, TYP } from './constants';\nimport { derToRawSignature } from './utils/der';\nimport { DpopClaims, DpopHeader } from './types';\nimport DeviceCrypto, { AccessLevel } from 'react-native-device-crypto';\nimport { getKeyAlias } from './utils/key-alias';\n\nconst KEY_ALIAS = getKeyAlias('dpop');\n\n/**\n * Generates the DPoP proof compatible with the Python backend.\n * Handles key creation (if missing), JWK construction, and signing.\n */\nexport const getDpop = async (): Promise<string | null> => {\n try {\n // 1. Get or Create Key in Secure Enclave\n // We strictly disable authentication to avoid biometric prompts\n const publicKey = await DeviceCrypto.getOrCreateAsymmetricKey(KEY_ALIAS, {\n accessLevel: AccessLevel.ALWAYS, // Key is always accessible in this device\n invalidateOnNewBiometry: false,\n });\n\n // 2. Prepare Public Key as JWK\n // Toaster backend expects JWK in the header\n const publicJwk = spkiToJwk(publicKey);\n\n // 3. Construct Payload\n const now = Math.floor(Date.now() / 1000);\n const claims: DpopClaims = {\n iat: now,\n jti: uuid.v4(),\n };\n\n const header: DpopHeader = {\n typ: TYP,\n alg: ALG,\n jwk: publicJwk,\n };\n\n // 4. Prepare Signing Input\n const headerB64 = toBase64Url(JSON.stringify(header));\n const payloadB64 = toBase64Url(JSON.stringify(claims));\n const signingInput = `${headerB64}.${payloadB64}`;\n\n // 5. Sign Data\n // DeviceCrypto returns a Base64 signature.\n const signatureBase64 = await DeviceCrypto.sign(KEY_ALIAS, signingInput, {\n // Biometry prompts should not be fired since the key is always accessible in this device\n biometryTitle: 'Sign DPoP',\n biometrySubTitle: 'Sign DPoP',\n biometryDescription: 'Sign DPoP',\n });\n\n // 6. Convert Signature (Toaster expects Raw R|S)\n const signatureB64 = derToRawSignature(signatureBase64);\n\n return `${signingInput}.${signatureB64}`;\n } catch (error) {\n console.error('DPoP Generation Error:', error);\n return null;\n }\n};\n\n/**\n * Removes the keys from the Secure Enclave\n * Returns true if the key was deleted successfully, false otherwise.\n * @returns {Promise<boolean>} True if the key was deleted successfully, false otherwise.\n */\nexport const deleteDpop = async (): Promise<boolean> => {\n try {\n return await DeviceCrypto.deleteKey(KEY_ALIAS);\n } catch (error) {\n console.error('DPoP Deletion Error:', error);\n return false;\n }\n};\n", "/**\n * Encodes a Uint8Array to a Base64 string.\n * Uses native 'btoa' by converting bytes to a binary string first.\n */\nexport const uint8ArrayToBase64 = (bytes: Uint8Array): string => {\n let binary = '';\n const len = bytes.byteLength;\n for (let i = 0; i < len; i++) {\n binary += String.fromCharCode(bytes[i]);\n }\n return btoa(binary);\n};\n\n/**\n * Decodes a Base64 string to a Uint8Array.\n * Uses native 'atob'.\n */\nexport const base64ToUint8Array = (base64: string): Uint8Array => {\n const binaryString = atob(base64);\n const len = binaryString.length;\n const bytes = new Uint8Array(len);\n for (let i = 0; i < len; i++) {\n bytes[i] = binaryString.charCodeAt(i);\n }\n return bytes;\n};\n\n/**\n * Converts a string (UTF-8) to Uint8Array.\n * Polyfill for simple ASCII/UTF-8 if TextEncoder is not available,\n * but TextEncoder is standard in RN 0.68+.\n */\nexport const stringToUint8Array = (str: string): Uint8Array => {\n // Use TextEncoder if available (standard in modern RN)\n if (typeof TextEncoder !== 'undefined') {\n return new TextEncoder().encode(str);\n }\n // Fallback for older environments\n const arr = [];\n for (let i = 0; i < str.length; i++) {\n let code = str.charCodeAt(i);\n if (code < 0x80) {\n arr.push(code);\n } else if (code < 0x800) {\n arr.push(0xc0 | (code >> 6), 0x80 | (code & 0x3f));\n } else if (code < 0xd800 || code >= 0xe000) {\n arr.push(0xe0 | (code >> 12), 0x80 | ((code >> 6) & 0x3f), 0x80 | (code & 0x3f));\n } else {\n i++;\n code = 0x10000 + (((code & 0x3ff) << 10) | (str.charCodeAt(i) & 0x3ff));\n arr.push(0xf0 | (code >> 18), 0x80 | ((code >> 12) & 0x3f), 0x80 | ((code >> 6) & 0x3f), 0x80 | (code & 0x3f));\n }\n }\n return new Uint8Array(arr);\n};\n\n/**\n * Encodes input (String or Uint8Array) to Base64Url (RFC 4648).\n * Removes padding '=', replaces '+' with '-', and '/' with '_'\n */\nexport const toBase64Url = (input: string | Uint8Array): string => {\n let bytes: Uint8Array;\n\n if (typeof input === 'string') {\n bytes = stringToUint8Array(input);\n } else {\n bytes = input;\n }\n\n const base64 = uint8ArrayToBase64(bytes);\n return base64.replace(/\\+/g, '-').replace(/\\//g, '_').replace(/=+$/, '');\n};\n", "import { base64ToUint8Array, toBase64Url } from './uint8';\n\n/**\n * Converts SPKI Public Key (Base64) to JWK format.\n * Extracts Raw X and Y coordinates from the uncompressed point.\n */\nexport const spkiToJwk = (spkiBase64: string) => {\n const buf = base64ToUint8Array(spkiBase64);\n // P-256 SPKI Header is 26 bytes. The key data (0x04 + X + Y) is at the end.\n // We explicitly look for the last 65 bytes (1 header byte + 32 bytes X + 32 bytes Y)\n const rawKey = buf.subarray(buf.length - 65);\n\n if (rawKey[0] !== 0x04) {\n throw new Error('Invalid Public Key format: Expected uncompressed point');\n }\n\n const x = rawKey.subarray(1, 33);\n const y = rawKey.subarray(33, 65);\n\n return {\n kty: 'EC',\n crv: 'P-256',\n x: toBase64Url(x),\n y: toBase64Url(y),\n };\n};\n", "export const ALG = 'ES256';\nexport const TYP = 'dpop+jwt';\n", "import { base64ToUint8Array, toBase64Url } from './uint8';\n\n/**\n * Converts a DER encoded signature (ASN.1) to a Raw R|S signature (64 bytes).\n * Device Crypto returns DER; Toaster backend expects Raw P1363.\n */\nexport const derToRawSignature = (derBase64: string): string => {\n const der = base64ToUint8Array(derBase64);\n\n // DER Structure: 0x30 | total_len | 0x02 | r_len | r_bytes | 0x02 | s_len | s_bytes\n let offset = 2; // Skip Sequence Tag (0x30) and Total Length\n\n // --- Read R ---\n if (der[offset] !== 0x02) throw new Error('Invalid DER: R tag missing');\n offset++; // skip tag\n const rLen = der[offset++]; // read length\n let rBytes = der.subarray(offset, offset + rLen);\n offset += rLen;\n\n // Handle ASN.1 Integer padding for R (remove leading 0x00 if present)\n if (rLen === 33 && rBytes[0] === 0x00) {\n rBytes = rBytes.subarray(1);\n }\n\n // --- Read S ---\n if (der[offset] !== 0x02) throw new Error('Invalid DER: S tag missing');\n offset++; // skip tag\n const sLen = der[offset++]; // read length\n let sBytes = der.subarray(offset, offset + sLen);\n\n // Handle ASN.1 Integer padding for S\n if (sLen === 33 && sBytes[0] === 0x00) {\n sBytes = sBytes.subarray(1);\n }\n\n // --- Construct Raw Signature (64 bytes) ---\n const rawSignature = new Uint8Array(64);\n\n // Copy R into the first 32 bytes (right-aligned/padded)\n rawSignature.set(rBytes, 32 - rBytes.length);\n\n // Copy S into the last 32 bytes (right-aligned/padded)\n rawSignature.set(sBytes, 64 - sBytes.length);\n\n return toBase64Url(rawSignature);\n};\n", "import { NativeModules, Platform } from 'react-native';\n\ninterface NativeModuleCheck {\n name: string;\n nativeModuleName: string;\n packageName: string;\n}\n\nconst REQUIRED_NATIVE_MODULES: NativeModuleCheck[] = [\n {\n name: 'react-native-keychain',\n nativeModuleName: 'RNKeychainManager',\n packageName: 'react-native-keychain',\n },\n {\n name: 'react-native-device-crypto',\n nativeModuleName: 'DeviceCrypto',\n packageName: 'react-native-device-crypto',\n },\n];\n\n// Track if warning has been shown to avoid spamming\nlet hasWarned = false;\n\n/**\n * Checks if all required native modules are properly installed and linked.\n * Logs a warning if any native module is missing.\n *\n * Note: Some native modules (like react-native-device-crypto) require hardware\n * features (e.g., Secure Enclave) that are not available in simulators/emulators.\n * The SDK will continue to work but certain security features may be degraded.\n */\nexport const checkNativeModules = (): void => {\n if (hasWarned) return;\n\n const missingModules: NativeModuleCheck[] = [];\n\n for (const module of REQUIRED_NATIVE_MODULES) {\n if (!NativeModules[module.nativeModuleName]) {\n missingModules.push(module);\n }\n }\n\n if (missingModules.length > 0) {\n hasWarned = true;\n\n const platform = Platform.OS;\n const moduleList = missingModules.map(m => ` - ${m.packageName}`).join('\\n');\n const installCommands = missingModules.map(m => `npm install ${m.packageName}`).join('\\n');\n\n const iosInstructions =\n platform === 'ios'\n ? `\nFor iOS, run:\n cd ios && pod install && cd ..\n`\n : '';\n\n const androidInstructions =\n platform === 'android'\n ? `\nFor Android, rebuild your app:\n npx react-native run-android\n`\n : '';\n\n console.warn(\n `@magic-sdk/react-native-bare: Missing native modules detected.\n\nThe following native modules are not linked:\n${moduleList}\n\nThe SDK will continue to work, but some security features may not function properly.\n\nNote: If you're running in a simulator/emulator, some native modules (like react-native-device-crypto) \nrequire hardware features (Secure Enclave) that are only available on physical devices.\n\nIf you're on a physical device and see this warning, please ensure the packages are installed and linked:\n\n1. Install the missing packages:\n${installCommands}\n\n2. Link the native modules:\n${iosInstructions}${androidInstructions}\n3. Rebuild your app completely.\n`,\n );\n }\n};\n", "import { SDKBase } from '@magic-sdk/provider';\nimport { ReactNativeWebViewController } from './react-native-webview-controller';\n\nexport class SDKBaseReactNative extends SDKBase {\n public get Relayer() {\n return (this.overlay as ReactNativeWebViewController).Relayer;\n }\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["index_exports", "__export", "Magic", "useInternetConnection", "__toCommonJS", "import_runtime", "import_provider", "processPolyfill", "import_whatwg_url", "import_buffer", "import_lodash", "import_react_native_device_info", "import_react", "import_react_native", "import_react_native_webview", "import_react_native_safe_area_context", "import_provider", "import_lodash", "import_async_storage", "import_react_native_event_listeners", "import_react", "import_netinfo", "useInternetConnection", "isConnected", "setIsConnected", "handleConnectionChange", "connectionInfo", "NetInfo", "Keychain", "import_react_native_device_info", "KEY_SUFFIX_MAP", "getKeyAlias", "key", "DeviceInfo", "SERVICE", "getKeyAlias", "KEY", "cachedRefreshToken", "setRefreshTokenInKeychain", "rt", "result", "error", "getRefreshTokenInKeychain", "keychainEntry", "import_react_native_uuid", "uint8ArrayToBase64", "bytes", "binary", "len", "i", "base64ToUint8Array", "base64", "binaryString", "stringToUint8Array", "str", "arr", "code", "toBase64Url", "input", "spkiToJwk", "spkiBase64", "buf", "base64ToUint8Array", "rawKey", "x", "y", "toBase64Url", "ALG", "TYP", "derToRawSignature", "derBase64", "der", "base64ToUint8Array", "offset", "rLen", "rBytes", "sLen", "sBytes", "rawSignature", "toBase64Url", "import_react_native_device_crypto", "KEY_ALIAS", "getKeyAlias", "getDpop", "publicKey", "DeviceCrypto", "publicJwk", "spkiToJwk", "claims", "uuid", "headerB64", "toBase64Url", "TYP", "ALG", "payloadB64", "signingInput", "signatureBase64", "signatureB64", "derToRawSignature", "error", "import_react_native", "REQUIRED_NATIVE_MODULES", "hasWarned", "checkNativeModules", "missingModules", "module", "platform", "moduleList", "m", "installCommands", "MAGIC_PAYLOAD_FLAG_TYPED_ARRAY", "OPEN_IN_DEVICE_BROWSER", "DEFAULT_BACKGROUND_COLOR", "MSG_POSTED_AFTER_INACTIVITY_EVENT", "LAST_MESSAGE_TIME", "createWebViewStyles", "ReactNativeWebViewController", "backgroundColor", "show", "setShow", "mountOverlay", "setMountOverlay", "isConnected", "useInternetConnection", "checkNativeModules", "AsyncStorage", "message", "webViewRef", "webView", "containerRef", "view", "showOverlay", "hideOverlay", "containerStyles", "handleWebViewMessage", "event", "React", "url", "data", "key", "value", "e", "magicEvent", "handler", "lastPostTimestamp", "lastPostDate", "now", "fiveMinutes", "setRefreshTokenInKeychain", "getRefreshTokenInKeychain", "getDpop", "import_provider", "SDKBaseReactNative", "import_async_storage", "__reExport", "index_exports", "_", "processPolyfill", "URLPolyfill", "URLSearchParamsPolyfill", "str", "b64Encoded", "Magic", "SDKBaseReactNative", "ReactNativeWebViewController", "AsyncStorage"]
7
7
  }
package/dist/es/index.js CHANGED
@@ -1,2 +1,28 @@
1
- import"regenerator-runtime/runtime";import{createSDK as $}from"@magic-sdk/provider";import*as K from"process";import{URL as Y,URLSearchParams as q}from"whatwg-url";import{Buffer as u}from"buffer";import z from"lodash";import{getBundleId as J}from"react-native-device-info";import h,{useState as w,useCallback as i,useMemo as P,useEffect as m}from"react";import{Linking as x,StyleSheet as D}from"react-native";import{WebView as T}from"react-native-webview";import{SafeAreaView as L}from"react-native-safe-area-context";import{ViewController as N,createModalNotReadyError as k}from"@magic-sdk/provider";import{isTypedArray as F}from"lodash";import p from"@react-native-async-storage/async-storage";import{EventRegister as b}from"react-native-event-listeners";import{useEffect as M,useState as V}from"react";import O from"@react-native-community/netinfo";var y=()=>{let[a,e]=V(!0);return M(()=>{let n=t=>{e(!!t.isConnected)};return O.addEventListener(n)},[]),a};var v="MAGIC_PAYLOAD_FLAG_TYPED_ARRAY",U="open_in_device_browser",W="#FFFFFF",E="msg_posted_after_inactivity_event",f="lastMessageTime";function B(){return D.create({"magic-webview":{flex:1,backgroundColor:"transparent"},"webview-container":{flex:1,width:"100%",backgroundColor:"transparent",position:"absolute",top:0,left:0,right:0,bottom:0},show:{zIndex:1e4,elevation:1e4},hide:{zIndex:-1e4,elevation:0}})}var d=class extends N{webView;container;styles;init(){this.webView=null,this.container=null,this.styles=B()}Relayer=e=>{let[n,t]=w(!1),[s,r]=w(!0),c=y();m(()=>{this.isConnectedToInternet=c},[c]),m(()=>(p.setItem(f,""),()=>{this.isReadyForRequest=!1}),[]),m(()=>{b.addEventListener(E,async o=>{this.isReadyForRequest=!1,r(!1),this.post(o.msgType,o.payload),await p.setItem(f,new Date().toISOString())})},[]),m(()=>{s||setTimeout(()=>r(!0),10)},[s]);let I=i(o=>{this.webView=o},[]),S=i(o=>{this.container={...o,showOverlay:R,hideOverlay:_}},[]),R=i(()=>{t(!0)},[]),_=i(()=>{t(!1)},[]),C=P(()=>[this.styles["webview-container"],n?{...this.styles.show,backgroundColor:e??W}:this.styles.hide],[n]),A=i(o=>{this.handleReactNativeWebViewMessage(o)},[]);return s?h.createElement(L,{ref:S,style:C},h.createElement(T,{ref:I,source:{uri:`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`},onMessage:A,style:this.styles["magic-webview"],webviewDebuggingEnabled:!0,autoManageStatusBarEnabled:!1,onShouldStartLoadWithRequest:o=>new URLSearchParams(o.url.split("?")[1]).get(U)?(x.openURL(o.url),!1):!0})):null};handleReactNativeWebViewMessage(e){let n=new URL(`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`);if(e.nativeEvent&&typeof e.nativeEvent.data=="string"&&(e.nativeEvent.url===`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`||e.nativeEvent.url===`${this.endpoint}/send/?params=${this.parameters}`||e.nativeEvent.url===this.endpoint||e.nativeEvent.title===`${n.hostname}/send/${n.search}`)){let t=JSON.parse(e.nativeEvent.data,(s,r)=>{try{if(r&&typeof r=="object"&&r.flag&&r.flag===v)return new global[r.constructor](r.data.split(","))}catch(c){console.log("Error parsing data",c)}return r});if(t&&t.msgType&&this.messageHandlers.size){t.response=t.response??{};let s={data:t};for(let r of this.messageHandlers.values())r(s)}}}async msgPostedAfterInactivity(){let e=await p.getItem(f);if(e){let n=new Date(e).getTime(),t=new Date().getTime(),s=5*60*1e3;return t-n>s}return!1}hideOverlay(){this.container&&this.container.hideOverlay()}showOverlay(){this.container&&this.container.showOverlay()}async _post(e){if(await this.msgPostedAfterInactivity()){b.emit(E,e);return}if(this.webView&&this.webView.postMessage)this.webView.postMessage(JSON.stringify(e,(n,t)=>F(t)?{constructor:t.constructor.name,data:t.toString(),flag:v}:t)),p.setItem(f,new Date().toISOString());else throw k()}checkRelayerExistsInDOM(){return Promise.resolve(!0)}reloadRelayer(){return Promise.resolve(void 0)}};import{SDKBase as G}from"@magic-sdk/provider";var g=class extends G{get Relayer(){return this.overlay.Relayer}};import l from"@react-native-async-storage/async-storage";export*from"@magic-sdk/provider";export*from"@magic-sdk/types";global.process=z.merge(global.process,K);global.process.browser=!1;global.Buffer=u;global.URL=Y;global.URLSearchParams=q;global.btoa=a=>u.from(a,"binary").toString("base64");global.atob=a=>u.from(a,"base64").toString("binary");var _e=$(g,{platform:"react-native",sdkName:"@magic-sdk/react-native-bare",version:"34.0.0",bundleId:J(),defaultEndpoint:"https://box.magic.link/",ViewController:d,configureStorage:async()=>({ready:async()=>Promise.resolve(!0),getItem:l.getItem,setItem:l.setItem,removeItem:l.removeItem,clear:l.clear,length:()=>{},key:()=>{},keys:l.getAllKeys,iterate:()=>{}})});export{_e as Magic,y as useInternetConnection};
1
+ import"regenerator-runtime/runtime";import{createSDK as ge}from"@magic-sdk/provider";import*as he from"process";import{URL as we,URLSearchParams as ve}from"whatwg-url";import{Buffer as x}from"buffer";import be from"lodash";import{getBundleId as Ee}from"react-native-device-info";import U,{useState as K,useCallback as f,useMemo as oe,useEffect as u}from"react";import{Linking as ae,StyleSheet as se}from"react-native";import{WebView as ie}from"react-native-webview";import{SafeAreaView as ce}from"react-native-safe-area-context";import{ViewController as le,createModalNotReadyError as pe}from"@magic-sdk/provider";import{isTypedArray as me}from"lodash";import h from"@react-native-async-storage/async-storage";import{EventRegister as V}from"react-native-event-listeners";import{useEffect as Y,useState as J}from"react";import j from"@react-native-community/netinfo";var S=()=>{let[r,e]=J(!0);return Y(()=>{let n=t=>{e(!!t.isConnected)};return j.addEventListener(n)},[]),r};import*as p from"react-native-keychain";import q from"react-native-device-info";var H={dpop:"magic.sdk.dpop",refreshToken:"magic.sdk.rt",refreshTokenService:"magic.sdk.rt.service"};function d(r){return`${q.getBundleId()}.${H[r]}`}var k=d("refreshTokenService"),z=d("refreshToken"),m=null,R=async r=>{if(m===r)return!0;try{let e=await p.setGenericPassword(z,r,{service:k,accessible:p.ACCESSIBLE.AFTER_FIRST_UNLOCK_THIS_DEVICE_ONLY});return m=r,e}catch(e){return console.error("Failed to set refresh token in keychain",e),!1}},A=async()=>{if(m!==null)return m;try{let r=await p.getGenericPassword({service:k});return r?(m=r.password,m):null}catch(r){return console.error("Failed to get refresh token in keychain",r),null}};import Z from"react-native-uuid";var X=r=>{let e="",n=r.byteLength;for(let t=0;t<n;t++)e+=String.fromCharCode(r[t]);return btoa(e)},g=r=>{let e=atob(r),n=e.length,t=new Uint8Array(n);for(let o=0;o<n;o++)t[o]=e.charCodeAt(o);return t},Q=r=>{if(typeof TextEncoder<"u")return new TextEncoder().encode(r);let e=[];for(let n=0;n<r.length;n++){let t=r.charCodeAt(n);t<128?e.push(t):t<2048?e.push(192|t>>6,128|t&63):t<55296||t>=57344?e.push(224|t>>12,128|t>>6&63,128|t&63):(n++,t=65536+((t&1023)<<10|r.charCodeAt(n)&1023),e.push(240|t>>18,128|t>>12&63,128|t>>6&63,128|t&63))}return new Uint8Array(e)},l=r=>{let e;return typeof r=="string"?e=Q(r):e=r,X(e).replace(/\+/g,"-").replace(/\//g,"_").replace(/=+$/,"")};var T=r=>{let e=g(r),n=e.subarray(e.length-65);if(n[0]!==4)throw new Error("Invalid Public Key format: Expected uncompressed point");let t=n.subarray(1,33),o=n.subarray(33,65);return{kty:"EC",crv:"P-256",x:l(t),y:l(o)}};var C="ES256",M="dpop+jwt";var P=r=>{let e=g(r),n=2;if(e[n]!==2)throw new Error("Invalid DER: R tag missing");n++;let t=e[n++],o=e.subarray(n,n+t);if(n+=t,t===33&&o[0]===0&&(o=o.subarray(1)),e[n]!==2)throw new Error("Invalid DER: S tag missing");n++;let a=e[n++],s=e.subarray(n,n+a);a===33&&s[0]===0&&(s=s.subarray(1));let c=new Uint8Array(64);return c.set(o,32-o.length),c.set(s,64-s.length),l(c)};import D,{AccessLevel as ee}from"react-native-device-crypto";var _=d("dpop"),N=async()=>{try{let r=await D.getOrCreateAsymmetricKey(_,{accessLevel:ee.ALWAYS,invalidateOnNewBiometry:!1}),e=T(r),t={iat:Math.floor(Date.now()/1e3),jti:Z.v4()},a=l(JSON.stringify({typ:M,alg:C,jwk:e})),s=l(JSON.stringify(t)),c=`${a}.${s}`,E=await D.sign(_,c,{biometryTitle:"Sign DPoP",biometrySubTitle:"Sign DPoP",biometryDescription:"Sign DPoP"}),I=P(E);return`${c}.${I}`}catch(r){return console.error("DPoP Generation Error:",r),null}};import{NativeModules as te,Platform as re}from"react-native";var ne=[{name:"react-native-keychain",nativeModuleName:"RNKeychainManager",packageName:"react-native-keychain"},{name:"react-native-device-crypto",nativeModuleName:"DeviceCrypto",packageName:"react-native-device-crypto"}],O=!1,L=()=>{if(O)return;let r=[];for(let e of ne)te[e.nativeModuleName]||r.push(e);if(r.length>0){O=!0;let e=re.OS,n=r.map(s=>` - ${s.packageName}`).join(`
2
+ `),t=r.map(s=>`npm install ${s.packageName}`).join(`
3
+ `);console.warn(`@magic-sdk/react-native-bare: Missing native modules detected.
4
+
5
+ The following native modules are not linked:
6
+ ${n}
7
+
8
+ The SDK will continue to work, but some security features may not function properly.
9
+
10
+ Note: If you're running in a simulator/emulator, some native modules (like react-native-device-crypto)
11
+ require hardware features (Secure Enclave) that are only available on physical devices.
12
+
13
+ If you're on a physical device and see this warning, please ensure the packages are installed and linked:
14
+
15
+ 1. Install the missing packages:
16
+ ${t}
17
+
18
+ 2. Link the native modules:
19
+ ${e==="ios"?`
20
+ For iOS, run:
21
+ cd ios && pod install && cd ..
22
+ `:""}${e==="android"?`
23
+ For Android, rebuild your app:
24
+ npx react-native run-android
25
+ `:""}
26
+ 3. Rebuild your app completely.
27
+ `)}};var F="MAGIC_PAYLOAD_FLAG_TYPED_ARRAY",de="open_in_device_browser",fe="#FFFFFF",B="msg_posted_after_inactivity_event",w="lastMessageTime";function ue(){return se.create({"magic-webview":{flex:1,backgroundColor:"transparent"},"webview-container":{flex:1,width:"100%",backgroundColor:"transparent",position:"absolute",top:0,left:0,right:0,bottom:0},show:{zIndex:1e4,elevation:1e4},hide:{zIndex:-1e4,elevation:0}})}var v=class extends le{webView;container;styles;init(){this.webView=null,this.container=null,this.styles=ue()}Relayer=e=>{let[n,t]=K(!1),[o,a]=K(!0),s=S();u(()=>{L()},[]),u(()=>{this.isConnectedToInternet=s},[s]),u(()=>(h.setItem(w,""),()=>{this.isReadyForRequest=!1}),[]),u(()=>{V.addEventListener(B,async i=>{this.isReadyForRequest=!1,a(!1),this.post(i.msgType,i.payload),await h.setItem(w,new Date().toISOString())})},[]),u(()=>{o||setTimeout(()=>a(!0),10)},[o]);let c=f(i=>{this.webView=i},[]),E=f(i=>{this.container={...i,showOverlay:I,hideOverlay:$}},[]),I=f(()=>{t(!0)},[]),$=f(()=>{t(!1)},[]),W=oe(()=>[this.styles["webview-container"],n?{...this.styles.show,backgroundColor:e??fe}:this.styles.hide],[n]),G=f(i=>{this.handleReactNativeWebViewMessage(i)},[]);return o?U.createElement(ce,{ref:E,style:W},U.createElement(ie,{ref:c,source:{uri:`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`},onMessage:G,style:this.styles["magic-webview"],webviewDebuggingEnabled:!0,autoManageStatusBarEnabled:!1,onShouldStartLoadWithRequest:i=>new URLSearchParams(i.url.split("?")[1]).get(de)?(ae.openURL(i.url),!1):!0})):null};handleReactNativeWebViewMessage(e){let n=new URL(`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`);if(e.nativeEvent&&typeof e.nativeEvent.data=="string"&&(e.nativeEvent.url===`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`||e.nativeEvent.url===`${this.endpoint}/send/?params=${this.parameters}`||e.nativeEvent.url===this.endpoint||e.nativeEvent.title===`${n.hostname}/send/${n.search}`)){let t=JSON.parse(e.nativeEvent.data,(o,a)=>{try{if(a&&typeof a=="object"&&a.flag&&a.flag===F)return new global[a.constructor](a.data.split(","))}catch(s){console.log("Error parsing data",s)}return a});if(t&&t.msgType&&this.messageHandlers.size){t.response=t.response??{};let o={data:t};for(let a of this.messageHandlers.values())a(o)}}}async msgPostedAfterInactivity(){let e=await h.getItem(w);if(e){let n=new Date(e).getTime(),t=new Date().getTime(),o=5*60*1e3;return t-n>o}return!1}hideOverlay(){this.container&&this.container.hideOverlay()}showOverlay(){this.container&&this.container.showOverlay()}async _post(e){if(await this.msgPostedAfterInactivity()){V.emit(B,e);return}if(this.webView&&this.webView.postMessage)this.webView.postMessage(JSON.stringify(e,(n,t)=>me(t)?{constructor:t.constructor.name,data:t.toString(),flag:F}:t)),h.setItem(w,new Date().toISOString());else throw pe()}async persistMagicEventRefreshToken(e){e?.data?.rt&&await R(e.data.rt)}async getRT(){return await A()}async getJWT(){try{return await N()}catch{return null}}checkRelayerExistsInDOM(){return Promise.resolve(!0)}reloadRelayer(){return Promise.resolve(void 0)}};import{SDKBase as ye}from"@magic-sdk/provider";var b=class extends ye{get Relayer(){return this.overlay.Relayer}};import y from"@react-native-async-storage/async-storage";export*from"@magic-sdk/provider";export*from"@magic-sdk/types";global.process=be.merge(global.process,he);global.process.browser=!1;global.Buffer=x;global.URL=we;global.URLSearchParams=ve;global.btoa=r=>x.from(r,"binary").toString("base64");global.atob=r=>x.from(r,"base64").toString("binary");var Et=ge(b,{platform:"react-native",sdkName:"@magic-sdk/react-native-bare",version:"34.0.0",bundleId:Ee(),defaultEndpoint:"https://box.magic.link/",ViewController:v,configureStorage:async()=>({ready:async()=>Promise.resolve(!0),getItem:y.getItem,setItem:y.setItem,removeItem:y.removeItem,clear:y.clear,length:()=>{},key:()=>{},keys:y.getAllKeys,iterate:()=>{}})});export{Et as Magic,S as useInternetConnection};
2
28
  //# sourceMappingURL=index.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
- "sources": ["../../src/index.ts", "../../src/react-native-webview-controller.tsx", "../../src/hooks.ts", "../../src/react-native-sdk-base.ts"],
4
- "sourcesContent": ["/* istanbul ignore file */\n\nimport 'regenerator-runtime/runtime';\n\nimport { createSDK, InstanceWithExtensions, MagicSDKExtensionsOption } from '@magic-sdk/provider';\nimport * as processPolyfill from 'process';\nimport { URL as URLPolyfill, URLSearchParams as URLSearchParamsPolyfill } from 'whatwg-url';\nimport { Buffer } from 'buffer';\nimport _ from 'lodash';\nimport { getBundleId } from 'react-native-device-info';\nimport { ReactNativeWebViewController } from './react-native-webview-controller';\nimport { SDKBaseReactNative } from './react-native-sdk-base';\nimport type localForage from 'localforage';\nimport AsyncStorage from '@react-native-async-storage/async-storage';\n\n// Web3 assumes a browser context, so we need\n// to provide `btoa` and `atob` shims.\n\n// We expect `global.process` to be a Node Process for web3.js usage\n// so we replace it here.\nglobal.process = _.merge(global.process, processPolyfill);\n\n(global.process as any).browser = false;\n\n// WHATWG URL requires global `Buffer` access.\nglobal.Buffer = Buffer;\n\n// Setup global WHATWG URL polyfills\nglobal.URL = URLPolyfill as any;\nglobal.URLSearchParams = URLSearchParamsPolyfill as any;\n\n/* istanbul ignore next */\nglobal.btoa = str => Buffer.from(str, 'binary').toString('base64');\n/* istanbul ignore next */\nglobal.atob = b64Encoded => Buffer.from(b64Encoded, 'base64').toString('binary');\n\nexport * from '@magic-sdk/provider';\nexport * from '@magic-sdk/types';\n\nexport const Magic = createSDK(SDKBaseReactNative, {\n platform: 'react-native',\n sdkName: '@magic-sdk/react-native-bare',\n version: process.env.BARE_REACT_NATIVE_VERSION!,\n bundleId: getBundleId(),\n defaultEndpoint: 'https://box.magic.link/',\n ViewController: ReactNativeWebViewController,\n configureStorage: /* istanbul ignore next */ async () => {\n return {\n ready: async () => Promise.resolve(true),\n getItem: AsyncStorage.getItem,\n setItem: AsyncStorage.setItem,\n removeItem: AsyncStorage.removeItem,\n clear: AsyncStorage.clear,\n length: () => {},\n key: () => {},\n keys: AsyncStorage.getAllKeys,\n iterate: () => {},\n } as unknown as typeof localForage;\n },\n});\n\nexport type Magic<T extends MagicSDKExtensionsOption<any> = MagicSDKExtensionsOption> = InstanceWithExtensions<\n SDKBaseReactNative,\n T\n>;\n\nexport { useInternetConnection } from './hooks';\n", "import React, { useState, useCallback, useMemo, useEffect } from 'react';\nimport { Linking, StyleSheet, View } from 'react-native';\nimport { WebView } from 'react-native-webview';\nimport { SafeAreaView } from 'react-native-safe-area-context';\nimport { ViewController, createModalNotReadyError } from '@magic-sdk/provider';\nimport { MagicMessageEvent } from '@magic-sdk/types';\nimport { isTypedArray } from 'lodash';\nimport AsyncStorage from '@react-native-async-storage/async-storage';\nimport { EventRegister } from 'react-native-event-listeners';\n/* global NodeJS */\nimport Global = NodeJS.Global;\nimport { useInternetConnection } from './hooks';\n\nconst MAGIC_PAYLOAD_FLAG_TYPED_ARRAY = 'MAGIC_PAYLOAD_FLAG_TYPED_ARRAY';\nconst OPEN_IN_DEVICE_BROWSER = 'open_in_device_browser';\nconst DEFAULT_BACKGROUND_COLOR = '#FFFFFF';\nconst MSG_POSTED_AFTER_INACTIVITY_EVENT = 'msg_posted_after_inactivity_event';\nconst LAST_MESSAGE_TIME = 'lastMessageTime';\n/**\n * Builds the Magic `<WebView>` overlay styles. These base styles enable\n * `<WebView>` UI to render above all other DOM content.\n */\nfunction createWebViewStyles() {\n return StyleSheet.create({\n 'magic-webview': {\n flex: 1,\n backgroundColor: 'transparent',\n },\n\n 'webview-container': {\n flex: 1,\n width: '100%',\n backgroundColor: 'transparent',\n position: 'absolute',\n top: 0,\n left: 0,\n right: 0,\n bottom: 0,\n },\n\n show: {\n zIndex: 10000,\n elevation: 10000,\n },\n\n hide: {\n zIndex: -10000,\n elevation: 0,\n },\n });\n}\n\n/**\n * Overloads React Native's `<View>` with helper methods we require to hide/show\n * the rendered `<WebView>`.\n */\ninterface ViewWrapper extends View {\n showOverlay: () => void;\n hideOverlay: () => void;\n}\n\n/**\n * View controller for the Magic `<WebView>` overlay.\n */\nexport class ReactNativeWebViewController extends ViewController {\n private webView!: WebView | null;\n private container!: ViewWrapper | null;\n private styles: any;\n\n protected init() {\n this.webView = null;\n this.container = null;\n this.styles = createWebViewStyles();\n }\n\n /**\n * Renders a React Native `<WebView>` with built-in message handling to and\n * from the Magic `<iframe>` context.\n */\n // Validating this logic requires lots of React-specific boilerplate. We will\n // revisit this method for unit testing in the future. For now, manual testing\n // is sufficient (this logic is stable right now and not expected to change in\n // the foreseeable future).\n /* istanbul ignore next */\n public Relayer: React.FC<{ backgroundColor?: string }> = backgroundColor => {\n const [show, setShow] = useState(false);\n const [mountOverlay, setMountOverlay] = useState(true);\n const isConnected = useInternetConnection();\n\n useEffect(() => {\n this.isConnectedToInternet = isConnected;\n }, [isConnected]);\n\n useEffect(() => {\n // reset lastMessage when webview is first mounted\n AsyncStorage.setItem(LAST_MESSAGE_TIME, '');\n return () => {\n this.isReadyForRequest = false;\n };\n }, []);\n\n useEffect(() => {\n EventRegister.addEventListener(MSG_POSTED_AFTER_INACTIVITY_EVENT, async message => {\n // If inactivity has been determined, the message is posted only after a brief\n // unmount and re-mount of the webview. This is to ensure the webview is accepting messages.\n // iOS kills webview processes after a certain period of inactivity, like when the app is\n // on background for long periods of time.\n this.isReadyForRequest = false;\n setMountOverlay(false);\n this.post(message.msgType, message.payload);\n await AsyncStorage.setItem(LAST_MESSAGE_TIME, new Date().toISOString());\n });\n }, []);\n\n useEffect(() => {\n if (!mountOverlay) {\n // Briefly unmount and re-mount the webview to ensure it's ready to accept messages.\n setTimeout(() => setMountOverlay(true), 10);\n }\n }, [mountOverlay]);\n\n /**\n * Saves a reference to the underlying `<WebView>` node so we can interact\n * with incoming messages.\n */\n const webViewRef = useCallback((webView: any): void => {\n this.webView = webView;\n }, []);\n\n /**\n * Saves a reference to the underlying `<View>` node so we can interact with\n * display styles.\n */\n const containerRef = useCallback((view: any): void => {\n this.container = {\n ...view,\n showOverlay,\n hideOverlay,\n };\n }, []);\n\n /**\n * Show the Magic `<WebView>` overlay.\n */\n const showOverlay = useCallback(() => {\n setShow(true);\n }, []);\n\n /**\n * Hide the Magic `<WebView>` overlay.\n */\n const hideOverlay = useCallback(() => {\n setShow(false);\n }, []);\n\n const containerStyles = useMemo(() => {\n return [\n this.styles['webview-container'],\n show\n ? {\n ...this.styles.show,\n backgroundColor: backgroundColor ?? DEFAULT_BACKGROUND_COLOR,\n }\n : this.styles.hide,\n ];\n }, [show]);\n\n const handleWebViewMessage = useCallback((event: unknown) => {\n this.handleReactNativeWebViewMessage(event);\n }, []);\n\n if (!mountOverlay) {\n return null;\n }\n\n return (\n <SafeAreaView ref={containerRef} style={containerStyles}>\n <WebView\n ref={webViewRef}\n source={{ uri: `${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}` }}\n onMessage={handleWebViewMessage}\n style={this.styles['magic-webview']}\n webviewDebuggingEnabled\n autoManageStatusBarEnabled={false}\n onShouldStartLoadWithRequest={event => {\n const queryParams = new URLSearchParams(event.url.split('?')[1]);\n const openInDeviceBrowser = queryParams.get(OPEN_IN_DEVICE_BROWSER);\n\n if (openInDeviceBrowser) {\n Linking.openURL(event.url);\n return false;\n }\n return true;\n }}\n />\n </SafeAreaView>\n );\n };\n\n /**\n * Route incoming messages from a React Native `<WebView>`.\n */\n private handleReactNativeWebViewMessage(event: any) {\n const url = new URL(`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`);\n\n if (\n event.nativeEvent &&\n typeof event.nativeEvent.data === 'string' &&\n /* Backward compatible */\n (event.nativeEvent.url === `${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}` ||\n event.nativeEvent.url === `${this.endpoint}/send/?params=${this.parameters}` ||\n event.nativeEvent.url === this.endpoint ||\n event.nativeEvent.title === `${url.hostname}/send/${url.search}`)\n ) {\n // Special parsing logic when dealing with TypedArray in the payload\n // Such change is required as JSON.stringify will manipulate the object and cause exceptions during parsing\n // The typed Array is stringified in Mgbox with a flag as notation.\n const data: any = JSON.parse(event.nativeEvent.data, (key, value) => {\n try {\n if (value && typeof value === 'object' && value.flag && value.flag === MAGIC_PAYLOAD_FLAG_TYPED_ARRAY) {\n return new (global[value.constructor as keyof Global] as any)(value.data.split(','));\n }\n\n // silently handles exception and return the original copy\n } catch (e) {\n console.log('Error parsing data', e);\n }\n return value;\n });\n if (data && data.msgType && this.messageHandlers.size) {\n // If the response object is undefined, we ensure it's at least an\n // empty object before passing to the event listener.\n\n data.response = data.response ?? {};\n\n // Reconstruct event from RN event\n const magicEvent: MagicMessageEvent = { data } as MagicMessageEvent;\n for (const handler of this.messageHandlers.values()) {\n handler(magicEvent);\n }\n }\n }\n }\n\n private async msgPostedAfterInactivity() {\n const lastPostTimestamp: string | null = await AsyncStorage.getItem(LAST_MESSAGE_TIME);\n if (lastPostTimestamp) {\n const lastPostDate = new Date(lastPostTimestamp).getTime();\n const now = new Date().getTime();\n const fiveMinutes = 5 * 60 * 1000;\n\n // there's been inactivity if the new message is posted\n // 5 min or more after the last message.\n return now - lastPostDate > fiveMinutes;\n }\n return false;\n }\n\n protected hideOverlay() {\n if (this.container) this.container.hideOverlay();\n }\n\n protected showOverlay() {\n if (this.container) this.container.showOverlay();\n }\n\n protected async _post(data: any) {\n if (await this.msgPostedAfterInactivity()) {\n EventRegister.emit(MSG_POSTED_AFTER_INACTIVITY_EVENT, data);\n return;\n }\n if (this.webView && this.webView.postMessage) {\n this.webView.postMessage(\n JSON.stringify(data, (key, value) => {\n // parse Typed Array to Stringify object\n if (isTypedArray(value)) {\n return {\n constructor: value.constructor.name,\n data: value.toString(),\n flag: MAGIC_PAYLOAD_FLAG_TYPED_ARRAY,\n };\n }\n return value;\n }),\n );\n AsyncStorage.setItem(LAST_MESSAGE_TIME, new Date().toISOString());\n } else {\n throw createModalNotReadyError();\n }\n }\n\n // Todo - implement these methods\n /* istanbul ignore next */\n protected checkRelayerExistsInDOM(): Promise<boolean> {\n return Promise.resolve(true);\n }\n\n /* istanbul ignore next */\n protected reloadRelayer(): Promise<void> {\n return Promise.resolve(undefined);\n }\n}\n", "import { useEffect, useState } from 'react';\nimport NetInfo, { NetInfoState } from '@react-native-community/netinfo';\n\nexport const useInternetConnection = () => {\n const [isConnected, setIsConnected] = useState(true);\n useEffect(() => {\n const handleConnectionChange = (connectionInfo: NetInfoState) => {\n setIsConnected(!!connectionInfo.isConnected);\n };\n\n // Subscribe to connection changes and cleanup on unmount\n return NetInfo.addEventListener(handleConnectionChange);\n }, []);\n\n return isConnected;\n};\n", "import { SDKBase } from '@magic-sdk/provider';\nimport { ReactNativeWebViewController } from './react-native-webview-controller';\n\nexport class SDKBaseReactNative extends SDKBase {\n public get Relayer() {\n return (this.overlay as ReactNativeWebViewController).Relayer;\n }\n}\n"],
5
- "mappings": "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",
6
- "names": ["createSDK", "processPolyfill", "URLPolyfill", "URLSearchParamsPolyfill", "Buffer", "_", "getBundleId", "React", "useState", "useCallback", "useMemo", "useEffect", "Linking", "StyleSheet", "WebView", "SafeAreaView", "ViewController", "createModalNotReadyError", "isTypedArray", "AsyncStorage", "EventRegister", "useEffect", "useState", "NetInfo", "useInternetConnection", "isConnected", "setIsConnected", "handleConnectionChange", "connectionInfo", "MAGIC_PAYLOAD_FLAG_TYPED_ARRAY", "OPEN_IN_DEVICE_BROWSER", "DEFAULT_BACKGROUND_COLOR", "MSG_POSTED_AFTER_INACTIVITY_EVENT", "LAST_MESSAGE_TIME", "createWebViewStyles", "StyleSheet", "ReactNativeWebViewController", "ViewController", "backgroundColor", "show", "setShow", "useState", "mountOverlay", "setMountOverlay", "isConnected", "useInternetConnection", "useEffect", "AsyncStorage", "EventRegister", "message", "webViewRef", "useCallback", "webView", "containerRef", "view", "showOverlay", "hideOverlay", "containerStyles", "useMemo", "handleWebViewMessage", "event", "React", "SafeAreaView", "WebView", "Linking", "url", "data", "key", "value", "e", "magicEvent", "handler", "lastPostTimestamp", "lastPostDate", "now", "fiveMinutes", "isTypedArray", "createModalNotReadyError", "SDKBase", "SDKBaseReactNative", "AsyncStorage", "_", "processPolyfill", "Buffer", "URLPolyfill", "URLSearchParamsPolyfill", "str", "b64Encoded", "Magic", "createSDK", "SDKBaseReactNative", "getBundleId", "ReactNativeWebViewController"]
3
+ "sources": ["../../src/index.ts", "../../src/react-native-webview-controller.tsx", "../../src/hooks.ts", "../../src/native-crypto/keychain.ts", "../../src/native-crypto/utils/key-alias.ts", "../../src/native-crypto/dpop.ts", "../../src/native-crypto/utils/uint8.ts", "../../src/native-crypto/utils/jwk.ts", "../../src/native-crypto/constants/index.ts", "../../src/native-crypto/utils/der.ts", "../../src/native-crypto/check-native-modules.ts", "../../src/react-native-sdk-base.ts"],
4
+ "sourcesContent": ["/* istanbul ignore file */\n\nimport 'regenerator-runtime/runtime';\n\nimport { createSDK, InstanceWithExtensions, MagicSDKExtensionsOption } from '@magic-sdk/provider';\nimport * as processPolyfill from 'process';\nimport { URL as URLPolyfill, URLSearchParams as URLSearchParamsPolyfill } from 'whatwg-url';\nimport { Buffer } from 'buffer';\nimport _ from 'lodash';\nimport { getBundleId } from 'react-native-device-info';\nimport { ReactNativeWebViewController } from './react-native-webview-controller';\nimport { SDKBaseReactNative } from './react-native-sdk-base';\nimport type localForage from 'localforage';\nimport AsyncStorage from '@react-native-async-storage/async-storage';\n\n// Web3 assumes a browser context, so we need\n// to provide `btoa` and `atob` shims.\n\n// We expect `global.process` to be a Node Process for web3.js usage\n// so we replace it here.\nglobal.process = _.merge(global.process, processPolyfill);\n\n(global.process as any).browser = false;\n\n// WHATWG URL requires global `Buffer` access.\nglobal.Buffer = Buffer;\n\n// Setup global WHATWG URL polyfills\nglobal.URL = URLPolyfill as any;\nglobal.URLSearchParams = URLSearchParamsPolyfill as any;\n\n/* istanbul ignore next */\nglobal.btoa = str => Buffer.from(str, 'binary').toString('base64');\n/* istanbul ignore next */\nglobal.atob = b64Encoded => Buffer.from(b64Encoded, 'base64').toString('binary');\n\nexport * from '@magic-sdk/provider';\nexport * from '@magic-sdk/types';\n\nexport const Magic = createSDK(SDKBaseReactNative, {\n platform: 'react-native',\n sdkName: '@magic-sdk/react-native-bare',\n version: process.env.BARE_REACT_NATIVE_VERSION!,\n bundleId: getBundleId(),\n defaultEndpoint: 'https://box.magic.link/',\n ViewController: ReactNativeWebViewController,\n configureStorage: /* istanbul ignore next */ async () => {\n return {\n ready: async () => Promise.resolve(true),\n getItem: AsyncStorage.getItem,\n setItem: AsyncStorage.setItem,\n removeItem: AsyncStorage.removeItem,\n clear: AsyncStorage.clear,\n length: () => {},\n key: () => {},\n keys: AsyncStorage.getAllKeys,\n iterate: () => {},\n } as unknown as typeof localForage;\n },\n});\n\nexport type Magic<T extends MagicSDKExtensionsOption<any> = MagicSDKExtensionsOption> = InstanceWithExtensions<\n SDKBaseReactNative,\n T\n>;\n\nexport { useInternetConnection } from './hooks';\n", "import React, { useState, useCallback, useMemo, useEffect } from 'react';\nimport { Linking, StyleSheet, View } from 'react-native';\nimport { WebView } from 'react-native-webview';\nimport { SafeAreaView } from 'react-native-safe-area-context';\nimport { ViewController, createModalNotReadyError } from '@magic-sdk/provider';\nimport { MagicMessageEvent } from '@magic-sdk/types';\nimport { isTypedArray } from 'lodash';\nimport AsyncStorage from '@react-native-async-storage/async-storage';\nimport { EventRegister } from 'react-native-event-listeners';\n/* global NodeJS */\nimport Global = NodeJS.Global;\nimport { useInternetConnection } from './hooks';\nimport { getRefreshTokenInKeychain, setRefreshTokenInKeychain } from './native-crypto/keychain';\nimport { getDpop } from './native-crypto/dpop';\nimport { checkNativeModules } from './native-crypto/check-native-modules';\n\nconst MAGIC_PAYLOAD_FLAG_TYPED_ARRAY = 'MAGIC_PAYLOAD_FLAG_TYPED_ARRAY';\nconst OPEN_IN_DEVICE_BROWSER = 'open_in_device_browser';\nconst DEFAULT_BACKGROUND_COLOR = '#FFFFFF';\nconst MSG_POSTED_AFTER_INACTIVITY_EVENT = 'msg_posted_after_inactivity_event';\nconst LAST_MESSAGE_TIME = 'lastMessageTime';\n/**\n * Builds the Magic `<WebView>` overlay styles. These base styles enable\n * `<WebView>` UI to render above all other DOM content.\n */\nfunction createWebViewStyles() {\n return StyleSheet.create({\n 'magic-webview': { flex: 1, backgroundColor: 'transparent' },\n\n 'webview-container': {\n flex: 1,\n width: '100%',\n backgroundColor: 'transparent',\n position: 'absolute',\n top: 0,\n left: 0,\n right: 0,\n bottom: 0,\n },\n\n show: { zIndex: 10000, elevation: 10000 },\n\n hide: { zIndex: -10000, elevation: 0 },\n });\n}\n\n/**\n * Overloads React Native's `<View>` with helper methods we require to hide/show\n * the rendered `<WebView>`.\n */\ninterface ViewWrapper extends View {\n showOverlay: () => void;\n hideOverlay: () => void;\n}\n\n/**\n * View controller for the Magic `<WebView>` overlay.\n */\nexport class ReactNativeWebViewController extends ViewController {\n private webView!: WebView | null;\n private container!: ViewWrapper | null;\n private styles: any;\n\n protected init() {\n this.webView = null;\n this.container = null;\n this.styles = createWebViewStyles();\n }\n\n /**\n * Renders a React Native `<WebView>` with built-in message handling to and\n * from the Magic `<iframe>` context.\n */\n // Validating this logic requires lots of React-specific boilerplate. We will\n // revisit this method for unit testing in the future. For now, manual testing\n // is sufficient (this logic is stable right now and not expected to change in\n // the foreseeable future).\n /* istanbul ignore next */\n public Relayer: React.FC<{ backgroundColor?: string }> = backgroundColor => {\n const [show, setShow] = useState(false);\n const [mountOverlay, setMountOverlay] = useState(true);\n const isConnected = useInternetConnection();\n\n useEffect(() => {\n checkNativeModules();\n }, []);\n\n useEffect(() => {\n this.isConnectedToInternet = isConnected;\n }, [isConnected]);\n\n useEffect(() => {\n // reset lastMessage when webview is first mounted\n AsyncStorage.setItem(LAST_MESSAGE_TIME, '');\n return () => {\n this.isReadyForRequest = false;\n };\n }, []);\n\n useEffect(() => {\n EventRegister.addEventListener(MSG_POSTED_AFTER_INACTIVITY_EVENT, async message => {\n // If inactivity has been determined, the message is posted only after a brief\n // unmount and re-mount of the webview. This is to ensure the webview is accepting messages.\n // iOS kills webview processes after a certain period of inactivity, like when the app is\n // on background for long periods of time.\n this.isReadyForRequest = false;\n setMountOverlay(false);\n this.post(message.msgType, message.payload);\n await AsyncStorage.setItem(LAST_MESSAGE_TIME, new Date().toISOString());\n });\n }, []);\n\n useEffect(() => {\n if (!mountOverlay) {\n // Briefly unmount and re-mount the webview to ensure it's ready to accept messages.\n setTimeout(() => setMountOverlay(true), 10);\n }\n }, [mountOverlay]);\n\n /**\n * Saves a reference to the underlying `<WebView>` node so we can interact\n * with incoming messages.\n */\n const webViewRef = useCallback((webView: any): void => {\n this.webView = webView;\n }, []);\n\n /**\n * Saves a reference to the underlying `<View>` node so we can interact with\n * display styles.\n */\n const containerRef = useCallback((view: any): void => {\n this.container = { ...view, showOverlay, hideOverlay };\n }, []);\n\n /**\n * Show the Magic `<WebView>` overlay.\n */\n const showOverlay = useCallback(() => {\n setShow(true);\n }, []);\n\n /**\n * Hide the Magic `<WebView>` overlay.\n */\n const hideOverlay = useCallback(() => {\n setShow(false);\n }, []);\n\n const containerStyles = useMemo(() => {\n return [\n this.styles['webview-container'],\n show ? { ...this.styles.show, backgroundColor: backgroundColor ?? DEFAULT_BACKGROUND_COLOR } : this.styles.hide,\n ];\n }, [show]);\n\n const handleWebViewMessage = useCallback((event: unknown) => {\n this.handleReactNativeWebViewMessage(event);\n }, []);\n\n if (!mountOverlay) {\n return null;\n }\n\n return (\n <SafeAreaView ref={containerRef} style={containerStyles}>\n <WebView\n ref={webViewRef}\n source={{ uri: `${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}` }}\n onMessage={handleWebViewMessage}\n style={this.styles['magic-webview']}\n webviewDebuggingEnabled\n autoManageStatusBarEnabled={false}\n onShouldStartLoadWithRequest={event => {\n const queryParams = new URLSearchParams(event.url.split('?')[1]);\n const openInDeviceBrowser = queryParams.get(OPEN_IN_DEVICE_BROWSER);\n\n if (openInDeviceBrowser) {\n Linking.openURL(event.url);\n return false;\n }\n return true;\n }}\n />\n </SafeAreaView>\n );\n };\n\n /**\n * Route incoming messages from a React Native `<WebView>`.\n */\n private handleReactNativeWebViewMessage(event: any) {\n const url = new URL(`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`);\n\n if (\n event.nativeEvent &&\n typeof event.nativeEvent.data === 'string' &&\n /* Backward compatible */\n (event.nativeEvent.url === `${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}` ||\n event.nativeEvent.url === `${this.endpoint}/send/?params=${this.parameters}` ||\n event.nativeEvent.url === this.endpoint ||\n event.nativeEvent.title === `${url.hostname}/send/${url.search}`)\n ) {\n // Special parsing logic when dealing with TypedArray in the payload\n // Such change is required as JSON.stringify will manipulate the object and cause exceptions during parsing\n // The typed Array is stringified in Mgbox with a flag as notation.\n const data: any = JSON.parse(event.nativeEvent.data, (key, value) => {\n try {\n if (value && typeof value === 'object' && value.flag && value.flag === MAGIC_PAYLOAD_FLAG_TYPED_ARRAY) {\n return new (global[value.constructor as keyof Global] as any)(value.data.split(','));\n }\n\n // silently handles exception and return the original copy\n } catch (e) {\n console.log('Error parsing data', e);\n }\n return value;\n });\n if (data && data.msgType && this.messageHandlers.size) {\n // If the response object is undefined, we ensure it's at least an\n // empty object before passing to the event listener.\n\n data.response = data.response ?? {};\n\n // Reconstruct event from RN event\n const magicEvent: MagicMessageEvent = { data } as MagicMessageEvent;\n for (const handler of this.messageHandlers.values()) {\n handler(magicEvent);\n }\n }\n }\n }\n\n private async msgPostedAfterInactivity() {\n const lastPostTimestamp: string | null = await AsyncStorage.getItem(LAST_MESSAGE_TIME);\n if (lastPostTimestamp) {\n const lastPostDate = new Date(lastPostTimestamp).getTime();\n const now = new Date().getTime();\n const fiveMinutes = 5 * 60 * 1000;\n\n // there's been inactivity if the new message is posted\n // 5 min or more after the last message.\n return now - lastPostDate > fiveMinutes;\n }\n return false;\n }\n\n protected hideOverlay() {\n if (this.container) this.container.hideOverlay();\n }\n\n protected showOverlay() {\n if (this.container) this.container.showOverlay();\n }\n\n protected async _post(data: any) {\n if (await this.msgPostedAfterInactivity()) {\n EventRegister.emit(MSG_POSTED_AFTER_INACTIVITY_EVENT, data);\n return;\n }\n if (this.webView && this.webView.postMessage) {\n this.webView.postMessage(\n JSON.stringify(data, (key, value) => {\n // parse Typed Array to Stringify object\n if (isTypedArray(value)) {\n return {\n constructor: value.constructor.name,\n data: value.toString(),\n flag: MAGIC_PAYLOAD_FLAG_TYPED_ARRAY,\n };\n }\n return value;\n }),\n );\n AsyncStorage.setItem(LAST_MESSAGE_TIME, new Date().toISOString());\n } else {\n throw createModalNotReadyError();\n }\n }\n\n // Overrides parent method to keep refresh token in keychain\n async persistMagicEventRefreshToken(event: MagicMessageEvent) {\n if (!event?.data?.rt) {\n return;\n }\n\n await setRefreshTokenInKeychain(event.data.rt);\n }\n\n // Overrides parent method to retrieve refresh token from keychain while creating a request\n async getRT(): Promise<string | null> {\n return await getRefreshTokenInKeychain();\n }\n\n async getJWT(): Promise<string | null | undefined> {\n try {\n return await getDpop();\n } catch (e) {\n return null;\n }\n }\n\n // Todo - implement these methods\n /* istanbul ignore next */\n protected checkRelayerExistsInDOM(): Promise<boolean> {\n return Promise.resolve(true);\n }\n\n /* istanbul ignore next */\n protected reloadRelayer(): Promise<void> {\n return Promise.resolve(undefined);\n }\n}\n", "import { useEffect, useState } from 'react';\nimport NetInfo, { NetInfoState } from '@react-native-community/netinfo';\n\nexport const useInternetConnection = () => {\n const [isConnected, setIsConnected] = useState(true);\n useEffect(() => {\n const handleConnectionChange = (connectionInfo: NetInfoState) => {\n setIsConnected(!!connectionInfo.isConnected);\n };\n\n // Subscribe to connection changes and cleanup on unmount\n return NetInfo.addEventListener(handleConnectionChange);\n }, []);\n\n return isConnected;\n};\n", "import * as Keychain from 'react-native-keychain';\nimport { getKeyAlias } from './utils/key-alias';\n\nconst SERVICE = getKeyAlias('refreshTokenService');\nconst KEY = getKeyAlias('refreshToken');\n\nlet cachedRefreshToken: string | null = null;\n\nexport const setRefreshTokenInKeychain = async (rt: string) => {\n // Skip write if token hasn't changed\n if (cachedRefreshToken === rt) {\n return true;\n }\n\n try {\n const result = await Keychain.setGenericPassword(KEY, rt, {\n service: SERVICE,\n accessible: Keychain.ACCESSIBLE.AFTER_FIRST_UNLOCK_THIS_DEVICE_ONLY,\n });\n cachedRefreshToken = rt; // Update cache on successful write\n return result;\n } catch (error) {\n console.error('Failed to set refresh token in keychain', error);\n return false;\n }\n};\n\nexport const getRefreshTokenInKeychain = async (): Promise<string | null> => {\n // Return cached value if available\n if (cachedRefreshToken !== null) {\n return cachedRefreshToken;\n }\n\n try {\n const keychainEntry = await Keychain.getGenericPassword({ service: SERVICE });\n if (!keychainEntry) return null;\n\n cachedRefreshToken = keychainEntry.password;\n return cachedRefreshToken;\n } catch (error) {\n console.error('Failed to get refresh token in keychain', error);\n return null;\n }\n};\n\nexport const removeRefreshTokenInKeychain = async () => {\n try {\n cachedRefreshToken = null; // Clear cache\n return await Keychain.resetGenericPassword({ service: SERVICE });\n } catch (error) {\n console.error('Failed to remove refresh token in keychain', error);\n return null;\n }\n};\n", "import DeviceInfo from 'react-native-device-info';\n\nconst KEY_SUFFIX_MAP = {\n dpop: 'magic.sdk.dpop',\n refreshToken: 'magic.sdk.rt',\n refreshTokenService: 'magic.sdk.rt.service',\n};\n\n/**\n * Returns the key alias for the given key.\n * Let's us to safely store the keys in the keychain and avoid conflicts with other apps using magic sdk.\n */\nexport function getKeyAlias(key: keyof typeof KEY_SUFFIX_MAP): string {\n const appId = DeviceInfo.getBundleId();\n return `${appId}.${KEY_SUFFIX_MAP[key]}`;\n}\n", "import uuid from 'react-native-uuid';\nimport { toBase64Url } from './utils/uint8';\nimport { spkiToJwk } from './utils/jwk';\nimport { ALG, TYP } from './constants';\nimport { derToRawSignature } from './utils/der';\nimport { DpopClaims, DpopHeader } from './types';\nimport DeviceCrypto, { AccessLevel } from 'react-native-device-crypto';\nimport { getKeyAlias } from './utils/key-alias';\n\nconst KEY_ALIAS = getKeyAlias('dpop');\n\n/**\n * Generates the DPoP proof compatible with the Python backend.\n * Handles key creation (if missing), JWK construction, and signing.\n */\nexport const getDpop = async (): Promise<string | null> => {\n try {\n // 1. Get or Create Key in Secure Enclave\n // We strictly disable authentication to avoid biometric prompts\n const publicKey = await DeviceCrypto.getOrCreateAsymmetricKey(KEY_ALIAS, {\n accessLevel: AccessLevel.ALWAYS, // Key is always accessible in this device\n invalidateOnNewBiometry: false,\n });\n\n // 2. Prepare Public Key as JWK\n // Toaster backend expects JWK in the header\n const publicJwk = spkiToJwk(publicKey);\n\n // 3. Construct Payload\n const now = Math.floor(Date.now() / 1000);\n const claims: DpopClaims = {\n iat: now,\n jti: uuid.v4(),\n };\n\n const header: DpopHeader = {\n typ: TYP,\n alg: ALG,\n jwk: publicJwk,\n };\n\n // 4. Prepare Signing Input\n const headerB64 = toBase64Url(JSON.stringify(header));\n const payloadB64 = toBase64Url(JSON.stringify(claims));\n const signingInput = `${headerB64}.${payloadB64}`;\n\n // 5. Sign Data\n // DeviceCrypto returns a Base64 signature.\n const signatureBase64 = await DeviceCrypto.sign(KEY_ALIAS, signingInput, {\n // Biometry prompts should not be fired since the key is always accessible in this device\n biometryTitle: 'Sign DPoP',\n biometrySubTitle: 'Sign DPoP',\n biometryDescription: 'Sign DPoP',\n });\n\n // 6. Convert Signature (Toaster expects Raw R|S)\n const signatureB64 = derToRawSignature(signatureBase64);\n\n return `${signingInput}.${signatureB64}`;\n } catch (error) {\n console.error('DPoP Generation Error:', error);\n return null;\n }\n};\n\n/**\n * Removes the keys from the Secure Enclave\n * Returns true if the key was deleted successfully, false otherwise.\n * @returns {Promise<boolean>} True if the key was deleted successfully, false otherwise.\n */\nexport const deleteDpop = async (): Promise<boolean> => {\n try {\n return await DeviceCrypto.deleteKey(KEY_ALIAS);\n } catch (error) {\n console.error('DPoP Deletion Error:', error);\n return false;\n }\n};\n", "/**\n * Encodes a Uint8Array to a Base64 string.\n * Uses native 'btoa' by converting bytes to a binary string first.\n */\nexport const uint8ArrayToBase64 = (bytes: Uint8Array): string => {\n let binary = '';\n const len = bytes.byteLength;\n for (let i = 0; i < len; i++) {\n binary += String.fromCharCode(bytes[i]);\n }\n return btoa(binary);\n};\n\n/**\n * Decodes a Base64 string to a Uint8Array.\n * Uses native 'atob'.\n */\nexport const base64ToUint8Array = (base64: string): Uint8Array => {\n const binaryString = atob(base64);\n const len = binaryString.length;\n const bytes = new Uint8Array(len);\n for (let i = 0; i < len; i++) {\n bytes[i] = binaryString.charCodeAt(i);\n }\n return bytes;\n};\n\n/**\n * Converts a string (UTF-8) to Uint8Array.\n * Polyfill for simple ASCII/UTF-8 if TextEncoder is not available,\n * but TextEncoder is standard in RN 0.68+.\n */\nexport const stringToUint8Array = (str: string): Uint8Array => {\n // Use TextEncoder if available (standard in modern RN)\n if (typeof TextEncoder !== 'undefined') {\n return new TextEncoder().encode(str);\n }\n // Fallback for older environments\n const arr = [];\n for (let i = 0; i < str.length; i++) {\n let code = str.charCodeAt(i);\n if (code < 0x80) {\n arr.push(code);\n } else if (code < 0x800) {\n arr.push(0xc0 | (code >> 6), 0x80 | (code & 0x3f));\n } else if (code < 0xd800 || code >= 0xe000) {\n arr.push(0xe0 | (code >> 12), 0x80 | ((code >> 6) & 0x3f), 0x80 | (code & 0x3f));\n } else {\n i++;\n code = 0x10000 + (((code & 0x3ff) << 10) | (str.charCodeAt(i) & 0x3ff));\n arr.push(0xf0 | (code >> 18), 0x80 | ((code >> 12) & 0x3f), 0x80 | ((code >> 6) & 0x3f), 0x80 | (code & 0x3f));\n }\n }\n return new Uint8Array(arr);\n};\n\n/**\n * Encodes input (String or Uint8Array) to Base64Url (RFC 4648).\n * Removes padding '=', replaces '+' with '-', and '/' with '_'\n */\nexport const toBase64Url = (input: string | Uint8Array): string => {\n let bytes: Uint8Array;\n\n if (typeof input === 'string') {\n bytes = stringToUint8Array(input);\n } else {\n bytes = input;\n }\n\n const base64 = uint8ArrayToBase64(bytes);\n return base64.replace(/\\+/g, '-').replace(/\\//g, '_').replace(/=+$/, '');\n};\n", "import { base64ToUint8Array, toBase64Url } from './uint8';\n\n/**\n * Converts SPKI Public Key (Base64) to JWK format.\n * Extracts Raw X and Y coordinates from the uncompressed point.\n */\nexport const spkiToJwk = (spkiBase64: string) => {\n const buf = base64ToUint8Array(spkiBase64);\n // P-256 SPKI Header is 26 bytes. The key data (0x04 + X + Y) is at the end.\n // We explicitly look for the last 65 bytes (1 header byte + 32 bytes X + 32 bytes Y)\n const rawKey = buf.subarray(buf.length - 65);\n\n if (rawKey[0] !== 0x04) {\n throw new Error('Invalid Public Key format: Expected uncompressed point');\n }\n\n const x = rawKey.subarray(1, 33);\n const y = rawKey.subarray(33, 65);\n\n return {\n kty: 'EC',\n crv: 'P-256',\n x: toBase64Url(x),\n y: toBase64Url(y),\n };\n};\n", "export const ALG = 'ES256';\nexport const TYP = 'dpop+jwt';\n", "import { base64ToUint8Array, toBase64Url } from './uint8';\n\n/**\n * Converts a DER encoded signature (ASN.1) to a Raw R|S signature (64 bytes).\n * Device Crypto returns DER; Toaster backend expects Raw P1363.\n */\nexport const derToRawSignature = (derBase64: string): string => {\n const der = base64ToUint8Array(derBase64);\n\n // DER Structure: 0x30 | total_len | 0x02 | r_len | r_bytes | 0x02 | s_len | s_bytes\n let offset = 2; // Skip Sequence Tag (0x30) and Total Length\n\n // --- Read R ---\n if (der[offset] !== 0x02) throw new Error('Invalid DER: R tag missing');\n offset++; // skip tag\n const rLen = der[offset++]; // read length\n let rBytes = der.subarray(offset, offset + rLen);\n offset += rLen;\n\n // Handle ASN.1 Integer padding for R (remove leading 0x00 if present)\n if (rLen === 33 && rBytes[0] === 0x00) {\n rBytes = rBytes.subarray(1);\n }\n\n // --- Read S ---\n if (der[offset] !== 0x02) throw new Error('Invalid DER: S tag missing');\n offset++; // skip tag\n const sLen = der[offset++]; // read length\n let sBytes = der.subarray(offset, offset + sLen);\n\n // Handle ASN.1 Integer padding for S\n if (sLen === 33 && sBytes[0] === 0x00) {\n sBytes = sBytes.subarray(1);\n }\n\n // --- Construct Raw Signature (64 bytes) ---\n const rawSignature = new Uint8Array(64);\n\n // Copy R into the first 32 bytes (right-aligned/padded)\n rawSignature.set(rBytes, 32 - rBytes.length);\n\n // Copy S into the last 32 bytes (right-aligned/padded)\n rawSignature.set(sBytes, 64 - sBytes.length);\n\n return toBase64Url(rawSignature);\n};\n", "import { NativeModules, Platform } from 'react-native';\n\ninterface NativeModuleCheck {\n name: string;\n nativeModuleName: string;\n packageName: string;\n}\n\nconst REQUIRED_NATIVE_MODULES: NativeModuleCheck[] = [\n {\n name: 'react-native-keychain',\n nativeModuleName: 'RNKeychainManager',\n packageName: 'react-native-keychain',\n },\n {\n name: 'react-native-device-crypto',\n nativeModuleName: 'DeviceCrypto',\n packageName: 'react-native-device-crypto',\n },\n];\n\n// Track if warning has been shown to avoid spamming\nlet hasWarned = false;\n\n/**\n * Checks if all required native modules are properly installed and linked.\n * Logs a warning if any native module is missing.\n *\n * Note: Some native modules (like react-native-device-crypto) require hardware\n * features (e.g., Secure Enclave) that are not available in simulators/emulators.\n * The SDK will continue to work but certain security features may be degraded.\n */\nexport const checkNativeModules = (): void => {\n if (hasWarned) return;\n\n const missingModules: NativeModuleCheck[] = [];\n\n for (const module of REQUIRED_NATIVE_MODULES) {\n if (!NativeModules[module.nativeModuleName]) {\n missingModules.push(module);\n }\n }\n\n if (missingModules.length > 0) {\n hasWarned = true;\n\n const platform = Platform.OS;\n const moduleList = missingModules.map(m => ` - ${m.packageName}`).join('\\n');\n const installCommands = missingModules.map(m => `npm install ${m.packageName}`).join('\\n');\n\n const iosInstructions =\n platform === 'ios'\n ? `\nFor iOS, run:\n cd ios && pod install && cd ..\n`\n : '';\n\n const androidInstructions =\n platform === 'android'\n ? `\nFor Android, rebuild your app:\n npx react-native run-android\n`\n : '';\n\n console.warn(\n `@magic-sdk/react-native-bare: Missing native modules detected.\n\nThe following native modules are not linked:\n${moduleList}\n\nThe SDK will continue to work, but some security features may not function properly.\n\nNote: If you're running in a simulator/emulator, some native modules (like react-native-device-crypto) \nrequire hardware features (Secure Enclave) that are only available on physical devices.\n\nIf you're on a physical device and see this warning, please ensure the packages are installed and linked:\n\n1. Install the missing packages:\n${installCommands}\n\n2. Link the native modules:\n${iosInstructions}${androidInstructions}\n3. Rebuild your app completely.\n`,\n );\n }\n};\n", "import { SDKBase } from '@magic-sdk/provider';\nimport { ReactNativeWebViewController } from './react-native-webview-controller';\n\nexport class SDKBaseReactNative extends SDKBase {\n public get Relayer() {\n return (this.overlay as ReactNativeWebViewController).Relayer;\n }\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["createSDK", "processPolyfill", "URLPolyfill", "URLSearchParamsPolyfill", "Buffer", "_", "getBundleId", "React", "useState", "useCallback", "useMemo", "useEffect", "Linking", "StyleSheet", "WebView", "SafeAreaView", "ViewController", "createModalNotReadyError", "isTypedArray", "AsyncStorage", "EventRegister", "useEffect", "useState", "NetInfo", "useInternetConnection", "isConnected", "setIsConnected", "handleConnectionChange", "connectionInfo", "Keychain", "DeviceInfo", "KEY_SUFFIX_MAP", "getKeyAlias", "key", "SERVICE", "getKeyAlias", "KEY", "cachedRefreshToken", "setRefreshTokenInKeychain", "rt", "result", "error", "getRefreshTokenInKeychain", "keychainEntry", "uuid", "uint8ArrayToBase64", "bytes", "binary", "len", "i", "base64ToUint8Array", "base64", "binaryString", "stringToUint8Array", "str", "arr", "code", "toBase64Url", "input", "spkiToJwk", "spkiBase64", "buf", "base64ToUint8Array", "rawKey", "x", "y", "toBase64Url", "ALG", "TYP", "derToRawSignature", "derBase64", "der", "base64ToUint8Array", "offset", "rLen", "rBytes", "sLen", "sBytes", "rawSignature", "toBase64Url", "DeviceCrypto", "AccessLevel", "KEY_ALIAS", "getKeyAlias", "getDpop", "publicKey", "DeviceCrypto", "AccessLevel", "publicJwk", "spkiToJwk", "claims", "uuid", "headerB64", "toBase64Url", "TYP", "ALG", "payloadB64", "signingInput", "signatureBase64", "signatureB64", "derToRawSignature", "error", "NativeModules", "Platform", "REQUIRED_NATIVE_MODULES", "hasWarned", "checkNativeModules", "missingModules", "module", "platform", "moduleList", "m", "installCommands", "MAGIC_PAYLOAD_FLAG_TYPED_ARRAY", "OPEN_IN_DEVICE_BROWSER", "DEFAULT_BACKGROUND_COLOR", "MSG_POSTED_AFTER_INACTIVITY_EVENT", "LAST_MESSAGE_TIME", "createWebViewStyles", "StyleSheet", "ReactNativeWebViewController", "ViewController", "backgroundColor", "show", "setShow", "useState", "mountOverlay", "setMountOverlay", "isConnected", "useInternetConnection", "useEffect", "checkNativeModules", "AsyncStorage", "EventRegister", "message", "webViewRef", "useCallback", "webView", "containerRef", "view", "showOverlay", "hideOverlay", "containerStyles", "useMemo", "handleWebViewMessage", "event", "React", "SafeAreaView", "WebView", "Linking", "url", "data", "key", "value", "e", "magicEvent", "handler", "lastPostTimestamp", "lastPostDate", "now", "fiveMinutes", "isTypedArray", "createModalNotReadyError", "setRefreshTokenInKeychain", "getRefreshTokenInKeychain", "getDpop", "SDKBase", "SDKBaseReactNative", "AsyncStorage", "_", "processPolyfill", "Buffer", "URLPolyfill", "URLSearchParamsPolyfill", "str", "b64Encoded", "Magic", "createSDK", "SDKBaseReactNative", "getBundleId", "ReactNativeWebViewController"]
7
7
  }
@@ -0,0 +1,9 @@
1
+ /**
2
+ * Checks if all required native modules are properly installed and linked.
3
+ * Logs a warning if any native module is missing.
4
+ *
5
+ * Note: Some native modules (like react-native-device-crypto) require hardware
6
+ * features (e.g., Secure Enclave) that are not available in simulators/emulators.
7
+ * The SDK will continue to work but certain security features may be degraded.
8
+ */
9
+ export declare const checkNativeModules: () => void;
@@ -0,0 +1,2 @@
1
+ export declare const ALG = "ES256";
2
+ export declare const TYP = "dpop+jwt";
@@ -0,0 +1,11 @@
1
+ /**
2
+ * Generates the DPoP proof compatible with the Python backend.
3
+ * Handles key creation (if missing), JWK construction, and signing.
4
+ */
5
+ export declare const getDpop: () => Promise<string | null>;
6
+ /**
7
+ * Removes the keys from the Secure Enclave
8
+ * Returns true if the key was deleted successfully, false otherwise.
9
+ * @returns {Promise<boolean>} True if the key was deleted successfully, false otherwise.
10
+ */
11
+ export declare const deleteDpop: () => Promise<boolean>;
@@ -0,0 +1,4 @@
1
+ import * as Keychain from 'react-native-keychain';
2
+ export declare const setRefreshTokenInKeychain: (rt: string) => Promise<boolean | Keychain.Result>;
3
+ export declare const getRefreshTokenInKeychain: () => Promise<string | null>;
4
+ export declare const removeRefreshTokenInKeychain: () => Promise<boolean | null>;
@@ -0,0 +1,18 @@
1
+ export interface DpopHeader {
2
+ typ: 'dpop+jwt';
3
+ alg: 'ES256';
4
+ jwk: JsonWebKey;
5
+ }
6
+ export interface DpopClaims {
7
+ iat: number;
8
+ jti: string;
9
+ htm?: string;
10
+ htu?: string;
11
+ }
12
+ export interface JsonWebKey {
13
+ kty: string;
14
+ crv: string;
15
+ x: string;
16
+ y: string;
17
+ ext?: boolean;
18
+ }
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Converts a DER encoded signature (ASN.1) to a Raw R|S signature (64 bytes).
3
+ * Device Crypto returns DER; Toaster backend expects Raw P1363.
4
+ */
5
+ export declare const derToRawSignature: (derBase64: string) => string;
@@ -0,0 +1,10 @@
1
+ /**
2
+ * Converts SPKI Public Key (Base64) to JWK format.
3
+ * Extracts Raw X and Y coordinates from the uncompressed point.
4
+ */
5
+ export declare const spkiToJwk: (spkiBase64: string) => {
6
+ kty: string;
7
+ crv: string;
8
+ x: string;
9
+ y: string;
10
+ };
@@ -0,0 +1,11 @@
1
+ declare const KEY_SUFFIX_MAP: {
2
+ dpop: string;
3
+ refreshToken: string;
4
+ refreshTokenService: string;
5
+ };
6
+ /**
7
+ * Returns the key alias for the given key.
8
+ * Let's us to safely store the keys in the keychain and avoid conflicts with other apps using magic sdk.
9
+ */
10
+ export declare function getKeyAlias(key: keyof typeof KEY_SUFFIX_MAP): string;
11
+ export {};
@@ -0,0 +1,21 @@
1
+ /**
2
+ * Encodes a Uint8Array to a Base64 string.
3
+ * Uses native 'btoa' by converting bytes to a binary string first.
4
+ */
5
+ export declare const uint8ArrayToBase64: (bytes: Uint8Array) => string;
6
+ /**
7
+ * Decodes a Base64 string to a Uint8Array.
8
+ * Uses native 'atob'.
9
+ */
10
+ export declare const base64ToUint8Array: (base64: string) => Uint8Array;
11
+ /**
12
+ * Converts a string (UTF-8) to Uint8Array.
13
+ * Polyfill for simple ASCII/UTF-8 if TextEncoder is not available,
14
+ * but TextEncoder is standard in RN 0.68+.
15
+ */
16
+ export declare const stringToUint8Array: (str: string) => Uint8Array;
17
+ /**
18
+ * Encodes input (String or Uint8Array) to Base64Url (RFC 4648).
19
+ * Removes padding '=', replaces '+' with '-', and '/' with '_'
20
+ */
21
+ export declare const toBase64Url: (input: string | Uint8Array) => string;
@@ -1,5 +1,6 @@
1
1
  import React from 'react';
2
2
  import { ViewController } from '@magic-sdk/provider';
3
+ import { MagicMessageEvent } from '@magic-sdk/types';
3
4
  /**
4
5
  * View controller for the Magic `<WebView>` overlay.
5
6
  */
@@ -23,6 +24,9 @@ export declare class ReactNativeWebViewController extends ViewController {
23
24
  protected hideOverlay(): void;
24
25
  protected showOverlay(): void;
25
26
  protected _post(data: any): Promise<void>;
27
+ persistMagicEventRefreshToken(event: MagicMessageEvent): Promise<void>;
28
+ getRT(): Promise<string | null>;
29
+ getJWT(): Promise<string | null | undefined>;
26
30
  protected checkRelayerExistsInDOM(): Promise<boolean>;
27
31
  protected reloadRelayer(): Promise<void>;
28
32
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@magic-sdk/react-native-bare",
3
- "version": "34.0.0",
3
+ "version": "34.1.0-canary.964.20943413682.0",
4
4
  "description": "Passwordless authentication for React Native (Bare).",
5
5
  "author": "Magic Labs <team@magic.link> (https://magic.link/)",
6
6
  "license": "MIT",
@@ -10,7 +10,8 @@
10
10
  },
11
11
  "homepage": "https://magic.link",
12
12
  "files": [
13
- "dist"
13
+ "dist",
14
+ "*.podspec"
14
15
  ],
15
16
  "target": "node",
16
17
  "main": "./dist/cjs/index.js",
@@ -28,6 +29,7 @@
28
29
  "process": "~0.11.10",
29
30
  "react-native-device-info": "^10.3.0",
30
31
  "react-native-event-listeners": "^1.0.7",
32
+ "react-native-uuid": "^2.0.3",
31
33
  "regenerator-runtime": "0.13.9",
32
34
  "tslib": "^2.0.3",
33
35
  "whatwg-url": "~8.1.0"
@@ -38,7 +40,9 @@
38
40
  "@testing-library/react-native": "^13.2.0",
39
41
  "react": "~19.1.0",
40
42
  "react-native": "~0.78.1",
43
+ "react-native-device-crypto": "^0.1.7",
41
44
  "react-native-device-info": "^10.3.0",
45
+ "react-native-keychain": "^10.0.0",
42
46
  "react-native-safe-area-context": "5.3.0",
43
47
  "react-native-webview": "^13.3.0",
44
48
  "react-test-renderer": "^19.1.0"
@@ -48,9 +52,11 @@
48
52
  "@react-native-community/netinfo": ">=9.0.0",
49
53
  "react": ">=16",
50
54
  "react-native": ">=0.60",
55
+ "react-native-device-crypto": "^0.1.7",
51
56
  "react-native-device-info": ">=10.3.0",
57
+ "react-native-keychain": "^10.0.0",
52
58
  "react-native-safe-area-context": ">=4.4.1",
53
59
  "react-native-webview": ">=12.4.0"
54
60
  },
55
- "gitHead": "f5f99093e297182a8903ef84ec8da017216efb1e"
61
+ "gitHead": "f7fff21be3219978c98c34c872565dcde4d8307c"
56
62
  }