@oxyhq/services 5.16.39 → 5.16.41

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 +41 -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 +38 -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 +13 -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 +40 -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 -704
  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 -698
  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 -773
  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,40 @@
1
+ /**
2
+ * Platform Adapters
3
+ *
4
+ * Factory function to create the appropriate platform adapter
5
+ */
6
+
7
+ import { createExpoAdapter, type PlatformAdapter as ExpoPlatformAdapter } from './expo';
8
+ import { createNodeAdapter, type PlatformAdapter as NodePlatformAdapter } from './node';
9
+
10
+ export type PlatformAdapter = ExpoPlatformAdapter | NodePlatformAdapter;
11
+
12
+ /**
13
+ * Detect platform and create appropriate adapter
14
+ */
15
+ export async function createPlatformAdapter(): Promise<PlatformAdapter> {
16
+ // Check if we're in Node.js
17
+ if (typeof process !== 'undefined' && process.versions != null && process.versions.node != null) {
18
+ // Check if we're in a React Native/Expo environment
19
+ try {
20
+ // Try to detect React Native
21
+ if (typeof navigator !== 'undefined' && navigator.product === 'ReactNative') {
22
+ return await createExpoAdapter();
23
+ }
24
+
25
+ // Check for Expo in Node (unlikely but possible)
26
+ const Platform = await import('react-native').then(m => m.Platform).catch(() => null);
27
+ if (Platform) {
28
+ return await createExpoAdapter();
29
+ }
30
+ } catch {
31
+ // Not React Native, use Node adapter
32
+ }
33
+
34
+ return await createNodeAdapter();
35
+ }
36
+
37
+ // Default to Expo adapter (React Native or Web)
38
+ return await createExpoAdapter();
39
+ }
40
+
@@ -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
+