@oxyhq/services 5.16.40 → 5.16.42

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.
Files changed (267) hide show
  1. package/lib/commonjs/adapters/expo/crypto.js +56 -0
  2. package/lib/commonjs/adapters/expo/crypto.js.map +1 -0
  3. package/lib/commonjs/adapters/expo/fetch.js +30 -0
  4. package/lib/commonjs/adapters/expo/fetch.js.map +1 -0
  5. package/lib/commonjs/adapters/expo/index.js +48 -0
  6. package/lib/commonjs/adapters/expo/index.js.map +1 -0
  7. package/lib/commonjs/adapters/expo/storage.js +201 -0
  8. package/lib/commonjs/adapters/expo/storage.js.map +1 -0
  9. package/lib/commonjs/adapters/index.js +50 -0
  10. package/lib/commonjs/adapters/index.js.map +1 -0
  11. package/lib/commonjs/adapters/node/crypto.js +40 -0
  12. package/lib/commonjs/adapters/node/crypto.js.map +1 -0
  13. package/lib/commonjs/adapters/node/fetch.js +62 -0
  14. package/lib/commonjs/adapters/node/fetch.js.map +1 -0
  15. package/lib/commonjs/adapters/node/index.js +34 -0
  16. package/lib/commonjs/adapters/node/index.js.map +1 -0
  17. package/lib/commonjs/adapters/node/storage.js +163 -0
  18. package/lib/commonjs/adapters/node/storage.js.map +1 -0
  19. package/lib/commonjs/core/identity-session/DeviceManager.js +237 -0
  20. package/lib/commonjs/core/identity-session/DeviceManager.js.map +1 -0
  21. package/lib/commonjs/core/identity-session/INTEGRATION_GUIDE.md +287 -0
  22. package/lib/commonjs/core/identity-session/IdentityManager.js +400 -0
  23. package/lib/commonjs/core/identity-session/IdentityManager.js.map +1 -0
  24. package/lib/commonjs/core/identity-session/IdentitySessionCore.js +394 -0
  25. package/lib/commonjs/core/identity-session/IdentitySessionCore.js.map +1 -0
  26. package/lib/commonjs/core/identity-session/RefreshManager.js +137 -0
  27. package/lib/commonjs/core/identity-session/RefreshManager.js.map +1 -0
  28. package/lib/commonjs/core/identity-session/SessionManager.js +427 -0
  29. package/lib/commonjs/core/identity-session/SessionManager.js.map +1 -0
  30. package/lib/commonjs/core/identity-session/createIdentitySessionCore.js +24 -0
  31. package/lib/commonjs/core/identity-session/createIdentitySessionCore.js.map +1 -0
  32. package/lib/commonjs/core/identity-session/errors.js +176 -0
  33. package/lib/commonjs/core/identity-session/errors.js.map +1 -0
  34. package/lib/commonjs/core/identity-session/index.js +80 -0
  35. package/lib/commonjs/core/identity-session/index.js.map +1 -0
  36. package/lib/commonjs/core/identity-session/types.js +2 -0
  37. package/lib/commonjs/core/identity-session/types.js.map +1 -0
  38. package/lib/commonjs/core/index.js +2 -21
  39. package/lib/commonjs/core/index.js.map +1 -1
  40. package/lib/commonjs/index.js +58 -8
  41. package/lib/commonjs/index.js.map +1 -1
  42. package/lib/commonjs/models/interfaces.js +7 -0
  43. package/lib/commonjs/models/interfaces.js.map +1 -1
  44. package/lib/commonjs/ui/context/OxyContext.js +434 -820
  45. package/lib/commonjs/ui/context/OxyContext.js.map +1 -1
  46. package/lib/commonjs/ui/hooks/useAvatarPicker.js +52 -0
  47. package/lib/commonjs/ui/hooks/useAvatarPicker.js.map +1 -0
  48. package/lib/commonjs/ui/hooks/useIdentityTransfer.js +125 -0
  49. package/lib/commonjs/ui/hooks/useIdentityTransfer.js.map +1 -0
  50. package/lib/commonjs/ui/hooks/useTransferCodesPersistence.js +81 -0
  51. package/lib/commonjs/ui/hooks/useTransferCodesPersistence.js.map +1 -0
  52. package/lib/commonjs/ui/screens/AccountCenterScreen.js +7 -2
  53. package/lib/commonjs/ui/screens/AccountCenterScreen.js.map +1 -1
  54. package/lib/commonjs/ui/screens/AccountSettingsScreen.js +12 -5
  55. package/lib/commonjs/ui/screens/AccountSettingsScreen.js.map +1 -1
  56. package/lib/commonjs/ui/screens/AccountSwitcherScreen.js +2 -2
  57. package/lib/commonjs/ui/screens/AccountSwitcherScreen.js.map +1 -1
  58. package/lib/commonjs/ui/screens/ProfileScreen.js +6 -6
  59. package/lib/commonjs/ui/screens/ProfileScreen.js.map +1 -1
  60. package/lib/commonjs/ui/utils/sessionHelpers.js +7 -1
  61. package/lib/commonjs/ui/utils/sessionHelpers.js.map +1 -1
  62. package/lib/commonjs/utils/index.js +0 -7
  63. package/lib/commonjs/utils/index.js.map +1 -1
  64. package/lib/commonjs/utils/sessionUtils.js +8 -1
  65. package/lib/commonjs/utils/sessionUtils.js.map +1 -1
  66. package/lib/module/adapters/expo/crypto.js +51 -0
  67. package/lib/module/adapters/expo/crypto.js.map +1 -0
  68. package/lib/module/adapters/expo/fetch.js +26 -0
  69. package/lib/module/adapters/expo/fetch.js.map +1 -0
  70. package/lib/module/adapters/expo/index.js +45 -0
  71. package/lib/module/adapters/expo/index.js.map +1 -0
  72. package/lib/module/adapters/expo/storage.js +198 -0
  73. package/lib/module/adapters/expo/storage.js.map +1 -0
  74. package/lib/module/adapters/index.js +47 -0
  75. package/lib/module/adapters/index.js.map +1 -0
  76. package/lib/module/adapters/node/crypto.js +36 -0
  77. package/lib/module/adapters/node/crypto.js.map +1 -0
  78. package/lib/module/adapters/node/fetch.js +57 -0
  79. package/lib/module/adapters/node/fetch.js.map +1 -0
  80. package/lib/module/adapters/node/index.js +31 -0
  81. package/lib/module/adapters/node/index.js.map +1 -0
  82. package/lib/module/adapters/node/storage.js +159 -0
  83. package/lib/module/adapters/node/storage.js.map +1 -0
  84. package/lib/module/core/identity-session/DeviceManager.js +232 -0
  85. package/lib/module/core/identity-session/DeviceManager.js.map +1 -0
  86. package/lib/module/core/identity-session/INTEGRATION_GUIDE.md +287 -0
  87. package/lib/module/core/identity-session/IdentityManager.js +395 -0
  88. package/lib/module/core/identity-session/IdentityManager.js.map +1 -0
  89. package/lib/module/core/identity-session/IdentitySessionCore.js +390 -0
  90. package/lib/module/core/identity-session/IdentitySessionCore.js.map +1 -0
  91. package/lib/module/core/identity-session/RefreshManager.js +132 -0
  92. package/lib/module/core/identity-session/RefreshManager.js.map +1 -0
  93. package/lib/module/core/identity-session/SessionManager.js +422 -0
  94. package/lib/module/core/identity-session/SessionManager.js.map +1 -0
  95. package/lib/module/core/identity-session/createIdentitySessionCore.js +21 -0
  96. package/lib/module/core/identity-session/createIdentitySessionCore.js.map +1 -0
  97. package/lib/module/core/identity-session/errors.js +170 -0
  98. package/lib/module/core/identity-session/errors.js.map +1 -0
  99. package/lib/module/core/identity-session/index.js +17 -0
  100. package/lib/module/core/identity-session/index.js.map +1 -0
  101. package/lib/module/core/identity-session/types.js +2 -0
  102. package/lib/module/core/identity-session/types.js.map +1 -0
  103. package/lib/module/core/index.js +2 -3
  104. package/lib/module/core/index.js.map +1 -1
  105. package/lib/module/index.js +12 -2
  106. package/lib/module/index.js.map +1 -1
  107. package/lib/module/models/interfaces.js +7 -0
  108. package/lib/module/models/interfaces.js.map +1 -1
  109. package/lib/module/ui/context/OxyContext.js +436 -822
  110. package/lib/module/ui/context/OxyContext.js.map +1 -1
  111. package/lib/module/ui/hooks/useAvatarPicker.js +48 -0
  112. package/lib/module/ui/hooks/useAvatarPicker.js.map +1 -0
  113. package/lib/module/ui/hooks/useIdentityTransfer.js +121 -0
  114. package/lib/module/ui/hooks/useIdentityTransfer.js.map +1 -0
  115. package/lib/module/ui/hooks/useTransferCodesPersistence.js +77 -0
  116. package/lib/module/ui/hooks/useTransferCodesPersistence.js.map +1 -0
  117. package/lib/module/ui/screens/AccountCenterScreen.js +7 -2
  118. package/lib/module/ui/screens/AccountCenterScreen.js.map +1 -1
  119. package/lib/module/ui/screens/AccountSettingsScreen.js +12 -5
  120. package/lib/module/ui/screens/AccountSettingsScreen.js.map +1 -1
  121. package/lib/module/ui/screens/AccountSwitcherScreen.js +2 -2
  122. package/lib/module/ui/screens/AccountSwitcherScreen.js.map +1 -1
  123. package/lib/module/ui/screens/ProfileScreen.js +6 -6
  124. package/lib/module/ui/screens/ProfileScreen.js.map +1 -1
  125. package/lib/module/ui/utils/sessionHelpers.js +7 -1
  126. package/lib/module/ui/utils/sessionHelpers.js.map +1 -1
  127. package/lib/module/utils/index.js +2 -1
  128. package/lib/module/utils/index.js.map +1 -1
  129. package/lib/module/utils/sessionUtils.js +8 -1
  130. package/lib/module/utils/sessionUtils.js.map +1 -1
  131. package/lib/typescript/adapters/expo/crypto.d.ts +17 -0
  132. package/lib/typescript/adapters/expo/crypto.d.ts.map +1 -0
  133. package/lib/typescript/adapters/expo/fetch.d.ts +16 -0
  134. package/lib/typescript/adapters/expo/fetch.d.ts.map +1 -0
  135. package/lib/typescript/adapters/expo/index.d.ts +23 -0
  136. package/lib/typescript/adapters/expo/index.d.ts.map +1 -0
  137. package/lib/typescript/adapters/expo/storage.d.ts +23 -0
  138. package/lib/typescript/adapters/expo/storage.d.ts.map +1 -0
  139. package/lib/typescript/adapters/index.d.ts +19 -0
  140. package/lib/typescript/adapters/index.d.ts.map +1 -0
  141. package/lib/typescript/adapters/node/crypto.d.ts +17 -0
  142. package/lib/typescript/adapters/node/crypto.d.ts.map +1 -0
  143. package/lib/typescript/adapters/node/fetch.d.ts +16 -0
  144. package/lib/typescript/adapters/node/fetch.d.ts.map +1 -0
  145. package/lib/typescript/adapters/node/index.d.ts +23 -0
  146. package/lib/typescript/adapters/node/index.d.ts.map +1 -0
  147. package/lib/typescript/adapters/node/storage.d.ts +23 -0
  148. package/lib/typescript/adapters/node/storage.d.ts.map +1 -0
  149. package/lib/typescript/core/identity-session/DeviceManager.d.ts +64 -0
  150. package/lib/typescript/core/identity-session/DeviceManager.d.ts.map +1 -0
  151. package/lib/typescript/core/identity-session/IdentityManager.d.ts +88 -0
  152. package/lib/typescript/core/identity-session/IdentityManager.d.ts.map +1 -0
  153. package/lib/typescript/core/identity-session/IdentitySessionCore.d.ts +141 -0
  154. package/lib/typescript/core/identity-session/IdentitySessionCore.d.ts.map +1 -0
  155. package/lib/typescript/core/identity-session/RefreshManager.d.ts +36 -0
  156. package/lib/typescript/core/identity-session/RefreshManager.d.ts.map +1 -0
  157. package/lib/typescript/core/identity-session/SessionManager.d.ts +104 -0
  158. package/lib/typescript/core/identity-session/SessionManager.d.ts.map +1 -0
  159. package/lib/typescript/core/identity-session/createIdentitySessionCore.d.ts +11 -0
  160. package/lib/typescript/core/identity-session/createIdentitySessionCore.d.ts.map +1 -0
  161. package/lib/typescript/core/identity-session/errors.d.ts +63 -0
  162. package/lib/typescript/core/identity-session/errors.d.ts.map +1 -0
  163. package/lib/typescript/core/identity-session/index.d.ts +14 -0
  164. package/lib/typescript/core/identity-session/index.d.ts.map +1 -0
  165. package/lib/typescript/core/identity-session/types.d.ts +196 -0
  166. package/lib/typescript/core/identity-session/types.d.ts.map +1 -0
  167. package/lib/typescript/core/index.d.ts +1 -3
  168. package/lib/typescript/core/index.d.ts.map +1 -1
  169. package/lib/typescript/core/mixins/index.d.ts +2 -2
  170. package/lib/typescript/index.d.ts +3 -2
  171. package/lib/typescript/index.d.ts.map +1 -1
  172. package/lib/typescript/models/interfaces.d.ts +5 -36
  173. package/lib/typescript/models/interfaces.d.ts.map +1 -1
  174. package/lib/typescript/models/session.d.ts +3 -16
  175. package/lib/typescript/models/session.d.ts.map +1 -1
  176. package/lib/typescript/ui/context/OxyContext.d.ts +2 -25
  177. package/lib/typescript/ui/context/OxyContext.d.ts.map +1 -1
  178. package/lib/typescript/ui/hooks/mutations/useAccountMutations.d.ts +7 -8
  179. package/lib/typescript/ui/hooks/mutations/useAccountMutations.d.ts.map +1 -1
  180. package/lib/typescript/ui/hooks/mutations/useServicesMutations.d.ts +1 -1
  181. package/lib/typescript/ui/hooks/mutations/useServicesMutations.d.ts.map +1 -1
  182. package/lib/typescript/ui/hooks/queries/useAccountQueries.d.ts +5 -5
  183. package/lib/typescript/ui/hooks/queries/useAccountQueries.d.ts.map +1 -1
  184. package/lib/typescript/ui/hooks/useAvatarPicker.d.ts +18 -0
  185. package/lib/typescript/ui/hooks/useAvatarPicker.d.ts.map +1 -0
  186. package/lib/typescript/ui/hooks/useIdentityTransfer.d.ts +24 -0
  187. package/lib/typescript/ui/hooks/useIdentityTransfer.d.ts.map +1 -0
  188. package/lib/typescript/ui/hooks/useTransferCodesPersistence.d.ts +6 -0
  189. package/lib/typescript/ui/hooks/useTransferCodesPersistence.d.ts.map +1 -0
  190. package/lib/typescript/ui/screens/AccountCenterScreen.d.ts.map +1 -1
  191. package/lib/typescript/ui/screens/AccountSettingsScreen.d.ts.map +1 -1
  192. package/lib/typescript/ui/utils/sessionHelpers.d.ts +1 -0
  193. package/lib/typescript/ui/utils/sessionHelpers.d.ts.map +1 -1
  194. package/lib/typescript/utils/index.d.ts +0 -2
  195. package/lib/typescript/utils/index.d.ts.map +1 -1
  196. package/lib/typescript/utils/sessionUtils.d.ts.map +1 -1
  197. package/package.json +1 -1
  198. package/src/adapters/expo/crypto.ts +55 -0
  199. package/src/adapters/expo/fetch.ts +28 -0
  200. package/src/adapters/expo/index.ts +51 -0
  201. package/src/adapters/expo/storage.ts +228 -0
  202. package/src/adapters/index.ts +47 -0
  203. package/src/adapters/node/crypto.ts +39 -0
  204. package/src/adapters/node/fetch.ts +59 -0
  205. package/src/adapters/node/index.ts +37 -0
  206. package/src/adapters/node/storage.ts +170 -0
  207. package/src/core/identity-session/DeviceManager.ts +273 -0
  208. package/src/core/identity-session/INTEGRATION_GUIDE.md +287 -0
  209. package/src/core/identity-session/IdentityManager.ts +474 -0
  210. package/src/core/identity-session/IdentitySessionCore.ts +464 -0
  211. package/src/core/identity-session/RefreshManager.ts +189 -0
  212. package/src/core/identity-session/SessionManager.ts +500 -0
  213. package/src/core/identity-session/createIdentitySessionCore.ts +19 -0
  214. package/src/core/identity-session/errors.ts +197 -0
  215. package/src/core/identity-session/index.ts +15 -0
  216. package/src/core/identity-session/types.ts +188 -0
  217. package/src/core/index.ts +3 -4
  218. package/src/index.ts +28 -3
  219. package/src/models/interfaces.ts +12 -39
  220. package/src/models/session.ts +6 -16
  221. package/src/ui/context/OxyContext.tsx +442 -871
  222. package/src/ui/hooks/auth/index.ts +1 -0
  223. package/src/ui/hooks/useAvatarPicker.ts +62 -0
  224. package/src/ui/hooks/useIdentityTransfer.ts +135 -0
  225. package/src/ui/hooks/useTransferCodesPersistence.ts +80 -0
  226. package/src/ui/screens/AccountCenterScreen.tsx +7 -2
  227. package/src/ui/screens/AccountSettingsScreen.tsx +15 -8
  228. package/src/ui/screens/AccountSwitcherScreen.tsx +2 -2
  229. package/src/ui/screens/ProfileScreen.tsx +10 -10
  230. package/src/ui/utils/sessionHelpers.ts +7 -0
  231. package/src/utils/index.ts +1 -2
  232. package/src/utils/sessionUtils.ts +8 -0
  233. package/lib/commonjs/ui/context/hooks/useAuthOperations.js +0 -732
  234. package/lib/commonjs/ui/context/hooks/useAuthOperations.js.map +0 -1
  235. package/lib/commonjs/ui/context/hooks/useDeviceManagement.js +0 -73
  236. package/lib/commonjs/ui/context/hooks/useDeviceManagement.js.map +0 -1
  237. package/lib/commonjs/ui/hooks/useDeviceManagement.js +0 -73
  238. package/lib/commonjs/ui/hooks/useDeviceManagement.js.map +0 -1
  239. package/lib/commonjs/ui/hooks/useSessionManagement.js +0 -281
  240. package/lib/commonjs/ui/hooks/useSessionManagement.js.map +0 -1
  241. package/lib/commonjs/utils/deviceManager.js +0 -177
  242. package/lib/commonjs/utils/deviceManager.js.map +0 -1
  243. package/lib/module/ui/context/hooks/useAuthOperations.js +0 -726
  244. package/lib/module/ui/context/hooks/useAuthOperations.js.map +0 -1
  245. package/lib/module/ui/context/hooks/useDeviceManagement.js +0 -68
  246. package/lib/module/ui/context/hooks/useDeviceManagement.js.map +0 -1
  247. package/lib/module/ui/hooks/useDeviceManagement.js +0 -68
  248. package/lib/module/ui/hooks/useDeviceManagement.js.map +0 -1
  249. package/lib/module/ui/hooks/useSessionManagement.js +0 -276
  250. package/lib/module/ui/hooks/useSessionManagement.js.map +0 -1
  251. package/lib/module/utils/deviceManager.js +0 -171
  252. package/lib/module/utils/deviceManager.js.map +0 -1
  253. package/lib/typescript/ui/context/hooks/useAuthOperations.d.ts +0 -59
  254. package/lib/typescript/ui/context/hooks/useAuthOperations.d.ts.map +0 -1
  255. package/lib/typescript/ui/context/hooks/useDeviceManagement.d.ts +0 -27
  256. package/lib/typescript/ui/context/hooks/useDeviceManagement.d.ts.map +0 -1
  257. package/lib/typescript/ui/hooks/useDeviceManagement.d.ts +0 -27
  258. package/lib/typescript/ui/hooks/useDeviceManagement.d.ts.map +0 -1
  259. package/lib/typescript/ui/hooks/useSessionManagement.d.ts +0 -41
  260. package/lib/typescript/ui/hooks/useSessionManagement.d.ts.map +0 -1
  261. package/lib/typescript/utils/deviceManager.d.ts +0 -66
  262. package/lib/typescript/utils/deviceManager.d.ts.map +0 -1
  263. package/src/ui/context/hooks/useAuthOperations.ts +0 -801
  264. package/src/ui/context/hooks/useDeviceManagement.ts +0 -108
  265. package/src/ui/hooks/useDeviceManagement.ts +0 -108
  266. package/src/ui/hooks/useSessionManagement.ts +0 -401
  267. package/src/utils/deviceManager.ts +0 -198
@@ -0,0 +1,228 @@
1
+ /**
2
+ * Expo 54 Storage Adapter
3
+ *
4
+ * Handles storage for React Native (expo-secure-store + AsyncStorage) and Web (localStorage)
5
+ */
6
+
7
+ import { Platform } from 'react-native';
8
+ import type { AsyncStorageStatic } from '@react-native-async-storage/async-storage';
9
+
10
+ // Lazy imports
11
+ let SecureStore: typeof import('expo-secure-store') | null = null;
12
+ let AsyncStorage: AsyncStorageStatic | null = null;
13
+
14
+ /**
15
+ * Initialize expo-secure-store (React Native only)
16
+ */
17
+ async function initSecureStore(): Promise<typeof import('expo-secure-store')> {
18
+ if (!SecureStore) {
19
+ try {
20
+ SecureStore = await import('expo-secure-store');
21
+ } catch (error) {
22
+ const errorMessage = error instanceof Error ? error.message : String(error);
23
+ throw new Error(`Failed to load expo-secure-store: ${errorMessage}`);
24
+ }
25
+ }
26
+ return SecureStore;
27
+ }
28
+
29
+ /**
30
+ * Initialize AsyncStorage (React Native only)
31
+ */
32
+ async function initAsyncStorage(): Promise<AsyncStorageStatic> {
33
+ if (!AsyncStorage) {
34
+ try {
35
+ const module = await import('@react-native-async-storage/async-storage');
36
+ AsyncStorage = (module.default as unknown) as AsyncStorageStatic;
37
+ } catch (error) {
38
+ const errorMessage = error instanceof Error ? error.message : String(error);
39
+ throw new Error(`Failed to load AsyncStorage: ${errorMessage}`);
40
+ }
41
+ }
42
+ return AsyncStorage;
43
+ }
44
+
45
+ /**
46
+ * Check if we're on React Native (not Web)
47
+ */
48
+ function isReactNative(): boolean {
49
+ try {
50
+ return Platform.OS !== 'web';
51
+ } catch {
52
+ return typeof navigator !== 'undefined' && navigator.product === 'ReactNative';
53
+ }
54
+ }
55
+
56
+ /**
57
+ * Check if we're on Web
58
+ */
59
+ function isWeb(): boolean {
60
+ try {
61
+ return Platform.OS === 'web';
62
+ } catch {
63
+ return typeof window !== 'undefined' && typeof navigator !== 'undefined' && navigator.product !== 'ReactNative';
64
+ }
65
+ }
66
+
67
+ /**
68
+ * Storage Adapter Interface
69
+ */
70
+ export interface StorageAdapter {
71
+ // Secure storage (para claves privadas) - SOLO NATIVE
72
+ secureGet(key: string): Promise<string | null>;
73
+ secureSet(key: string, value: string): Promise<void>;
74
+ secureDelete(key: string): Promise<void>;
75
+ isSecureStorageAvailable(): boolean;
76
+
77
+ // Regular storage (para sesiones, etc.) - Native + Web
78
+ get(key: string): Promise<string | null>;
79
+ set(key: string, value: string): Promise<void>;
80
+ remove(key: string): Promise<void>;
81
+ clear(): Promise<void>;
82
+ }
83
+
84
+ /**
85
+ * Create Expo Storage Adapter
86
+ */
87
+ export async function createExpoStorageAdapter(): Promise<StorageAdapter> {
88
+ const isNative = isReactNative();
89
+ const isWebPlatform = isWeb();
90
+
91
+ if (isNative) {
92
+ // React Native: Use expo-secure-store + AsyncStorage
93
+ const secureStore = await initSecureStore();
94
+ const asyncStorage = await initAsyncStorage();
95
+
96
+ return {
97
+ // Secure storage (expo-secure-store)
98
+ async secureGet(key: string): Promise<string | null> {
99
+ try {
100
+ return await secureStore.getItemAsync(key);
101
+ } catch (error) {
102
+ console.warn('[ExpoStorage] Failed to get secure item:', error);
103
+ return null;
104
+ }
105
+ },
106
+
107
+ async secureSet(key: string, value: string): Promise<void> {
108
+ try {
109
+ await secureStore.setItemAsync(key, value, {
110
+ keychainAccessible: secureStore.WHEN_UNLOCKED_THIS_DEVICE_ONLY,
111
+ });
112
+ } catch (error) {
113
+ console.warn('[ExpoStorage] Failed to set secure item:', error);
114
+ throw error;
115
+ }
116
+ },
117
+
118
+ async secureDelete(key: string): Promise<void> {
119
+ try {
120
+ await secureStore.deleteItemAsync(key);
121
+ } catch (error) {
122
+ console.warn('[ExpoStorage] Failed to delete secure item:', error);
123
+ }
124
+ },
125
+
126
+ isSecureStorageAvailable(): boolean {
127
+ return true;
128
+ },
129
+
130
+ // Regular storage (AsyncStorage)
131
+ async get(key: string): Promise<string | null> {
132
+ try {
133
+ return await asyncStorage.getItem(key);
134
+ } catch (error) {
135
+ console.warn('[ExpoStorage] Failed to get item:', error);
136
+ return null;
137
+ }
138
+ },
139
+
140
+ async set(key: string, value: string): Promise<void> {
141
+ try {
142
+ await asyncStorage.setItem(key, value);
143
+ } catch (error) {
144
+ console.warn('[ExpoStorage] Failed to set item:', error);
145
+ throw error;
146
+ }
147
+ },
148
+
149
+ async remove(key: string): Promise<void> {
150
+ try {
151
+ await asyncStorage.removeItem(key);
152
+ } catch (error) {
153
+ console.warn('[ExpoStorage] Failed to remove item:', error);
154
+ }
155
+ },
156
+
157
+ async clear(): Promise<void> {
158
+ try {
159
+ await asyncStorage.clear();
160
+ } catch (error) {
161
+ console.warn('[ExpoStorage] Failed to clear storage:', error);
162
+ }
163
+ },
164
+ };
165
+ } else if (isWebPlatform) {
166
+ // Web: Use localStorage only (no secure storage)
167
+ if (typeof window === 'undefined' || typeof window.localStorage === 'undefined') {
168
+ throw new Error('localStorage is not available in this environment');
169
+ }
170
+
171
+ return {
172
+ // Secure storage not available on Web
173
+ async secureGet(_key: string): Promise<string | null> {
174
+ return null;
175
+ },
176
+
177
+ async secureSet(_key: string, _value: string): Promise<void> {
178
+ // No-op on Web
179
+ },
180
+
181
+ async secureDelete(_key: string): Promise<void> {
182
+ // No-op on Web
183
+ },
184
+
185
+ isSecureStorageAvailable(): boolean {
186
+ return false;
187
+ },
188
+
189
+ // Regular storage (localStorage)
190
+ async get(key: string): Promise<string | null> {
191
+ try {
192
+ return window.localStorage.getItem(key);
193
+ } catch (error) {
194
+ console.warn('[ExpoStorage] Failed to get item:', error);
195
+ return null;
196
+ }
197
+ },
198
+
199
+ async set(key: string, value: string): Promise<void> {
200
+ try {
201
+ window.localStorage.setItem(key, value);
202
+ } catch (error) {
203
+ console.warn('[ExpoStorage] Failed to set item:', error);
204
+ throw error;
205
+ }
206
+ },
207
+
208
+ async remove(key: string): Promise<void> {
209
+ try {
210
+ window.localStorage.removeItem(key);
211
+ } catch (error) {
212
+ console.warn('[ExpoStorage] Failed to remove item:', error);
213
+ }
214
+ },
215
+
216
+ async clear(): Promise<void> {
217
+ try {
218
+ window.localStorage.clear();
219
+ } catch (error) {
220
+ console.warn('[ExpoStorage] Failed to clear storage:', error);
221
+ }
222
+ },
223
+ };
224
+ } else {
225
+ throw new Error('Unknown platform for Expo storage adapter');
226
+ }
227
+ }
228
+
@@ -0,0 +1,47 @@
1
+ /**
2
+ * Platform Adapters
3
+ *
4
+ * Factory function to create the appropriate platform adapter
5
+ *
6
+ * Uses dynamic imports for Node.js adapter to avoid Metro bundling Node-specific code
7
+ * in React Native environments.
8
+ */
9
+
10
+ import { createExpoAdapter, type PlatformAdapter as ExpoPlatformAdapter } from './expo';
11
+ import type { PlatformAdapter as NodePlatformAdapter } from './node';
12
+
13
+ export type PlatformAdapter = ExpoPlatformAdapter | NodePlatformAdapter;
14
+
15
+ /**
16
+ * Detect platform and create appropriate adapter
17
+ *
18
+ * Uses dynamic imports for Node.js adapter to ensure Metro doesn't process
19
+ * Node-specific code (like 'crypto' module) in React Native bundles.
20
+ */
21
+ export async function createPlatformAdapter(): Promise<PlatformAdapter> {
22
+ // Detect if we're in Node.js environment
23
+ const isNode = typeof process !== 'undefined' && process.versions?.node != null;
24
+
25
+ // Detect if we're in React Native
26
+ const isReactNative = typeof navigator !== 'undefined' && navigator.product === 'ReactNative';
27
+
28
+ // If we're in Node.js AND not React Native, consider using Node adapter
29
+ if (isNode && !isReactNative) {
30
+ // Double-check: try to import react-native to confirm we're not in React Native
31
+ // This is a safety check for edge cases where Node.js might be available but we're in RN
32
+ try {
33
+ await import('react-native');
34
+ // If we can import react-native, we're in React Native environment
35
+ return await createExpoAdapter();
36
+ } catch {
37
+ // Cannot import react-native, we're in pure Node.js
38
+ // Use dynamic import to avoid Metro processing Node-specific code
39
+ const { createNodeAdapter } = await import('./node/index.js');
40
+ return await createNodeAdapter();
41
+ }
42
+ }
43
+
44
+ // Default: Expo adapter (React Native or Web)
45
+ return await createExpoAdapter();
46
+ }
47
+
@@ -0,0 +1,39 @@
1
+ /**
2
+ * Node.js Crypto Adapter
3
+ *
4
+ * Uses Node's built-in crypto module
5
+ */
6
+
7
+ import * as crypto from 'crypto';
8
+
9
+ /**
10
+ * Crypto Adapter Interface
11
+ */
12
+ export interface CryptoAdapter {
13
+ getRandomBytes(length: number): Promise<Uint8Array>;
14
+ digestStringAsync(algorithm: 'SHA256' | 'SHA512', data: string): Promise<string>;
15
+ }
16
+
17
+ /**
18
+ * Create Node Crypto Adapter
19
+ */
20
+ export async function createNodeCryptoAdapter(): Promise<CryptoAdapter> {
21
+ return {
22
+ /**
23
+ * Generate cryptographically secure random bytes
24
+ */
25
+ async getRandomBytes(length: number): Promise<Uint8Array> {
26
+ return new Uint8Array(crypto.randomBytes(length));
27
+ },
28
+
29
+ /**
30
+ * Hash a string using the specified algorithm
31
+ */
32
+ async digestStringAsync(algorithm: 'SHA256' | 'SHA512', data: string): Promise<string> {
33
+ const hash = crypto.createHash(algorithm.toLowerCase());
34
+ hash.update(data);
35
+ return hash.digest('hex');
36
+ },
37
+ };
38
+ }
39
+
@@ -0,0 +1,59 @@
1
+ /**
2
+ * Node.js Fetch Adapter
3
+ *
4
+ * Uses node-fetch or native fetch (Node 18+)
5
+ */
6
+
7
+ // Lazy import
8
+ let nodeFetch: { default: typeof fetch } | null = null;
9
+
10
+ /**
11
+ * Initialize node-fetch
12
+ */
13
+ async function initNodeFetch(): Promise<{ default: typeof fetch }> {
14
+ if (!nodeFetch) {
15
+ try {
16
+ // Try native fetch first (Node 18+)
17
+ if (typeof fetch !== 'undefined') {
18
+ // Use a wrapper to match the interface
19
+ nodeFetch = {
20
+ default: fetch,
21
+ };
22
+ return nodeFetch;
23
+ }
24
+
25
+ // Fallback to node-fetch (dynamic import to avoid type errors if not installed)
26
+ // @ts-ignore - node-fetch may not be installed, handled at runtime
27
+ const nodeFetchModule = await import('node-fetch');
28
+ nodeFetch = {
29
+ default: (nodeFetchModule.default || nodeFetchModule) as unknown as typeof fetch,
30
+ };
31
+ } catch (error) {
32
+ const errorMessage = error instanceof Error ? error.message : String(error);
33
+ throw new Error(`Failed to load fetch: ${errorMessage}. Please use Node.js 18+ or install node-fetch`);
34
+ }
35
+ }
36
+ return nodeFetch;
37
+ }
38
+
39
+ /**
40
+ * Fetch Adapter Interface
41
+ */
42
+ export interface FetchAdapter {
43
+ fetch(url: string, options?: RequestInit): Promise<Response>;
44
+ }
45
+
46
+ /**
47
+ * Create Node Fetch Adapter
48
+ */
49
+ export async function createNodeFetchAdapter(): Promise<FetchAdapter> {
50
+ const fetchModule = await initNodeFetch();
51
+ const fetchFn = (fetchModule as any).default || fetchModule;
52
+
53
+ return {
54
+ fetch(url: string, options?: RequestInit): Promise<Response> {
55
+ return fetchFn(url, options);
56
+ },
57
+ };
58
+ }
59
+
@@ -0,0 +1,37 @@
1
+ /**
2
+ * Node.js Platform Adapter
3
+ *
4
+ * Provides storage, crypto, and fetch adapters for Node.js
5
+ */
6
+
7
+ import { createNodeStorageAdapter, type StorageAdapter } from './storage';
8
+ import { createNodeCryptoAdapter, type CryptoAdapter } from './crypto';
9
+ import { createNodeFetchAdapter, type FetchAdapter } from './fetch';
10
+ import type { Platform as PlatformType } from '../../core/identity-session/types';
11
+
12
+ /**
13
+ * Platform Adapter Interface
14
+ */
15
+ export interface PlatformAdapter {
16
+ storage: StorageAdapter;
17
+ crypto: CryptoAdapter;
18
+ fetch: FetchAdapter;
19
+ platform: PlatformType;
20
+ }
21
+
22
+ /**
23
+ * Create Node Platform Adapter
24
+ */
25
+ export async function createNodeAdapter(): Promise<PlatformAdapter> {
26
+ const storage = await createNodeStorageAdapter();
27
+ const crypto = await createNodeCryptoAdapter();
28
+ const fetchAdapter = await createNodeFetchAdapter();
29
+
30
+ return {
31
+ storage,
32
+ crypto,
33
+ fetch: fetchAdapter,
34
+ platform: 'node',
35
+ };
36
+ }
37
+
@@ -0,0 +1,170 @@
1
+ /**
2
+ * Node.js Storage Adapter
3
+ *
4
+ * Uses filesystem for storage
5
+ */
6
+
7
+ import * as fs from 'fs/promises';
8
+ import * as path from 'path';
9
+ import * as os from 'os';
10
+
11
+ /**
12
+ * Storage Adapter Interface
13
+ */
14
+ export interface StorageAdapter {
15
+ // Secure storage (for private keys)
16
+ secureGet(key: string): Promise<string | null>;
17
+ secureSet(key: string, value: string): Promise<void>;
18
+ secureDelete(key: string): Promise<void>;
19
+ isSecureStorageAvailable(): boolean;
20
+
21
+ // Regular storage
22
+ get(key: string): Promise<string | null>;
23
+ set(key: string, value: string): Promise<void>;
24
+ remove(key: string): Promise<void>;
25
+ clear(): Promise<void>;
26
+ }
27
+
28
+ /**
29
+ * Get storage directory
30
+ */
31
+ function getStorageDir(): string {
32
+ const homeDir = os.homedir();
33
+ return path.join(homeDir, '.oxy', 'storage');
34
+ }
35
+
36
+ /**
37
+ * Get secure storage directory
38
+ */
39
+ function getSecureStorageDir(): string {
40
+ const homeDir = os.homedir();
41
+ return path.join(homeDir, '.oxy', 'secure');
42
+ }
43
+
44
+ /**
45
+ * Ensure directory exists
46
+ */
47
+ async function ensureDir(dirPath: string): Promise<void> {
48
+ try {
49
+ await fs.mkdir(dirPath, { recursive: true });
50
+ } catch (error) {
51
+ // Ignore if already exists
52
+ if ((error as NodeJS.ErrnoException).code !== 'EEXIST') {
53
+ throw error;
54
+ }
55
+ }
56
+ }
57
+
58
+ /**
59
+ * Get file path for a key
60
+ */
61
+ function getFilePath(key: string, secure: boolean = false): string {
62
+ const baseDir = secure ? getSecureStorageDir() : getStorageDir();
63
+ // Sanitize key to prevent directory traversal
64
+ const sanitizedKey = key.replace(/[^a-zA-Z0-9_-]/g, '_');
65
+ return path.join(baseDir, `${sanitizedKey}.json`);
66
+ }
67
+
68
+ /**
69
+ * Create Node Storage Adapter
70
+ */
71
+ export async function createNodeStorageAdapter(): Promise<StorageAdapter> {
72
+ // Ensure directories exist
73
+ await ensureDir(getStorageDir());
74
+ await ensureDir(getSecureStorageDir());
75
+
76
+ return {
77
+ // Secure storage
78
+ async secureGet(key: string): Promise<string | null> {
79
+ try {
80
+ const filePath = getFilePath(key, true);
81
+ const data = await fs.readFile(filePath, 'utf-8');
82
+ const parsed = JSON.parse(data);
83
+ return parsed.value || null;
84
+ } catch (error) {
85
+ if ((error as NodeJS.ErrnoException).code === 'ENOENT') {
86
+ return null;
87
+ }
88
+ console.warn('[NodeStorage] Failed to get secure item:', error);
89
+ return null;
90
+ }
91
+ },
92
+
93
+ async secureSet(key: string, value: string): Promise<void> {
94
+ try {
95
+ const filePath = getFilePath(key, true);
96
+ const data = JSON.stringify({ value, timestamp: Date.now() });
97
+ await fs.writeFile(filePath, data, { mode: 0o600 }); // Read/write for owner only
98
+ } catch (error) {
99
+ console.warn('[NodeStorage] Failed to set secure item:', error);
100
+ throw error;
101
+ }
102
+ },
103
+
104
+ async secureDelete(key: string): Promise<void> {
105
+ try {
106
+ const filePath = getFilePath(key, true);
107
+ await fs.unlink(filePath);
108
+ } catch (error) {
109
+ if ((error as NodeJS.ErrnoException).code !== 'ENOENT') {
110
+ console.warn('[NodeStorage] Failed to delete secure item:', error);
111
+ }
112
+ }
113
+ },
114
+
115
+ isSecureStorageAvailable(): boolean {
116
+ return true;
117
+ },
118
+
119
+ // Regular storage
120
+ async get(key: string): Promise<string | null> {
121
+ try {
122
+ const filePath = getFilePath(key, false);
123
+ const data = await fs.readFile(filePath, 'utf-8');
124
+ const parsed = JSON.parse(data);
125
+ return parsed.value || null;
126
+ } catch (error) {
127
+ if ((error as NodeJS.ErrnoException).code === 'ENOENT') {
128
+ return null;
129
+ }
130
+ console.warn('[NodeStorage] Failed to get item:', error);
131
+ return null;
132
+ }
133
+ },
134
+
135
+ async set(key: string, value: string): Promise<void> {
136
+ try {
137
+ const filePath = getFilePath(key, false);
138
+ const data = JSON.stringify({ value, timestamp: Date.now() });
139
+ await fs.writeFile(filePath, data);
140
+ } catch (error) {
141
+ console.warn('[NodeStorage] Failed to set item:', error);
142
+ throw error;
143
+ }
144
+ },
145
+
146
+ async remove(key: string): Promise<void> {
147
+ try {
148
+ const filePath = getFilePath(key, false);
149
+ await fs.unlink(filePath);
150
+ } catch (error) {
151
+ if ((error as NodeJS.ErrnoException).code !== 'ENOENT') {
152
+ console.warn('[NodeStorage] Failed to remove item:', error);
153
+ }
154
+ }
155
+ },
156
+
157
+ async clear(): Promise<void> {
158
+ try {
159
+ const storageDir = getStorageDir();
160
+ const files = await fs.readdir(storageDir);
161
+ await Promise.all(
162
+ files.map(file => fs.unlink(path.join(storageDir, file)))
163
+ );
164
+ } catch (error) {
165
+ console.warn('[NodeStorage] Failed to clear storage:', error);
166
+ }
167
+ },
168
+ };
169
+ }
170
+