@magic-sdk/react-native-bare 32.2.0 → 32.2.1-canary.964.19821266156.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.
- package/dist/cjs/index.js +1 -1
- package/dist/cjs/index.js.map +4 -4
- package/dist/es/index.js +1 -1
- package/dist/es/index.js.map +4 -4
- package/dist/types/dpop/constants/index.d.ts +3 -0
- package/dist/types/dpop/dpop.d.ts +11 -0
- package/dist/types/dpop/utils/der.d.ts +5 -0
- package/dist/types/dpop/utils/dpop-alias.d.ts +1 -0
- package/dist/types/dpop/utils/jwk.d.ts +10 -0
- package/dist/types/dpop/utils/uint8.d.ts +21 -0
- package/dist/types/keychain.d.ts +4 -0
- package/dist/types/react-native-webview-controller.d.ts +4 -0
- package/package.json +6 -3
package/dist/cjs/index.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
"use strict";var
|
|
1
|
+
"use strict";var ae=Object.create;var A=Object.defineProperty;var ie=Object.getOwnPropertyDescriptor;var ce=Object.getOwnPropertyNames;var le=Object.getPrototypeOf,pe=Object.prototype.hasOwnProperty;var me=(o,e)=>{for(var t in e)A(o,t,{get:e[t],enumerable:!0})},x=(o,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let a of ce(e))!pe.call(o,a)&&a!==t&&A(o,a,{get:()=>e[a],enumerable:!(r=ie(e,a))||r.enumerable});return o},s=(o,e,t)=>(x(o,e,"default"),t&&x(t,e,"default")),f=(o,e,t)=>(t=o!=null?ae(le(o)):{},x(e||!o||!o.__esModule?A(t,"default",{value:o,enumerable:!0}):t,o)),fe=o=>x(A({},"__esModule",{value:!0}),o);var n={};me(n,{Magic:()=>Ee,useInternetConnection:()=>v});module.exports=fe(n);var Je=require("regenerator-runtime/runtime"),oe=require("@magic-sdk/provider"),be=f(require("process")),D=require("whatwg-url"),k=require("buffer"),ne=f(require("lodash")),se=require("react-native-device-info");var i=f(require("react")),P=require("react-native"),Z=require("react-native-webview"),ee=require("react-native-safe-area-context"),T=require("@magic-sdk/provider"),te=require("lodash"),w=f(require("@react-native-async-storage/async-storage")),B=require("react-native-event-listeners");var S=require("react"),M=f(require("@react-native-community/netinfo")),v=()=>{let[o,e]=(0,S.useState)(!0);return(0,S.useEffect)(()=>{let t=r=>{e(!!r.isConnected)};return M.default.addEventListener(t)},[]),o};var u=f(require("react-native-keychain")),V="magic_sdk_rt",ye="magic_rt",N=async o=>await u.setGenericPassword(ye,o,{service:V,accessible:u.ACCESSIBLE.AFTER_FIRST_UNLOCK_THIS_DEVICE_ONLY}),F=async()=>{let o=await u.getGenericPassword({service:V});return o?o.password:null};var p=f(require("@sbaiahmed1/react-native-biometrics")),z=f(require("react-native-uuid"));var ge=o=>{let e="",t=o.byteLength;for(let r=0;r<t;r++)e+=String.fromCharCode(o[r]);return btoa(e)},I=o=>{let e=atob(o),t=e.length,r=new Uint8Array(t);for(let a=0;a<t;a++)r[a]=e.charCodeAt(a);return r},ue=o=>{if(typeof TextEncoder<"u")return new TextEncoder().encode(o);let e=[];for(let t=0;t<o.length;t++){let r=o.charCodeAt(t);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):(t++,r=65536+((r&1023)<<10|o.charCodeAt(t)&1023),e.push(240|r>>18,128|r>>12&63,128|r>>6&63,128|r&63))}return new Uint8Array(e)},y=o=>{let e;return typeof o=="string"?e=ue(o):e=o,ge(e).replace(/\+/g,"-").replace(/\//g,"_").replace(/=+$/,"")};var $=o=>{let e=I(o),t=e.subarray(e.length-65);if(t[0]!==4)throw new Error("Invalid Public Key format: Expected uncompressed point");let r=t.subarray(1,33),a=t.subarray(33,65);return{kty:"EC",crv:"P-256",x:y(r),y:y(a)}};var G="magiclink.dpop",W="ES256",Y="dpop+jwt";var J=o=>{let e=I(o),t=2;if(e[t]!==2)throw new Error("Invalid DER: R tag missing");t++;let r=e[t++],a=e.subarray(t,t+r);if(t+=r,r===33&&a[0]===0&&(a=a.subarray(1)),e[t]!==2)throw new Error("Invalid DER: S tag missing");t++;let c=e[t++],m=e.subarray(t,t+c);c===33&&m[0]===0&&(m=m.subarray(1));let g=new Uint8Array(64);return g.set(a,32-a.length),g.set(m,64-m.length),y(g)};var q=require("@sbaiahmed1/react-native-biometrics");var j=async()=>`${await(0,q.getDefaultKeyAlias)()}.${G}`;var H=async(o,e)=>{try{let t=await j();await p.configureKeyAlias(t);let r=await p.getKeyAttributes(t),a="";if(!r.exists)a=(await p.createKeys(t,"ec256",p.BiometricStrength.Weak)).publicKey;else{let E=await p.getAllKeys(t);E.keys.length>0?a=E.keys[0].publicKey:a=(await p.createKeys(t,"ec256",p.BiometricStrength.Weak)).publicKey}let c=$(a),m=Math.floor(Date.now()/1e3),g=z.default.v4(),h={iat:m,jti:g};o&&(h.htm=o.toUpperCase()),e&&(h.htu=e);let K=y(JSON.stringify({typ:Y,alg:W,jwk:c})),O=y(JSON.stringify(h)),b=`${K}.${O}`,l=await p.verifyKeySignature(t,b);if(!l.success||!l.signature)throw new Error(`Signing failed: ${l.error||"No signature returned"}`);let U=J(l.signature);return`${b}.${U}`}catch(t){throw console.error("DPoP Generation Error:",t),t}};var X="MAGIC_PAYLOAD_FLAG_TYPED_ARRAY",de="open_in_device_browser",he="#FFFFFF",Q="msg_posted_after_inactivity_event",R="lastMessageTime";function we(){return P.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 _=class extends T.ViewController{webView;container;styles;init(){this.webView=null,this.container=null,this.styles=we()}Relayer=e=>{let[t,r]=(0,i.useState)(!1),[a,c]=(0,i.useState)(!0),m=v();(0,i.useEffect)(()=>{this.isConnectedToInternet=m},[m]),(0,i.useEffect)(()=>(w.default.setItem(R,""),()=>{this.isReadyForRequest=!1}),[]),(0,i.useEffect)(()=>{B.EventRegister.addEventListener(Q,async l=>{this.isReadyForRequest=!1,c(!1),this.post(l.msgType,l.payload),await w.default.setItem(R,new Date().toISOString())})},[]),(0,i.useEffect)(()=>{a||setTimeout(()=>c(!0),10)},[a]);let g=(0,i.useCallback)(l=>{this.webView=l},[]),h=(0,i.useCallback)(l=>{this.container={...l,showOverlay:L,hideOverlay:K}},[]),L=(0,i.useCallback)(()=>{r(!0)},[]),K=(0,i.useCallback)(()=>{r(!1)},[]),O=(0,i.useMemo)(()=>[this.styles["webview-container"],t?{...this.styles.show,backgroundColor:e??he}:this.styles.hide],[t]),b=(0,i.useCallback)(l=>{this.handleReactNativeWebViewMessage(l)},[]);return a?i.default.createElement(ee.SafeAreaView,{ref:h,style:O},i.default.createElement(Z.WebView,{ref:g,source:{uri:`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`},onMessage:b,style:this.styles["magic-webview"],webviewDebuggingEnabled:!0,autoManageStatusBarEnabled:!1,onShouldStartLoadWithRequest:l=>new URLSearchParams(l.url.split("?")[1]).get(de)?(P.Linking.openURL(l.url),!1):!0})):null};handleReactNativeWebViewMessage(e){let t=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===`${t.hostname}/send/${t.search}`)){let r=JSON.parse(e.nativeEvent.data,(a,c)=>{try{if(c&&typeof c=="object"&&c.flag&&c.flag===X)return new global[c.constructor](c.data.split(","))}catch(m){console.log("Error parsing data",m)}return c});if(r&&r.msgType&&this.messageHandlers.size){r.response=r.response??{};let a={data:r};for(let c of this.messageHandlers.values())c(a)}}}async msgPostedAfterInactivity(){let e=await w.default.getItem(R);if(e){let t=new Date(e).getTime(),r=new Date().getTime(),a=5*60*1e3;return r-t>a}return!1}hideOverlay(){this.container&&this.container.hideOverlay()}showOverlay(){this.container&&this.container.showOverlay()}async _post(e){if(await this.msgPostedAfterInactivity()){B.EventRegister.emit(Q,e);return}if(this.webView&&this.webView.postMessage)this.webView.postMessage(JSON.stringify(e,(t,r)=>(0,te.isTypedArray)(r)?{constructor:r.constructor.name,data:r.toString(),flag:X}:r)),w.default.setItem(R,new Date().toISOString());else throw(0,T.createModalNotReadyError)()}async persistMagicEventRefreshToken(e){e.data.rt&&N(e.data.rt)}async getRT(){return F()}async getJWT(){try{console.log("CREATING DPOP");let e=await H();return console.log({dpop:e}),e}catch{return null}}checkRelayerExistsInDOM(){return Promise.resolve(!0)}reloadRelayer(){return Promise.resolve(void 0)}};var re=require("@magic-sdk/provider"),C=class extends re.SDKBase{get Relayer(){return this.overlay.Relayer}};var d=f(require("@react-native-async-storage/async-storage"));s(n,require("@magic-sdk/provider"),module.exports);s(n,require("@magic-sdk/types"),module.exports);global.process=ne.default.merge(global.process,be);global.process.browser=!1;global.Buffer=k.Buffer;global.URL=D.URL;global.URLSearchParams=D.URLSearchParams;global.btoa=o=>k.Buffer.from(o,"binary").toString("base64");global.atob=o=>k.Buffer.from(o,"base64").toString("binary");var Ee=(0,oe.createSDK)(C,{platform:"react-native",sdkName:"@magic-sdk/react-native-bare",version:"32.2.0",bundleId:(0,se.getBundleId)(),defaultEndpoint:"https://box.magic.link/",ViewController:_,configureStorage:async()=>({ready:async()=>Promise.resolve(!0),getItem:d.default.getItem,setItem:d.default.setItem,removeItem:d.default.removeItem,clear:d.default.clear,length:()=>{},key:()=>{},keys:d.default.getAllKeys,iterate:()=>{}})});0&&(module.exports={Magic,useInternetConnection,...require("@magic-sdk/provider"),...require("@magic-sdk/types")});
|
|
2
2
|
//# sourceMappingURL=index.js.map
|
package/dist/cjs/index.js.map
CHANGED
|
@@ -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": "
|
|
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/keychain.ts", "../../src/dpop/dpop.ts", "../../src/dpop/utils/uint8.ts", "../../src/dpop/utils/jwk.ts", "../../src/dpop/constants/index.ts", "../../src/dpop/utils/der.ts", "../../src/dpop/utils/dpop-alias.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 './keychain';\nimport { getDpop } from './dpop/dpop';\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 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 setRefreshTokenInKeychain(event.data.rt);\n }\n\n // Overrides parent method to retrieve refresh token from keychain while creating a request\n async getRT() {\n return getRefreshTokenInKeychain();\n }\n\n async getJWT(): Promise<string | null | undefined> {\n try {\n console.log('CREATING DPOP');\n const dpop = await getDpop();\n console.log({ dpop });\n return dpop;\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';\n\nconst SERVICE = 'magic_sdk_rt';\nconst KEY = 'magic_rt';\n\nexport const setRefreshTokenInKeychain = async (rt: string) => {\n return await Keychain.setGenericPassword(KEY, rt, {\n service: SERVICE,\n accessible: Keychain.ACCESSIBLE.AFTER_FIRST_UNLOCK_THIS_DEVICE_ONLY,\n });\n};\n\nexport const getRefreshTokenInKeychain = async () => {\n const keychainEntry = await Keychain.getGenericPassword({ service: SERVICE });\n if (!keychainEntry) return null;\n\n return keychainEntry.password;\n};\n\nexport const removeRefreshTokenInKeychain = async () => {\n return await Keychain.resetGenericPassword({ service: SERVICE });\n};\n", "import * as Biometrics from '@sbaiahmed1/react-native-biometrics';\nimport uuid from 'react-native-uuid'; // Ensure you have installed: npm install react-native-uuid\nimport { toBase64Url } from './utils/uint8';\nimport { spkiToJwk } from './utils/jwk';\nimport { ALG, TYP } from './constants';\nimport { derToRawSignature } from './utils/der';\nimport { getDpopAlias } from './utils/dpop-alias';\n\n/**\n * Generates the DPoP proof compatible with the Python backend.\n * Handles key creation (if missing), JWK construction, and signing.\n * @param httpMethod - The HTTP method (e.g., 'POST')\n * @param httpUrl - The HTTP URL being accessed\n */\nexport const getDpop = async (httpMethod?: string, httpUrl?: string): Promise<string> => {\n try {\n // 1. Configure Isolation\n const DPOP_KEY_ALIAS = await getDpopAlias();\n await Biometrics.configureKeyAlias(DPOP_KEY_ALIAS);\n\n // 2. Retrieve Public Key (Check if exists)\n const keyAttrs = await Biometrics.getKeyAttributes(DPOP_KEY_ALIAS);\n let publicKeyBase64 = '';\n\n if (!keyAttrs.exists) {\n // Create new keys in Secure Enclave (ec256)\n const keyResult = await Biometrics.createKeys(DPOP_KEY_ALIAS, 'ec256', Biometrics.BiometricStrength.Weak);\n publicKeyBase64 = keyResult.publicKey;\n } else {\n // Retrieve existing key\n const allKeys = await Biometrics.getAllKeys(DPOP_KEY_ALIAS);\n if (allKeys.keys.length > 0) {\n publicKeyBase64 = allKeys.keys[0].publicKey;\n } else {\n // Fallback: Re-create if lookup failed but attributes said it existed\n const keyResult = await Biometrics.createKeys(DPOP_KEY_ALIAS, 'ec256', Biometrics.BiometricStrength.Weak);\n publicKeyBase64 = keyResult.publicKey;\n }\n }\n\n // 3. Construct JWK\n const publicJwk = spkiToJwk(publicKeyBase64);\n\n // 4. Construct Payload\n const iat = Math.floor(Date.now() / 1000);\n const jti = uuid.v4();\n\n const payload: any = {\n iat,\n jti,\n };\n\n if (httpMethod) payload.htm = httpMethod.toUpperCase();\n if (httpUrl) payload.htu = httpUrl;\n\n // 5. Construct Header\n const header = {\n typ: TYP,\n alg: ALG,\n jwk: publicJwk,\n };\n\n // 6. Prepare Signing Input\n const headerB64 = toBase64Url(JSON.stringify(header));\n const payloadB64 = toBase64Url(JSON.stringify(payload));\n const signingInput = `${headerB64}.${payloadB64}`;\n\n // 7. Sign\n const signatureResult = await Biometrics.verifyKeySignature(DPOP_KEY_ALIAS, signingInput);\n\n if (!signatureResult.success || !signatureResult.signature) {\n throw new Error(`Signing failed: ${signatureResult.error || 'No signature returned'}`);\n }\n\n // 8. Convert Signature (Toaster expects Raw R|S)\n const signatureB64 = derToRawSignature(signatureResult.signature);\n\n // 9. Return DPoP String\n return `${signingInput}.${signatureB64}`;\n } catch (error) {\n console.error('DPoP Generation Error:', error);\n throw error;\n }\n};\n\n/**\n * Removes existing keys and invalidates the DPoP\n */\nexport const deleteDpop = async (): Promise<boolean> => {\n try {\n const DPOP_KEY_ALIAS = await getDpopAlias();\n const result = await Biometrics.deleteKeys(DPOP_KEY_ALIAS);\n return result.success;\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", "// Unique alias suffix to isolate SDK keys from the rest of the app\nexport const DPOP_KEY_ALIAS_SUFFIX = 'magiclink.dpop';\nexport 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 * React Native Biometrics returns DER; Python 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 { getDefaultKeyAlias } from '@sbaiahmed1/react-native-biometrics';\nimport { DPOP_KEY_ALIAS_SUFFIX } from '../constants';\n\nexport const getDpopAlias = async () => {\n // Each client integrating our SDK gets unique defaultAlias based on their bundle id or package name\n // We append our own suffix to the defaultAlias\n const defaultAlias = await getDefaultKeyAlias();\n const DPOP_KEY_ALIAS = `${defaultAlias}.${DPOP_KEY_ALIAS_SUFFIX}`;\n\n return DPOP_KEY_ALIAS;\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", "SERVICE", "KEY", "setRefreshTokenInKeychain", "rt", "getRefreshTokenInKeychain", "keychainEntry", "Biometrics", "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", "DPOP_KEY_ALIAS_SUFFIX", "ALG", "TYP", "derToRawSignature", "derBase64", "der", "base64ToUint8Array", "offset", "rLen", "rBytes", "sLen", "sBytes", "rawSignature", "toBase64Url", "import_react_native_biometrics", "getDpopAlias", "DPOP_KEY_ALIAS_SUFFIX", "getDpop", "httpMethod", "httpUrl", "DPOP_KEY_ALIAS", "getDpopAlias", "keyAttrs", "publicKeyBase64", "allKeys", "publicJwk", "spkiToJwk", "iat", "jti", "uuid", "payload", "headerB64", "toBase64Url", "TYP", "ALG", "payloadB64", "signingInput", "signatureResult", "signatureB64", "derToRawSignature", "error", "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", "setRefreshTokenInKeychain", "getRefreshTokenInKeychain", "dpop", "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,2 @@
|
|
|
1
|
-
import"regenerator-runtime/runtime";import{createSDK as
|
|
1
|
+
import"regenerator-runtime/runtime";import{createSDK as pe}from"@magic-sdk/provider";import*as me from"process";import{URL as fe,URLSearchParams as ye}from"whatwg-url";import{Buffer as R}from"buffer";import ge from"lodash";import{getBundleId as ue}from"react-native-device-info";import V,{useState as N,useCallback as y,useMemo as Q,useEffect as w}from"react";import{Linking as Z,StyleSheet as ee}from"react-native";import{WebView as te}from"react-native-webview";import{SafeAreaView as re}from"react-native-safe-area-context";import{ViewController as oe,createModalNotReadyError as ne}from"@magic-sdk/provider";import{isTypedArray as se}from"lodash";import b from"@react-native-async-storage/async-storage";import{EventRegister as F}from"react-native-event-listeners";import{useEffect as W,useState as Y}from"react";import J from"@react-native-community/netinfo";var I=()=>{let[o,e]=Y(!0);return W(()=>{let r=t=>{e(!!t.isConnected)};return J.addEventListener(r)},[]),o};import*as m from"react-native-keychain";var T="magic_sdk_rt",q="magic_rt",C=async o=>await m.setGenericPassword(q,o,{service:T,accessible:m.ACCESSIBLE.AFTER_FIRST_UNLOCK_THIS_DEVICE_ONLY}),D=async()=>{let o=await m.getGenericPassword({service:T});return o?o.password:null};import*as i from"@sbaiahmed1/react-native-biometrics";import X from"react-native-uuid";var j=o=>{let e="",r=o.byteLength;for(let t=0;t<r;t++)e+=String.fromCharCode(o[t]);return btoa(e)},h=o=>{let e=atob(o),r=e.length,t=new Uint8Array(r);for(let n=0;n<r;n++)t[n]=e.charCodeAt(n);return t},z=o=>{if(typeof TextEncoder<"u")return new TextEncoder().encode(o);let e=[];for(let r=0;r<o.length;r++){let t=o.charCodeAt(r);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):(r++,t=65536+((t&1023)<<10|o.charCodeAt(r)&1023),e.push(240|t>>18,128|t>>12&63,128|t>>6&63,128|t&63))}return new Uint8Array(e)},l=o=>{let e;return typeof o=="string"?e=z(o):e=o,j(e).replace(/\+/g,"-").replace(/\//g,"_").replace(/=+$/,"")};var k=o=>{let e=h(o),r=e.subarray(e.length-65);if(r[0]!==4)throw new Error("Invalid Public Key format: Expected uncompressed point");let t=r.subarray(1,33),n=r.subarray(33,65);return{kty:"EC",crv:"P-256",x:l(t),y:l(n)}};var K="magiclink.dpop",O="ES256",B="dpop+jwt";var L=o=>{let e=h(o),r=2;if(e[r]!==2)throw new Error("Invalid DER: R tag missing");r++;let t=e[r++],n=e.subarray(r,r+t);if(r+=t,t===33&&n[0]===0&&(n=n.subarray(1)),e[r]!==2)throw new Error("Invalid DER: S tag missing");r++;let s=e[r++],c=e.subarray(r,r+s);s===33&&c[0]===0&&(c=c.subarray(1));let p=new Uint8Array(64);return p.set(n,32-n.length),p.set(c,64-c.length),l(p)};import{getDefaultKeyAlias as H}from"@sbaiahmed1/react-native-biometrics";var U=async()=>`${await H()}.${K}`;var M=async(o,e)=>{try{let r=await U();await i.configureKeyAlias(r);let t=await i.getKeyAttributes(r),n="";if(!t.exists)n=(await i.createKeys(r,"ec256",i.BiometricStrength.Weak)).publicKey;else{let d=await i.getAllKeys(r);d.keys.length>0?n=d.keys[0].publicKey:n=(await i.createKeys(r,"ec256",i.BiometricStrength.Weak)).publicKey}let s=k(n),c=Math.floor(Date.now()/1e3),p=X.v4(),f={iat:c,jti:p};o&&(f.htm=o.toUpperCase()),e&&(f.htu=e);let S=l(JSON.stringify({typ:B,alg:O,jwk:s})),v=l(JSON.stringify(f)),u=`${S}.${v}`,a=await i.verifyKeySignature(r,u);if(!a.success||!a.signature)throw new Error(`Signing failed: ${a.error||"No signature returned"}`);let P=L(a.signature);return`${u}.${P}`}catch(r){throw console.error("DPoP Generation Error:",r),r}};var $="MAGIC_PAYLOAD_FLAG_TYPED_ARRAY",ae="open_in_device_browser",ie="#FFFFFF",G="msg_posted_after_inactivity_event",E="lastMessageTime";function ce(){return ee.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 x=class extends oe{webView;container;styles;init(){this.webView=null,this.container=null,this.styles=ce()}Relayer=e=>{let[r,t]=N(!1),[n,s]=N(!0),c=I();w(()=>{this.isConnectedToInternet=c},[c]),w(()=>(b.setItem(E,""),()=>{this.isReadyForRequest=!1}),[]),w(()=>{F.addEventListener(G,async a=>{this.isReadyForRequest=!1,s(!1),this.post(a.msgType,a.payload),await b.setItem(E,new Date().toISOString())})},[]),w(()=>{n||setTimeout(()=>s(!0),10)},[n]);let p=y(a=>{this.webView=a},[]),f=y(a=>{this.container={...a,showOverlay:_,hideOverlay:S}},[]),_=y(()=>{t(!0)},[]),S=y(()=>{t(!1)},[]),v=Q(()=>[this.styles["webview-container"],r?{...this.styles.show,backgroundColor:e??ie}:this.styles.hide],[r]),u=y(a=>{this.handleReactNativeWebViewMessage(a)},[]);return n?V.createElement(re,{ref:f,style:v},V.createElement(te,{ref:p,source:{uri:`${this.endpoint}/send/?params=${encodeURIComponent(this.parameters)}`},onMessage:u,style:this.styles["magic-webview"],webviewDebuggingEnabled:!0,autoManageStatusBarEnabled:!1,onShouldStartLoadWithRequest:a=>new URLSearchParams(a.url.split("?")[1]).get(ae)?(Z.openURL(a.url),!1):!0})):null};handleReactNativeWebViewMessage(e){let r=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===`${r.hostname}/send/${r.search}`)){let t=JSON.parse(e.nativeEvent.data,(n,s)=>{try{if(s&&typeof s=="object"&&s.flag&&s.flag===$)return new global[s.constructor](s.data.split(","))}catch(c){console.log("Error parsing data",c)}return s});if(t&&t.msgType&&this.messageHandlers.size){t.response=t.response??{};let n={data:t};for(let s of this.messageHandlers.values())s(n)}}}async msgPostedAfterInactivity(){let e=await b.getItem(E);if(e){let r=new Date(e).getTime(),t=new Date().getTime(),n=5*60*1e3;return t-r>n}return!1}hideOverlay(){this.container&&this.container.hideOverlay()}showOverlay(){this.container&&this.container.showOverlay()}async _post(e){if(await this.msgPostedAfterInactivity()){F.emit(G,e);return}if(this.webView&&this.webView.postMessage)this.webView.postMessage(JSON.stringify(e,(r,t)=>se(t)?{constructor:t.constructor.name,data:t.toString(),flag:$}:t)),b.setItem(E,new Date().toISOString());else throw ne()}async persistMagicEventRefreshToken(e){e.data.rt&&C(e.data.rt)}async getRT(){return D()}async getJWT(){try{console.log("CREATING DPOP");let e=await M();return console.log({dpop:e}),e}catch{return null}}checkRelayerExistsInDOM(){return Promise.resolve(!0)}reloadRelayer(){return Promise.resolve(void 0)}};import{SDKBase as le}from"@magic-sdk/provider";var A=class extends le{get Relayer(){return this.overlay.Relayer}};import g from"@react-native-async-storage/async-storage";export*from"@magic-sdk/provider";export*from"@magic-sdk/types";global.process=ge.merge(global.process,me);global.process.browser=!1;global.Buffer=R;global.URL=fe;global.URLSearchParams=ye;global.btoa=o=>R.from(o,"binary").toString("base64");global.atob=o=>R.from(o,"base64").toString("binary");var pt=pe(A,{platform:"react-native",sdkName:"@magic-sdk/react-native-bare",version:"32.2.0",bundleId:ue(),defaultEndpoint:"https://box.magic.link/",ViewController:x,configureStorage:async()=>({ready:async()=>Promise.resolve(!0),getItem:g.getItem,setItem:g.setItem,removeItem:g.removeItem,clear:g.clear,length:()=>{},key:()=>{},keys:g.getAllKeys,iterate:()=>{}})});export{pt as Magic,I as useInternetConnection};
|
|
2
2
|
//# sourceMappingURL=index.js.map
|
package/dist/es/index.js.map
CHANGED
|
@@ -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": "AAEA,MAAO,8BAEP,OAAS,aAAAA,
|
|
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/keychain.ts", "../../src/dpop/dpop.ts", "../../src/dpop/utils/uint8.ts", "../../src/dpop/utils/jwk.ts", "../../src/dpop/constants/index.ts", "../../src/dpop/utils/der.ts", "../../src/dpop/utils/dpop-alias.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 './keychain';\nimport { getDpop } from './dpop/dpop';\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 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 setRefreshTokenInKeychain(event.data.rt);\n }\n\n // Overrides parent method to retrieve refresh token from keychain while creating a request\n async getRT() {\n return getRefreshTokenInKeychain();\n }\n\n async getJWT(): Promise<string | null | undefined> {\n try {\n console.log('CREATING DPOP');\n const dpop = await getDpop();\n console.log({ dpop });\n return dpop;\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';\n\nconst SERVICE = 'magic_sdk_rt';\nconst KEY = 'magic_rt';\n\nexport const setRefreshTokenInKeychain = async (rt: string) => {\n return await Keychain.setGenericPassword(KEY, rt, {\n service: SERVICE,\n accessible: Keychain.ACCESSIBLE.AFTER_FIRST_UNLOCK_THIS_DEVICE_ONLY,\n });\n};\n\nexport const getRefreshTokenInKeychain = async () => {\n const keychainEntry = await Keychain.getGenericPassword({ service: SERVICE });\n if (!keychainEntry) return null;\n\n return keychainEntry.password;\n};\n\nexport const removeRefreshTokenInKeychain = async () => {\n return await Keychain.resetGenericPassword({ service: SERVICE });\n};\n", "import * as Biometrics from '@sbaiahmed1/react-native-biometrics';\nimport uuid from 'react-native-uuid'; // Ensure you have installed: npm install react-native-uuid\nimport { toBase64Url } from './utils/uint8';\nimport { spkiToJwk } from './utils/jwk';\nimport { ALG, TYP } from './constants';\nimport { derToRawSignature } from './utils/der';\nimport { getDpopAlias } from './utils/dpop-alias';\n\n/**\n * Generates the DPoP proof compatible with the Python backend.\n * Handles key creation (if missing), JWK construction, and signing.\n * @param httpMethod - The HTTP method (e.g., 'POST')\n * @param httpUrl - The HTTP URL being accessed\n */\nexport const getDpop = async (httpMethod?: string, httpUrl?: string): Promise<string> => {\n try {\n // 1. Configure Isolation\n const DPOP_KEY_ALIAS = await getDpopAlias();\n await Biometrics.configureKeyAlias(DPOP_KEY_ALIAS);\n\n // 2. Retrieve Public Key (Check if exists)\n const keyAttrs = await Biometrics.getKeyAttributes(DPOP_KEY_ALIAS);\n let publicKeyBase64 = '';\n\n if (!keyAttrs.exists) {\n // Create new keys in Secure Enclave (ec256)\n const keyResult = await Biometrics.createKeys(DPOP_KEY_ALIAS, 'ec256', Biometrics.BiometricStrength.Weak);\n publicKeyBase64 = keyResult.publicKey;\n } else {\n // Retrieve existing key\n const allKeys = await Biometrics.getAllKeys(DPOP_KEY_ALIAS);\n if (allKeys.keys.length > 0) {\n publicKeyBase64 = allKeys.keys[0].publicKey;\n } else {\n // Fallback: Re-create if lookup failed but attributes said it existed\n const keyResult = await Biometrics.createKeys(DPOP_KEY_ALIAS, 'ec256', Biometrics.BiometricStrength.Weak);\n publicKeyBase64 = keyResult.publicKey;\n }\n }\n\n // 3. Construct JWK\n const publicJwk = spkiToJwk(publicKeyBase64);\n\n // 4. Construct Payload\n const iat = Math.floor(Date.now() / 1000);\n const jti = uuid.v4();\n\n const payload: any = {\n iat,\n jti,\n };\n\n if (httpMethod) payload.htm = httpMethod.toUpperCase();\n if (httpUrl) payload.htu = httpUrl;\n\n // 5. Construct Header\n const header = {\n typ: TYP,\n alg: ALG,\n jwk: publicJwk,\n };\n\n // 6. Prepare Signing Input\n const headerB64 = toBase64Url(JSON.stringify(header));\n const payloadB64 = toBase64Url(JSON.stringify(payload));\n const signingInput = `${headerB64}.${payloadB64}`;\n\n // 7. Sign\n const signatureResult = await Biometrics.verifyKeySignature(DPOP_KEY_ALIAS, signingInput);\n\n if (!signatureResult.success || !signatureResult.signature) {\n throw new Error(`Signing failed: ${signatureResult.error || 'No signature returned'}`);\n }\n\n // 8. Convert Signature (Toaster expects Raw R|S)\n const signatureB64 = derToRawSignature(signatureResult.signature);\n\n // 9. Return DPoP String\n return `${signingInput}.${signatureB64}`;\n } catch (error) {\n console.error('DPoP Generation Error:', error);\n throw error;\n }\n};\n\n/**\n * Removes existing keys and invalidates the DPoP\n */\nexport const deleteDpop = async (): Promise<boolean> => {\n try {\n const DPOP_KEY_ALIAS = await getDpopAlias();\n const result = await Biometrics.deleteKeys(DPOP_KEY_ALIAS);\n return result.success;\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", "// Unique alias suffix to isolate SDK keys from the rest of the app\nexport const DPOP_KEY_ALIAS_SUFFIX = 'magiclink.dpop';\nexport 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 * React Native Biometrics returns DER; Python 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 { getDefaultKeyAlias } from '@sbaiahmed1/react-native-biometrics';\nimport { DPOP_KEY_ALIAS_SUFFIX } from '../constants';\n\nexport const getDpopAlias = async () => {\n // Each client integrating our SDK gets unique defaultAlias based on their bundle id or package name\n // We append our own suffix to the defaultAlias\n const defaultAlias = await getDefaultKeyAlias();\n const DPOP_KEY_ALIAS = `${defaultAlias}.${DPOP_KEY_ALIAS_SUFFIX}`;\n\n return DPOP_KEY_ALIAS;\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", "SERVICE", "KEY", "setRefreshTokenInKeychain", "rt", "getRefreshTokenInKeychain", "keychainEntry", "Biometrics", "uuid", "uint8ArrayToBase64", "bytes", "binary", "len", "i", "base64ToUint8Array", "base64", "binaryString", "stringToUint8Array", "str", "arr", "code", "toBase64Url", "input", "spkiToJwk", "spkiBase64", "buf", "base64ToUint8Array", "rawKey", "x", "y", "toBase64Url", "DPOP_KEY_ALIAS_SUFFIX", "ALG", "TYP", "derToRawSignature", "derBase64", "der", "base64ToUint8Array", "offset", "rLen", "rBytes", "sLen", "sBytes", "rawSignature", "toBase64Url", "getDefaultKeyAlias", "getDpopAlias", "getDefaultKeyAlias", "DPOP_KEY_ALIAS_SUFFIX", "getDpop", "httpMethod", "httpUrl", "DPOP_KEY_ALIAS", "getDpopAlias", "keyAttrs", "publicKeyBase64", "allKeys", "publicJwk", "spkiToJwk", "iat", "jti", "uuid", "payload", "headerB64", "toBase64Url", "TYP", "ALG", "payloadB64", "signingInput", "signatureResult", "signatureB64", "derToRawSignature", "error", "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", "setRefreshTokenInKeychain", "getRefreshTokenInKeychain", "dpop", "getDpop", "SDKBase", "SDKBaseReactNative", "AsyncStorage", "_", "processPolyfill", "Buffer", "URLPolyfill", "URLSearchParamsPolyfill", "str", "b64Encoded", "Magic", "createSDK", "SDKBaseReactNative", "getBundleId", "ReactNativeWebViewController"]
|
|
7
7
|
}
|
|
@@ -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
|
+
* @param httpMethod - The HTTP method (e.g., 'POST')
|
|
5
|
+
* @param httpUrl - The HTTP URL being accessed
|
|
6
|
+
*/
|
|
7
|
+
export declare const getDpop: (httpMethod?: string, httpUrl?: string) => Promise<string>;
|
|
8
|
+
/**
|
|
9
|
+
* Removes existing keys and invalidates the DPoP
|
|
10
|
+
*/
|
|
11
|
+
export declare const deleteDpop: () => Promise<boolean>;
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare const getDpopAlias: () => Promise<string>;
|
|
@@ -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;
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
import * as Keychain from 'react-native-keychain';
|
|
2
|
+
export declare const setRefreshTokenInKeychain: (rt: string) => Promise<false | Keychain.Result>;
|
|
3
|
+
export declare const getRefreshTokenInKeychain: () => Promise<string | null>;
|
|
4
|
+
export declare const removeRefreshTokenInKeychain: () => Promise<boolean>;
|
|
@@ -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": "32.2.0",
|
|
3
|
+
"version": "32.2.1-canary.964.19821266156.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",
|
|
@@ -18,9 +18,10 @@
|
|
|
18
18
|
"types": "./dist/types/index.d.ts",
|
|
19
19
|
"dependencies": {
|
|
20
20
|
"@aveq-research/localforage-asyncstorage-driver": "^3.0.1",
|
|
21
|
-
"@magic-sdk/provider": "
|
|
21
|
+
"@magic-sdk/provider": "31.2.1-canary.964.19821266156.0",
|
|
22
22
|
"@magic-sdk/types": "^25.2.0",
|
|
23
23
|
"@react-native-async-storage/async-storage": "^2.1.2",
|
|
24
|
+
"@sbaiahmed1/react-native-biometrics": "^0.9.1",
|
|
24
25
|
"@types/lodash": "^4.14.158",
|
|
25
26
|
"buffer": "~5.6.0",
|
|
26
27
|
"localforage": "^1.7.4",
|
|
@@ -28,6 +29,8 @@
|
|
|
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-keychain": "^10.0.0",
|
|
33
|
+
"react-native-uuid": "^2.0.3",
|
|
31
34
|
"regenerator-runtime": "0.13.9",
|
|
32
35
|
"tslib": "^2.0.3",
|
|
33
36
|
"whatwg-url": "~8.1.0"
|
|
@@ -52,5 +55,5 @@
|
|
|
52
55
|
"react-native-safe-area-context": ">=4.4.1",
|
|
53
56
|
"react-native-webview": ">=12.4.0"
|
|
54
57
|
},
|
|
55
|
-
"gitHead": "
|
|
58
|
+
"gitHead": "07b73fe86c8b0d4f8fb7bda7d404f396ca6d1cc6"
|
|
56
59
|
}
|