@opendatalabs/vana-sdk 2.2.3 → 3.0.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.
Files changed (622) hide show
  1. package/README.md +76 -92
  2. package/dist/auth/errors.cjs +54 -0
  3. package/dist/auth/errors.cjs.map +1 -0
  4. package/dist/auth/errors.d.ts +26 -0
  5. package/dist/auth/errors.js +28 -0
  6. package/dist/auth/errors.js.map +1 -0
  7. package/dist/auth/pkce.cjs +100 -0
  8. package/dist/auth/pkce.cjs.map +1 -0
  9. package/dist/auth/pkce.d.ts +55 -0
  10. package/dist/auth/pkce.js +71 -0
  11. package/dist/auth/pkce.js.map +1 -0
  12. package/dist/auth/token-store.cjs +59 -0
  13. package/dist/auth/token-store.cjs.map +1 -0
  14. package/dist/auth/token-store.d.ts +61 -0
  15. package/dist/auth/token-store.js +35 -0
  16. package/dist/auth/token-store.js.map +1 -0
  17. package/dist/auth/web3-signed-builder.cjs +70 -0
  18. package/dist/auth/web3-signed-builder.cjs.map +1 -0
  19. package/dist/auth/web3-signed-builder.d.ts +47 -0
  20. package/dist/auth/web3-signed-builder.js +45 -0
  21. package/dist/auth/web3-signed-builder.js.map +1 -0
  22. package/dist/auth/web3-signed.cjs +125 -0
  23. package/dist/auth/web3-signed.cjs.map +1 -0
  24. package/dist/auth/web3-signed.d.ts +59 -0
  25. package/dist/auth/web3-signed.js +104 -0
  26. package/dist/auth/web3-signed.js.map +1 -0
  27. package/dist/chains/definitions.cjs +2 -6
  28. package/dist/chains/definitions.cjs.map +1 -1
  29. package/dist/chains/definitions.d.ts +1 -7
  30. package/dist/chains/definitions.js +2 -6
  31. package/dist/chains/definitions.js.map +1 -1
  32. package/dist/config/chains.d.ts +18 -0
  33. package/dist/config/contracts.config.cjs +7 -95
  34. package/dist/config/contracts.config.cjs.map +1 -1
  35. package/dist/config/contracts.config.d.ts +0 -54
  36. package/dist/config/contracts.config.js +6 -93
  37. package/dist/config/contracts.config.js.map +1 -1
  38. package/dist/config/default-services.cjs +0 -10
  39. package/dist/config/default-services.cjs.map +1 -1
  40. package/dist/config/default-services.d.ts +1 -20
  41. package/dist/config/default-services.js +0 -9
  42. package/dist/config/default-services.js.map +1 -1
  43. package/dist/crypto/ecies/interface.cjs +2 -0
  44. package/dist/crypto/ecies/interface.cjs.map +1 -1
  45. package/dist/crypto/ecies/interface.js +2 -0
  46. package/dist/crypto/ecies/interface.js.map +1 -1
  47. package/dist/crypto/envelope/openpgp.cjs +59 -0
  48. package/dist/crypto/envelope/openpgp.cjs.map +1 -0
  49. package/dist/crypto/envelope/openpgp.d.ts +28 -0
  50. package/dist/crypto/envelope/openpgp.js +24 -0
  51. package/dist/crypto/envelope/openpgp.js.map +1 -0
  52. package/dist/crypto/keys/derive.cjs +65 -0
  53. package/dist/crypto/keys/derive.cjs.map +1 -0
  54. package/dist/crypto/keys/derive.d.ts +45 -0
  55. package/dist/crypto/keys/derive.js +38 -0
  56. package/dist/crypto/keys/derive.js.map +1 -0
  57. package/dist/errors.cjs +10 -0
  58. package/dist/errors.cjs.map +1 -1
  59. package/dist/errors.js +10 -0
  60. package/dist/errors.js.map +1 -1
  61. package/dist/generated/abi/VanaPoolEntityImplementation.cjs +65 -0
  62. package/dist/generated/abi/VanaPoolEntityImplementation.cjs.map +1 -1
  63. package/dist/generated/abi/VanaPoolEntityImplementation.d.ts +51 -0
  64. package/dist/generated/abi/VanaPoolEntityImplementation.js +65 -0
  65. package/dist/generated/abi/VanaPoolEntityImplementation.js.map +1 -1
  66. package/dist/generated/abi/VanaPoolStakingImplementation.cjs +187 -19
  67. package/dist/generated/abi/VanaPoolStakingImplementation.cjs.map +1 -1
  68. package/dist/generated/abi/VanaPoolStakingImplementation.d.ts +144 -14
  69. package/dist/generated/abi/VanaPoolStakingImplementation.js +187 -19
  70. package/dist/generated/abi/VanaPoolStakingImplementation.js.map +1 -1
  71. package/dist/generated/abi/index.cjs +2 -37
  72. package/dist/generated/abi/index.cjs.map +1 -1
  73. package/dist/generated/abi/index.d.ts +2687 -9119
  74. package/dist/generated/abi/index.js +2 -29
  75. package/dist/generated/abi/index.js.map +1 -1
  76. package/dist/generated/addresses.cjs +5 -107
  77. package/dist/generated/addresses.cjs.map +1 -1
  78. package/dist/generated/addresses.d.ts +5 -99
  79. package/dist/generated/addresses.js +5 -105
  80. package/dist/generated/addresses.js.map +1 -1
  81. package/dist/index.browser.d.ts +23 -138
  82. package/dist/index.browser.js +32090 -112
  83. package/dist/index.browser.js.map +7 -1
  84. package/dist/index.node.cjs +32809 -157
  85. package/dist/index.node.cjs.map +7 -1
  86. package/dist/index.node.d.ts +22 -208
  87. package/dist/index.node.js +32716 -131
  88. package/dist/index.node.js.map +7 -1
  89. package/dist/protocol/data-file.cjs +56 -0
  90. package/dist/protocol/data-file.cjs.map +1 -0
  91. package/dist/protocol/data-file.d.ts +20 -0
  92. package/dist/protocol/data-file.js +30 -0
  93. package/dist/protocol/data-file.js.map +1 -0
  94. package/dist/protocol/eip712.cjs +123 -0
  95. package/dist/protocol/eip712.cjs.map +1 -0
  96. package/dist/protocol/eip712.d.ts +117 -0
  97. package/dist/protocol/eip712.js +90 -0
  98. package/dist/protocol/eip712.js.map +1 -0
  99. package/dist/protocol/gateway.cjs +226 -0
  100. package/dist/protocol/gateway.cjs.map +1 -0
  101. package/dist/protocol/gateway.d.ts +120 -0
  102. package/dist/protocol/gateway.js +202 -0
  103. package/dist/protocol/gateway.js.map +1 -0
  104. package/dist/protocol/scopes.cjs +78 -0
  105. package/dist/protocol/scopes.cjs.map +1 -0
  106. package/dist/protocol/scopes.d.ts +13 -0
  107. package/dist/protocol/scopes.js +50 -0
  108. package/dist/protocol/scopes.js.map +1 -0
  109. package/dist/{types/atomicStore.cjs → storage/default.cjs} +9 -8
  110. package/dist/storage/default.cjs.map +1 -0
  111. package/dist/storage/default.d.ts +4 -0
  112. package/dist/storage/default.js +8 -0
  113. package/dist/storage/default.js.map +1 -0
  114. package/dist/storage/index.cjs +11 -2
  115. package/dist/storage/index.cjs.map +1 -1
  116. package/dist/storage/index.d.ts +9 -0
  117. package/dist/storage/index.js +7 -1
  118. package/dist/storage/index.js.map +1 -1
  119. package/dist/storage/providers/callback-storage.cjs +1 -0
  120. package/dist/storage/providers/callback-storage.cjs.map +1 -1
  121. package/dist/storage/providers/callback-storage.js +1 -0
  122. package/dist/storage/providers/callback-storage.js.map +1 -1
  123. package/dist/storage/providers/dropbox.cjs +1 -0
  124. package/dist/storage/providers/dropbox.cjs.map +1 -1
  125. package/dist/storage/providers/dropbox.js +1 -0
  126. package/dist/storage/providers/dropbox.js.map +1 -1
  127. package/dist/storage/providers/google-drive.cjs +1 -0
  128. package/dist/storage/providers/google-drive.cjs.map +1 -1
  129. package/dist/storage/providers/google-drive.js +1 -0
  130. package/dist/storage/providers/google-drive.js.map +1 -1
  131. package/dist/storage/providers/ipfs.cjs +1 -0
  132. package/dist/storage/providers/ipfs.cjs.map +1 -1
  133. package/dist/storage/providers/ipfs.js +1 -0
  134. package/dist/storage/providers/ipfs.js.map +1 -1
  135. package/dist/storage/providers/pinata.cjs +1 -0
  136. package/dist/storage/providers/pinata.cjs.map +1 -1
  137. package/dist/storage/providers/pinata.js +1 -0
  138. package/dist/storage/providers/pinata.js.map +1 -1
  139. package/dist/storage/providers/r2.cjs +376 -0
  140. package/dist/storage/providers/r2.cjs.map +1 -0
  141. package/dist/storage/providers/r2.d.ts +91 -0
  142. package/dist/storage/providers/r2.js +354 -0
  143. package/dist/storage/providers/r2.js.map +1 -0
  144. package/dist/storage/providers/vana-storage.cjs +251 -0
  145. package/dist/storage/providers/vana-storage.cjs.map +1 -0
  146. package/dist/storage/providers/vana-storage.d.ts +100 -0
  147. package/dist/storage/providers/vana-storage.js +231 -0
  148. package/dist/storage/providers/vana-storage.js.map +1 -0
  149. package/dist/types/config.cjs +0 -34
  150. package/dist/types/config.cjs.map +1 -1
  151. package/dist/types/config.d.ts +1 -607
  152. package/dist/types/config.js +0 -22
  153. package/dist/types/config.js.map +1 -1
  154. package/dist/types/contracts.cjs.map +1 -1
  155. package/dist/types/contracts.d.ts +1 -1
  156. package/dist/types/index.cjs +2 -33
  157. package/dist/types/index.cjs.map +1 -1
  158. package/dist/types/index.d.ts +2 -33
  159. package/dist/types/index.js +1 -35
  160. package/dist/types/index.js.map +1 -1
  161. package/dist/types/ps-errors.cjs +66 -0
  162. package/dist/types/ps-errors.cjs.map +1 -0
  163. package/dist/types/ps-errors.d.ts +25 -0
  164. package/dist/types/ps-errors.js +41 -0
  165. package/dist/types/ps-errors.js.map +1 -0
  166. package/dist/types.cjs.map +1 -1
  167. package/dist/types.d.ts +0 -29
  168. package/dist/types.js.map +1 -1
  169. package/package.json +7 -25
  170. package/dist/client/enhancedResponse.cjs +0 -164
  171. package/dist/client/enhancedResponse.cjs.map +0 -1
  172. package/dist/client/enhancedResponse.d.ts +0 -120
  173. package/dist/client/enhancedResponse.js +0 -138
  174. package/dist/client/enhancedResponse.js.map +0 -1
  175. package/dist/controllers/__tests__/data-consistency-integration.test.d.ts +0 -7
  176. package/dist/controllers/base.cjs +0 -116
  177. package/dist/controllers/base.cjs.map +0 -1
  178. package/dist/controllers/base.d.ts +0 -94
  179. package/dist/controllers/base.js +0 -92
  180. package/dist/controllers/base.js.map +0 -1
  181. package/dist/controllers/data.cjs +0 -2633
  182. package/dist/controllers/data.cjs.map +0 -1
  183. package/dist/controllers/data.d.ts +0 -1067
  184. package/dist/controllers/data.js +0 -2626
  185. package/dist/controllers/data.js.map +0 -1
  186. package/dist/controllers/operations.cjs +0 -430
  187. package/dist/controllers/operations.cjs.map +0 -1
  188. package/dist/controllers/operations.d.ts +0 -229
  189. package/dist/controllers/operations.js +0 -406
  190. package/dist/controllers/operations.js.map +0 -1
  191. package/dist/controllers/permissions.cjs +0 -4368
  192. package/dist/controllers/permissions.cjs.map +0 -1
  193. package/dist/controllers/permissions.d.ts +0 -1411
  194. package/dist/controllers/permissions.js +0 -4344
  195. package/dist/controllers/permissions.js.map +0 -1
  196. package/dist/controllers/protocol.cjs +0 -183
  197. package/dist/controllers/protocol.cjs.map +0 -1
  198. package/dist/controllers/protocol.d.ts +0 -138
  199. package/dist/controllers/protocol.js +0 -163
  200. package/dist/controllers/protocol.js.map +0 -1
  201. package/dist/controllers/schemas.cjs +0 -678
  202. package/dist/controllers/schemas.cjs.map +0 -1
  203. package/dist/controllers/schemas.d.ts +0 -293
  204. package/dist/controllers/schemas.js +0 -654
  205. package/dist/controllers/schemas.js.map +0 -1
  206. package/dist/controllers/server.cjs +0 -643
  207. package/dist/controllers/server.cjs.map +0 -1
  208. package/dist/controllers/server.d.ts +0 -322
  209. package/dist/controllers/server.js +0 -624
  210. package/dist/controllers/server.js.map +0 -1
  211. package/dist/core/__tests__/pollingManager.test.d.ts +0 -4
  212. package/dist/core/apiClient.cjs +0 -378
  213. package/dist/core/apiClient.cjs.map +0 -1
  214. package/dist/core/apiClient.d.ts +0 -286
  215. package/dist/core/apiClient.js +0 -359
  216. package/dist/core/apiClient.js.map +0 -1
  217. package/dist/core/generics.cjs +0 -417
  218. package/dist/core/generics.cjs.map +0 -1
  219. package/dist/core/generics.d.ts +0 -205
  220. package/dist/core/generics.js +0 -386
  221. package/dist/core/generics.js.map +0 -1
  222. package/dist/core/health.cjs +0 -289
  223. package/dist/core/health.cjs.map +0 -1
  224. package/dist/core/health.d.ts +0 -143
  225. package/dist/core/health.js +0 -265
  226. package/dist/core/health.js.map +0 -1
  227. package/dist/core/inMemoryNonceManager.cjs +0 -138
  228. package/dist/core/inMemoryNonceManager.cjs.map +0 -1
  229. package/dist/core/inMemoryNonceManager.d.ts +0 -69
  230. package/dist/core/inMemoryNonceManager.js +0 -114
  231. package/dist/core/inMemoryNonceManager.js.map +0 -1
  232. package/dist/core/nonceManager.cjs +0 -304
  233. package/dist/core/nonceManager.cjs.map +0 -1
  234. package/dist/core/nonceManager.d.ts +0 -116
  235. package/dist/core/nonceManager.js +0 -280
  236. package/dist/core/nonceManager.js.map +0 -1
  237. package/dist/core/pollingManager.cjs +0 -292
  238. package/dist/core/pollingManager.cjs.map +0 -1
  239. package/dist/core/pollingManager.d.ts +0 -120
  240. package/dist/core/pollingManager.js +0 -268
  241. package/dist/core/pollingManager.js.map +0 -1
  242. package/dist/core.cjs +0 -777
  243. package/dist/core.cjs.map +0 -1
  244. package/dist/core.d.ts +0 -493
  245. package/dist/core.js +0 -752
  246. package/dist/core.js.map +0 -1
  247. package/dist/diagnostics.cjs +0 -37
  248. package/dist/diagnostics.cjs.map +0 -1
  249. package/dist/diagnostics.d.ts +0 -24
  250. package/dist/diagnostics.js +0 -13
  251. package/dist/diagnostics.js.map +0 -1
  252. package/dist/diagnostics.test.d.ts +0 -1
  253. package/dist/generated/abi/DLPPerformanceImplementation.cjs +0 -1202
  254. package/dist/generated/abi/DLPPerformanceImplementation.cjs.map +0 -1
  255. package/dist/generated/abi/DLPPerformanceImplementation.d.ts +0 -914
  256. package/dist/generated/abi/DLPPerformanceImplementation.js +0 -1178
  257. package/dist/generated/abi/DLPPerformanceImplementation.js.map +0 -1
  258. package/dist/generated/abi/DLPRewardDeployerImplementation.cjs +0 -1112
  259. package/dist/generated/abi/DLPRewardDeployerImplementation.cjs.map +0 -1
  260. package/dist/generated/abi/DLPRewardDeployerImplementation.d.ts +0 -840
  261. package/dist/generated/abi/DLPRewardDeployerImplementation.js +0 -1088
  262. package/dist/generated/abi/DLPRewardDeployerImplementation.js.map +0 -1
  263. package/dist/generated/abi/DLPRewardDeployerTreasuryImplementation.cjs +0 -612
  264. package/dist/generated/abi/DLPRewardDeployerTreasuryImplementation.cjs.map +0 -1
  265. package/dist/generated/abi/DLPRewardDeployerTreasuryImplementation.d.ts +0 -451
  266. package/dist/generated/abi/DLPRewardDeployerTreasuryImplementation.js +0 -588
  267. package/dist/generated/abi/DLPRewardDeployerTreasuryImplementation.js.map +0 -1
  268. package/dist/generated/abi/DLPRewardSwapImplementation.cjs +0 -939
  269. package/dist/generated/abi/DLPRewardSwapImplementation.cjs.map +0 -1
  270. package/dist/generated/abi/DLPRewardSwapImplementation.d.ts +0 -705
  271. package/dist/generated/abi/DLPRewardSwapImplementation.js +0 -915
  272. package/dist/generated/abi/DLPRewardSwapImplementation.js.map +0 -1
  273. package/dist/generated/abi/DLPRootImplementation.cjs +0 -1644
  274. package/dist/generated/abi/DLPRootImplementation.cjs.map +0 -1
  275. package/dist/generated/abi/DLPRootImplementation.d.ts +0 -1246
  276. package/dist/generated/abi/DLPRootImplementation.js +0 -1620
  277. package/dist/generated/abi/DLPRootImplementation.js.map +0 -1
  278. package/dist/generated/abi/DataLiquidityPoolImplementation.cjs +0 -985
  279. package/dist/generated/abi/DataLiquidityPoolImplementation.cjs.map +0 -1
  280. package/dist/generated/abi/DataLiquidityPoolImplementation.d.ts +0 -735
  281. package/dist/generated/abi/DataLiquidityPoolImplementation.js +0 -961
  282. package/dist/generated/abi/DataLiquidityPoolImplementation.js.map +0 -1
  283. package/dist/generated/abi/SwapHelperImplementation.cjs +0 -976
  284. package/dist/generated/abi/SwapHelperImplementation.cjs.map +0 -1
  285. package/dist/generated/abi/SwapHelperImplementation.d.ts +0 -728
  286. package/dist/generated/abi/SwapHelperImplementation.js +0 -952
  287. package/dist/generated/abi/SwapHelperImplementation.js.map +0 -1
  288. package/dist/generated/abi/TeePoolImplementation.cjs +0 -1313
  289. package/dist/generated/abi/TeePoolImplementation.cjs.map +0 -1
  290. package/dist/generated/abi/TeePoolImplementation.d.ts +0 -992
  291. package/dist/generated/abi/TeePoolImplementation.js +0 -1289
  292. package/dist/generated/abi/TeePoolImplementation.js.map +0 -1
  293. package/dist/generated/event-types.cjs +0 -17
  294. package/dist/generated/event-types.cjs.map +0 -1
  295. package/dist/generated/event-types.d.ts +0 -809
  296. package/dist/generated/event-types.js +0 -1
  297. package/dist/generated/event-types.js.map +0 -1
  298. package/dist/generated/eventRegistry.cjs +0 -4470
  299. package/dist/generated/eventRegistry.cjs.map +0 -1
  300. package/dist/generated/eventRegistry.d.ts +0 -14
  301. package/dist/generated/eventRegistry.js +0 -4445
  302. package/dist/generated/eventRegistry.js.map +0 -1
  303. package/dist/generated/server/server-exports.cjs +0 -45
  304. package/dist/generated/server/server-exports.cjs.map +0 -1
  305. package/dist/generated/server/server-exports.d.ts +0 -36
  306. package/dist/generated/server/server-exports.js +0 -19
  307. package/dist/generated/server/server-exports.js.map +0 -1
  308. package/dist/generated/server/server.cjs +0 -17
  309. package/dist/generated/server/server.cjs.map +0 -1
  310. package/dist/generated/server/server.d.ts +0 -907
  311. package/dist/generated/server/server.js +0 -1
  312. package/dist/generated/server/server.js.map +0 -1
  313. package/dist/generated/subgraph.cjs +0 -1440
  314. package/dist/generated/subgraph.cjs.map +0 -1
  315. package/dist/generated/subgraph.d.ts +0 -6113
  316. package/dist/generated/subgraph.js +0 -1404
  317. package/dist/generated/subgraph.js.map +0 -1
  318. package/dist/lib/__tests__/redisAtomicStore.test.d.ts +0 -1
  319. package/dist/lib/redisAtomicStore.cjs +0 -201
  320. package/dist/lib/redisAtomicStore.cjs.map +0 -1
  321. package/dist/lib/redisAtomicStore.d.ts +0 -120
  322. package/dist/lib/redisAtomicStore.js +0 -177
  323. package/dist/lib/redisAtomicStore.js.map +0 -1
  324. package/dist/server/relayerHandler.cjs +0 -452
  325. package/dist/server/relayerHandler.cjs.map +0 -1
  326. package/dist/server/relayerHandler.d.ts +0 -69
  327. package/dist/server/relayerHandler.js +0 -428
  328. package/dist/server/relayerHandler.js.map +0 -1
  329. package/dist/tests/abi.test.d.ts +0 -1
  330. package/dist/tests/chains-definitions.test.d.ts +0 -1
  331. package/dist/tests/core-encryption.test.d.ts +0 -1
  332. package/dist/tests/core-extended.test.d.ts +0 -1
  333. package/dist/tests/core-generics-coverage.test.d.ts +0 -1
  334. package/dist/tests/coverage-boost.test.d.ts +0 -1
  335. package/dist/tests/crypto-cross-platform-compatibility.test.d.ts +0 -1
  336. package/dist/tests/data-addfile-permissions-schema.test.d.ts +0 -1
  337. package/dist/tests/data-additional-methods.test.d.ts +0 -1
  338. package/dist/tests/data-controller-edge-cases.test.d.ts +0 -1
  339. package/dist/tests/data-ipfs-gateways.test.d.ts +0 -1
  340. package/dist/tests/data-relayer.test.d.ts +0 -1
  341. package/dist/tests/data-schema-validation.test.d.ts +0 -1
  342. package/dist/tests/data-simple-methods.test.d.ts +0 -1
  343. package/dist/tests/data-upload-owner-validation.test.d.ts +0 -1
  344. package/dist/tests/data.test.d.ts +0 -1
  345. package/dist/tests/demo-integration.test.d.ts +0 -1
  346. package/dist/tests/demo-trusted-server-integration.test.d.ts +0 -1
  347. package/dist/tests/download-relayer.test.d.ts +0 -1
  348. package/dist/tests/dual-mode-permissions.test.d.ts +0 -1
  349. package/dist/tests/dual-mode-trusted-servers.test.d.ts +0 -1
  350. package/dist/tests/encryption-correct-implementation.test.d.ts +0 -1
  351. package/dist/tests/encryption-coverage.test.d.ts +0 -1
  352. package/dist/tests/encryption-edge-cases.test.d.ts +0 -1
  353. package/dist/tests/encryption-utils-updated.test.d.ts +0 -1
  354. package/dist/tests/errors-coverage.test.d.ts +0 -1
  355. package/dist/tests/factories/mockFactory.d.ts +0 -316
  356. package/dist/tests/fakes/FakeStorageManager.d.ts +0 -200
  357. package/dist/tests/fakes/FakeStorageManager.test.d.ts +0 -1
  358. package/dist/tests/fakes/FakeWaitForTransactionEvents.d.ts +0 -170
  359. package/dist/tests/fakes/FakeWaitForTransactionEvents.test.d.ts +0 -1
  360. package/dist/tests/fakes/fake-pgp-port.d.ts +0 -13
  361. package/dist/tests/grantValidation-edge-cases.test.d.ts +0 -1
  362. package/dist/tests/grantValidation-unreachable-branch.test.d.ts +0 -1
  363. package/dist/tests/helper-methods.test.d.ts +0 -1
  364. package/dist/tests/helpers/typedMocks.d.ts +0 -64
  365. package/dist/tests/index-browser.test.d.ts +0 -1
  366. package/dist/tests/index-node.test.d.ts +0 -1
  367. package/dist/tests/index.test.d.ts +0 -1
  368. package/dist/tests/mocks/platformAdapter.d.ts +0 -12
  369. package/dist/tests/new-permissions-methods.test.d.ts +0 -1
  370. package/dist/tests/no-buffer-browser.test.d.ts +0 -1
  371. package/dist/tests/permissions-grantee.test.d.ts +0 -1
  372. package/dist/tests/permissions-revoke-relayer.test.d.ts +0 -1
  373. package/dist/tests/permissions-schema-validation.test.d.ts +0 -1
  374. package/dist/tests/permissions-server-files.test.d.ts +0 -1
  375. package/dist/tests/permissions-transaction-options.test.d.ts +0 -1
  376. package/dist/tests/permissions-trust-servers.test.d.ts +0 -1
  377. package/dist/tests/permissions.test.d.ts +0 -1
  378. package/dist/tests/personal.test.d.ts +0 -1
  379. package/dist/tests/platform-browser.test.d.ts +0 -1
  380. package/dist/tests/platform-crypto-expanded.test.d.ts +0 -1
  381. package/dist/tests/platform-crypto.test.d.ts +0 -1
  382. package/dist/tests/platform-index.test.d.ts +0 -1
  383. package/dist/tests/platform-node.test.d.ts +0 -1
  384. package/dist/tests/platform-shared-utils.test.d.ts +0 -1
  385. package/dist/tests/platform-updated.test.d.ts +0 -1
  386. package/dist/tests/protocol-additional-methods.test.d.ts +0 -1
  387. package/dist/tests/protocol.test.d.ts +0 -1
  388. package/dist/tests/read-only-mode.test.d.ts +0 -1
  389. package/dist/tests/relayer-integration.test.d.ts +0 -1
  390. package/dist/tests/relayer-unified.test.d.ts +0 -1
  391. package/dist/tests/schemas.test.d.ts +0 -1
  392. package/dist/tests/server-relayer-handler.test.d.ts +0 -1
  393. package/dist/tests/signatureFormatter.test.d.ts +0 -1
  394. package/dist/tests/trusted-server-queries.test.d.ts +0 -1
  395. package/dist/tests/typedDataConverter.test.d.ts +0 -1
  396. package/dist/tests/types-contracts.test.d.ts +0 -1
  397. package/dist/tests/types-data.test.d.ts +0 -1
  398. package/dist/tests/types-external-apis.test.d.ts +0 -1
  399. package/dist/tests/types-generics.test.d.ts +0 -1
  400. package/dist/tests/types-permissions.test.d.ts +0 -1
  401. package/dist/tests/types-upload-params.test.d.ts +0 -1
  402. package/dist/tests/types.test.d.ts +0 -1
  403. package/dist/tests/utils-formatters.test.d.ts +0 -1
  404. package/dist/tests/utils-grantFiles-edge-cases.test.d.ts +0 -1
  405. package/dist/tests/utils-grantFiles-validation.test.d.ts +0 -1
  406. package/dist/tests/utils-grantFiles.test.d.ts +0 -1
  407. package/dist/tests/utils-grantValidation-consolidated.test.d.ts +0 -1
  408. package/dist/tests/utils-grants.test.d.ts +0 -1
  409. package/dist/tests/utils-ipfs-additional.test.d.ts +0 -1
  410. package/dist/tests/utils-ipfs.test.d.ts +0 -4
  411. package/dist/tests/utils-schemaValidation.test.d.ts +0 -1
  412. package/dist/tests/vana.test.d.ts +0 -1
  413. package/dist/tests/wallet-crypto-compatibility.test.d.ts +0 -1
  414. package/dist/types/atomicStore.cjs.map +0 -1
  415. package/dist/types/atomicStore.d.ts +0 -236
  416. package/dist/types/atomicStore.js +0 -7
  417. package/dist/types/atomicStore.js.map +0 -1
  418. package/dist/types/blockchain.cjs +0 -17
  419. package/dist/types/blockchain.cjs.map +0 -1
  420. package/dist/types/blockchain.d.ts +0 -85
  421. package/dist/types/blockchain.js +0 -1
  422. package/dist/types/blockchain.js.map +0 -1
  423. package/dist/types/controller-context.cjs +0 -17
  424. package/dist/types/controller-context.cjs.map +0 -1
  425. package/dist/types/controller-context.d.ts +0 -68
  426. package/dist/types/controller-context.js +0 -1
  427. package/dist/types/controller-context.js.map +0 -1
  428. package/dist/types/data.cjs +0 -17
  429. package/dist/types/data.cjs.map +0 -1
  430. package/dist/types/data.d.ts +0 -763
  431. package/dist/types/data.js +0 -1
  432. package/dist/types/data.js.map +0 -1
  433. package/dist/types/external-apis.cjs +0 -61
  434. package/dist/types/external-apis.cjs.map +0 -1
  435. package/dist/types/external-apis.d.ts +0 -184
  436. package/dist/types/external-apis.js +0 -34
  437. package/dist/types/external-apis.js.map +0 -1
  438. package/dist/types/generics.cjs +0 -17
  439. package/dist/types/generics.cjs.map +0 -1
  440. package/dist/types/generics.d.ts +0 -518
  441. package/dist/types/generics.js +0 -1
  442. package/dist/types/generics.js.map +0 -1
  443. package/dist/types/operationStore.cjs +0 -17
  444. package/dist/types/operationStore.cjs.map +0 -1
  445. package/dist/types/operationStore.d.ts +0 -171
  446. package/dist/types/operationStore.js +0 -1
  447. package/dist/types/operationStore.js.map +0 -1
  448. package/dist/types/operations.cjs +0 -53
  449. package/dist/types/operations.cjs.map +0 -1
  450. package/dist/types/operations.d.ts +0 -204
  451. package/dist/types/operations.js +0 -26
  452. package/dist/types/operations.js.map +0 -1
  453. package/dist/types/options.cjs +0 -17
  454. package/dist/types/options.cjs.map +0 -1
  455. package/dist/types/options.d.ts +0 -308
  456. package/dist/types/options.js +0 -1
  457. package/dist/types/options.js.map +0 -1
  458. package/dist/types/permissions.cjs +0 -17
  459. package/dist/types/permissions.cjs.map +0 -1
  460. package/dist/types/permissions.d.ts +0 -955
  461. package/dist/types/permissions.js +0 -1
  462. package/dist/types/permissions.js.map +0 -1
  463. package/dist/types/personal.cjs +0 -17
  464. package/dist/types/personal.cjs.map +0 -1
  465. package/dist/types/personal.d.ts +0 -174
  466. package/dist/types/personal.js +0 -1
  467. package/dist/types/personal.js.map +0 -1
  468. package/dist/types/relayer.cjs +0 -17
  469. package/dist/types/relayer.cjs.map +0 -1
  470. package/dist/types/relayer.d.ts +0 -552
  471. package/dist/types/relayer.js +0 -1
  472. package/dist/types/relayer.js.map +0 -1
  473. package/dist/types/transactionResults.cjs +0 -17
  474. package/dist/types/transactionResults.cjs.map +0 -1
  475. package/dist/types/transactionResults.d.ts +0 -193
  476. package/dist/types/transactionResults.js +0 -1
  477. package/dist/types/transactionResults.js.map +0 -1
  478. package/dist/types/utils.cjs +0 -17
  479. package/dist/types/utils.cjs.map +0 -1
  480. package/dist/types/utils.d.ts +0 -771
  481. package/dist/types/utils.js +0 -1
  482. package/dist/types/utils.js.map +0 -1
  483. package/dist/utils/__tests__/chainQuery.test.d.ts +0 -1
  484. package/dist/utils/__tests__/parseTransaction.test.d.ts +0 -1
  485. package/dist/utils/__tests__/pojo-serialization.test.d.ts +0 -1
  486. package/dist/utils/__tests__/signatureCache.test.d.ts +0 -1
  487. package/dist/utils/__tests__/subgraphConsistency.test.d.ts +0 -4
  488. package/dist/utils/__tests__/subgraphPagination.test.d.ts +0 -4
  489. package/dist/utils/__tests__/transaction-edge-cases.test.d.ts +0 -1
  490. package/dist/utils/__tests__/transactionHelpers.test.d.ts +0 -1
  491. package/dist/utils/__tests__/urlResolver.test.d.ts +0 -4
  492. package/dist/utils/blockchain/registry.cjs +0 -81
  493. package/dist/utils/blockchain/registry.cjs.map +0 -1
  494. package/dist/utils/blockchain/registry.d.ts +0 -32
  495. package/dist/utils/blockchain/registry.js +0 -56
  496. package/dist/utils/blockchain/registry.js.map +0 -1
  497. package/dist/utils/blockchain/registry.test.d.ts +0 -1
  498. package/dist/utils/chainQuery.cjs +0 -107
  499. package/dist/utils/chainQuery.cjs.map +0 -1
  500. package/dist/utils/chainQuery.d.ts +0 -31
  501. package/dist/utils/chainQuery.js +0 -82
  502. package/dist/utils/chainQuery.js.map +0 -1
  503. package/dist/utils/download.cjs +0 -69
  504. package/dist/utils/download.cjs.map +0 -1
  505. package/dist/utils/download.d.ts +0 -40
  506. package/dist/utils/download.js +0 -45
  507. package/dist/utils/download.js.map +0 -1
  508. package/dist/utils/encryption.cjs +0 -176
  509. package/dist/utils/encryption.cjs.map +0 -1
  510. package/dist/utils/encryption.d.ts +0 -271
  511. package/dist/utils/encryption.js +0 -142
  512. package/dist/utils/encryption.js.map +0 -1
  513. package/dist/utils/formatters.cjs +0 -55
  514. package/dist/utils/formatters.cjs.map +0 -1
  515. package/dist/utils/formatters.d.ts +0 -118
  516. package/dist/utils/formatters.js +0 -28
  517. package/dist/utils/formatters.js.map +0 -1
  518. package/dist/utils/grantFiles.cjs +0 -181
  519. package/dist/utils/grantFiles.cjs.map +0 -1
  520. package/dist/utils/grantFiles.d.ts +0 -172
  521. package/dist/utils/grantFiles.js +0 -143
  522. package/dist/utils/grantFiles.js.map +0 -1
  523. package/dist/utils/grantValidation.cjs +0 -243
  524. package/dist/utils/grantValidation.cjs.map +0 -1
  525. package/dist/utils/grantValidation.d.ts +0 -226
  526. package/dist/utils/grantValidation.js +0 -201
  527. package/dist/utils/grantValidation.js.map +0 -1
  528. package/dist/utils/grants.cjs +0 -108
  529. package/dist/utils/grants.cjs.map +0 -1
  530. package/dist/utils/grants.d.ts +0 -148
  531. package/dist/utils/grants.js +0 -82
  532. package/dist/utils/grants.js.map +0 -1
  533. package/dist/utils/ipfs.cjs +0 -128
  534. package/dist/utils/ipfs.cjs.map +0 -1
  535. package/dist/utils/ipfs.d.ts +0 -88
  536. package/dist/utils/ipfs.js +0 -97
  537. package/dist/utils/ipfs.js.map +0 -1
  538. package/dist/utils/multicall.cjs +0 -233
  539. package/dist/utils/multicall.cjs.map +0 -1
  540. package/dist/utils/multicall.d.ts +0 -126
  541. package/dist/utils/multicall.js +0 -208
  542. package/dist/utils/multicall.js.map +0 -1
  543. package/dist/utils/parseTransactionPojo.cjs +0 -87
  544. package/dist/utils/parseTransactionPojo.cjs.map +0 -1
  545. package/dist/utils/parseTransactionPojo.d.ts +0 -31
  546. package/dist/utils/parseTransactionPojo.js +0 -63
  547. package/dist/utils/parseTransactionPojo.js.map +0 -1
  548. package/dist/utils/schemaValidation.cjs +0 -258
  549. package/dist/utils/schemaValidation.cjs.map +0 -1
  550. package/dist/utils/schemaValidation.d.ts +0 -168
  551. package/dist/utils/schemaValidation.js +0 -219
  552. package/dist/utils/schemaValidation.js.map +0 -1
  553. package/dist/utils/signatureCache.cjs +0 -192
  554. package/dist/utils/signatureCache.cjs.map +0 -1
  555. package/dist/utils/signatureCache.d.ts +0 -172
  556. package/dist/utils/signatureCache.js +0 -167
  557. package/dist/utils/signatureCache.js.map +0 -1
  558. package/dist/utils/signatureFormatter.cjs +0 -42
  559. package/dist/utils/signatureFormatter.cjs.map +0 -1
  560. package/dist/utils/signatureFormatter.d.ts +0 -36
  561. package/dist/utils/signatureFormatter.js +0 -18
  562. package/dist/utils/signatureFormatter.js.map +0 -1
  563. package/dist/utils/subgraphConsistency.cjs +0 -184
  564. package/dist/utils/subgraphConsistency.cjs.map +0 -1
  565. package/dist/utils/subgraphConsistency.d.ts +0 -65
  566. package/dist/utils/subgraphConsistency.js +0 -155
  567. package/dist/utils/subgraphConsistency.js.map +0 -1
  568. package/dist/utils/subgraphMetaCache.cjs +0 -101
  569. package/dist/utils/subgraphMetaCache.cjs.map +0 -1
  570. package/dist/utils/subgraphMetaCache.d.ts +0 -56
  571. package/dist/utils/subgraphMetaCache.js +0 -76
  572. package/dist/utils/subgraphMetaCache.js.map +0 -1
  573. package/dist/utils/subgraphPagination.cjs +0 -104
  574. package/dist/utils/subgraphPagination.cjs.map +0 -1
  575. package/dist/utils/subgraphPagination.d.ts +0 -78
  576. package/dist/utils/subgraphPagination.js +0 -78
  577. package/dist/utils/subgraphPagination.js.map +0 -1
  578. package/dist/utils/tests/multicall.test.d.ts +0 -1
  579. package/dist/utils/transactionHelpers.cjs +0 -54
  580. package/dist/utils/transactionHelpers.cjs.map +0 -1
  581. package/dist/utils/transactionHelpers.d.ts +0 -80
  582. package/dist/utils/transactionHelpers.js +0 -29
  583. package/dist/utils/transactionHelpers.js.map +0 -1
  584. package/dist/utils/typeGuards.cjs +0 -109
  585. package/dist/utils/typeGuards.cjs.map +0 -1
  586. package/dist/utils/typeGuards.d.ts +0 -138
  587. package/dist/utils/typeGuards.js +0 -74
  588. package/dist/utils/typeGuards.js.map +0 -1
  589. package/dist/utils/typedDataConverter.cjs +0 -43
  590. package/dist/utils/typedDataConverter.cjs.map +0 -1
  591. package/dist/utils/typedDataConverter.d.ts +0 -46
  592. package/dist/utils/typedDataConverter.js +0 -19
  593. package/dist/utils/typedDataConverter.js.map +0 -1
  594. package/dist/utils/urlResolver.cjs +0 -62
  595. package/dist/utils/urlResolver.cjs.map +0 -1
  596. package/dist/utils/urlResolver.d.ts +0 -56
  597. package/dist/utils/urlResolver.js +0 -37
  598. package/dist/utils/urlResolver.js.map +0 -1
  599. package/dist/utils/wallet.cjs +0 -63
  600. package/dist/utils/wallet.cjs.map +0 -1
  601. package/dist/utils/wallet.d.ts +0 -94
  602. package/dist/utils/wallet.js +0 -37
  603. package/dist/utils/wallet.js.map +0 -1
  604. package/dist/utils/withEvents.cjs +0 -44
  605. package/dist/utils/withEvents.cjs.map +0 -1
  606. package/dist/utils/withEvents.d.ts +0 -56
  607. package/dist/utils/withEvents.js +0 -18
  608. package/dist/utils/withEvents.js.map +0 -1
  609. /package/dist/{__tests__/waitForTransactionEvents.test.d.ts → auth/pkce.test.d.ts} +0 -0
  610. /package/dist/{client/__tests__/enhancedResponse.test.d.ts → auth/token-store.test.d.ts} +0 -0
  611. /package/dist/{controllers/__tests__/operations.processQueue.test.d.ts → auth/web3-signed.test.d.ts} +0 -0
  612. /package/dist/{controllers/__tests__/schemas-edge-cases.test.d.ts → crypto/envelope/openpgp.test.d.ts} +0 -0
  613. /package/dist/{controllers/data-error-handling.test.d.ts → crypto/keys/derive.test.d.ts} +0 -0
  614. /package/dist/{tests/errors.test.d.ts → errors.test.d.ts} +0 -0
  615. /package/dist/{controllers/server-additional.test.d.ts → protocol/data-file.test.d.ts} +0 -0
  616. /package/dist/{core/__tests__/health.test.d.ts → protocol/eip712.test.d.ts} +0 -0
  617. /package/dist/{core/__tests__/inMemoryNonceManager.test.d.ts → protocol/gateway.test.d.ts} +0 -0
  618. /package/dist/{core/__tests__/nonceManager.test.d.ts → protocol/scopes.test.d.ts} +0 -0
  619. /package/dist/{core/core.test.d.ts → storage/tests/defaultStorage.test.d.ts} +0 -0
  620. /package/dist/{core/tests/apiClient.test.d.ts → storage/tests/r2Storage.test.d.ts} +0 -0
  621. /package/dist/{core/tests/client.test.d.ts → storage/tests/vanaStorage.test.d.ts} +0 -0
  622. /package/dist/{core/tests/generics.test.d.ts → types/ps-errors.test.d.ts} +0 -0
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/storage/providers/dropbox.ts"],"sourcesContent":["/**\n * Dropbox Storage Provider for Vana SDK\n *\n * Implements the storage interface for Dropbox using its API.\n */\n\nimport {\n StorageError,\n type StorageProvider,\n type StorageUploadResult,\n type StorageFile,\n type StorageListOptions,\n type StorageProviderConfig,\n} from \"../index\";\n\nexport interface DropboxConfig {\n /** OAuth2 access token */\n accessToken: string;\n /** Optional refresh token for token renewal */\n refreshToken?: string;\n /** OAuth2 client ID */\n clientId?: string;\n /** OAuth2 client secret */\n clientSecret?: string;\n /** Root path for uploads (defaults to '/Vana Data') */\n rootPath?: string;\n}\n\ninterface DropboxUploadResponse {\n name: string;\n path_lower: string;\n path_display: string;\n id: string;\n size: number;\n}\n\ninterface DropboxSharedLinkResponse {\n url: string;\n}\n\ninterface DropboxListResponse {\n entries: Array<{\n \".tag\": \"file\" | \"folder\";\n name: string;\n path_lower: string;\n id: string;\n server_modified: string;\n size: number;\n }>;\n has_more: boolean;\n cursor: string;\n}\n\n/**\n * Dropbox Storage Provider\n *\n * @remarks\n * Implements the storage interface for Dropbox. Requires OAuth2 authentication.\n *\n * @category Storage\n */\nexport class DropboxStorage implements StorageProvider {\n private readonly apiUrl = \"https://api.dropboxapi.com/2\";\n private readonly contentUrl = \"https://content.dropboxapi.com/2\";\n private readonly rootPath: string;\n\n constructor(private config: DropboxConfig) {\n if (!config.accessToken) {\n throw new StorageError(\n \"Dropbox access token is required\",\n \"MISSING_ACCESS_TOKEN\",\n \"dropbox\",\n );\n }\n this.rootPath = config.rootPath ?? \"/Vana Data\";\n }\n\n async upload(file: Blob, filename?: string): Promise<StorageUploadResult> {\n try {\n const fileName = filename ?? `vana-file-${Date.now()}.dat`;\n const path = `${this.rootPath}/${fileName}`;\n\n const response = await fetch(`${this.contentUrl}/files/upload`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/octet-stream\",\n \"Dropbox-API-Arg\": JSON.stringify({\n path,\n mode: \"add\",\n autorename: true,\n mute: false,\n }),\n },\n body: file,\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to upload to Dropbox: ${error}`,\n \"UPLOAD_FAILED\",\n \"dropbox\",\n );\n }\n\n const result = (await response.json()) as DropboxUploadResponse;\n const sharedLinkUrl = await this.createSharedLink(result.path_lower);\n\n // Convert the shareable URL to a direct download URL before returning.\n // This ensures the correct, raw-content URL is stored on-chain.\n const directDownloadUrl = sharedLinkUrl\n .replace(\"www.dropbox.com\", \"dl.dropboxusercontent.com\")\n .replace(\"?dl=1\", \"\");\n\n return {\n url: directDownloadUrl,\n size: file.size,\n contentType: file.type || \"application/octet-stream\",\n metadata: {\n id: result.id,\n path: result.path_display,\n },\n };\n } catch (error) {\n if (error instanceof StorageError) throw error;\n throw new StorageError(\n `Dropbox upload error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"UPLOAD_ERROR\",\n \"dropbox\",\n );\n }\n }\n\n async download(url: string): Promise<Blob> {\n // URL to force a direct download instead of showing the preview page.\n // This is done by changing the hostname from 'www.dropbox.com' to 'dl.dropboxusercontent.com'.\n const downloadUrl = url.replace(\n \"www.dropbox.com\",\n \"dl.dropboxusercontent.com\",\n );\n try {\n const response = await fetch(downloadUrl);\n\n if (!response.ok) {\n throw new StorageError(\n `Failed to download from Dropbox: ${response.statusText}`,\n \"DOWNLOAD_FAILED\",\n \"dropbox\",\n );\n }\n return response.blob();\n } catch (error) {\n if (error instanceof StorageError) throw error;\n throw new StorageError(\n `Dropbox download error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DOWNLOAD_ERROR\",\n \"dropbox\",\n );\n }\n }\n\n async list(options?: StorageListOptions): Promise<StorageFile[]> {\n if (!this.config.accessToken) {\n throw new StorageError(\n \"Access token not provided\",\n \"AUTH_ERROR\",\n \"dropbox\",\n );\n }\n\n try {\n const response = await fetch(`${this.apiUrl}/files/list_folder`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n path: this.rootPath,\n limit: options?.limit ?? 100,\n include_deleted: false,\n }),\n });\n\n if (!response.ok) {\n throw new StorageError(\n `Failed to list Dropbox files: ${await response.text()}`,\n \"LIST_FAILED\",\n \"dropbox\",\n );\n }\n\n const result = (await response.json()) as DropboxListResponse;\n\n return result.entries\n .filter((entry) => entry[\".tag\"] === \"file\")\n .map((file) => ({\n id: file.id,\n name: file.name,\n url: `dropbox://${file.path_lower}`, // Placeholder URL\n size: file.size,\n contentType: \"application/octet-stream\", // Dropbox API doesn't provide this in list\n createdAt: new Date(file.server_modified),\n }));\n } catch (error) {\n if (error instanceof StorageError) throw error;\n throw new StorageError(\n `Dropbox list error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"LIST_ERROR\",\n \"dropbox\",\n );\n }\n }\n\n async delete(url: string): Promise<boolean> {\n try {\n const path = new URL(url).pathname; // Assuming a direct URL format that includes the path\n const response = await fetch(`${this.apiUrl}/files/delete_v2`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({ path }),\n });\n\n if (!response.ok && response.status !== 404) {\n const error = await response.text();\n throw new StorageError(\n `Failed to delete from Dropbox: ${error}`,\n \"DELETE_FAILED\",\n \"dropbox\",\n );\n }\n\n return true;\n } catch (error) {\n if (error instanceof StorageError) throw error;\n throw new StorageError(\n `Dropbox delete error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DELETE_ERROR\",\n \"dropbox\",\n );\n }\n }\n\n getConfig(): StorageProviderConfig {\n return {\n name: \"Dropbox\",\n type: \"dropbox\",\n requiresAuth: true,\n features: {\n upload: true,\n download: true,\n list: true,\n delete: true,\n },\n };\n }\n\n private async createSharedLink(path: string): Promise<string> {\n const response = await fetch(\n `${this.apiUrl}/sharing/create_shared_link_with_settings`,\n {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n path,\n settings: {\n requested_visibility: \"public\",\n },\n }),\n },\n );\n\n if (!response.ok) {\n const errorData = await response.json();\n // If link already exists, Dropbox returns a 409 with the existing link\n if (\n response.status === 409 &&\n errorData.error?.shared_link_already_exists\n ) {\n return errorData.error.shared_link_already_exists.metadata.url;\n }\n throw new StorageError(\n `Failed to create shared link: ${JSON.stringify(errorData)}`,\n \"LINK_CREATION_FAILED\",\n \"dropbox\",\n );\n }\n\n const result = (await response.json()) as DropboxSharedLinkResponse;\n // Modify URL for direct download\n return result.url.replace(\"?dl=0\", \"?dl=1\");\n }\n}\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../../../src/storage/providers/dropbox.ts"],"sourcesContent":["/**\n * Dropbox Storage Provider for Vana SDK\n *\n * Implements the storage interface for Dropbox using its API.\n */\n\nimport {\n StorageError,\n type StorageProvider,\n type StorageUploadResult,\n type StorageFile,\n type StorageListOptions,\n type StorageProviderConfig,\n} from \"../index\";\n\nexport interface DropboxConfig {\n /** OAuth2 access token */\n accessToken: string;\n /** Optional refresh token for token renewal */\n refreshToken?: string;\n /** OAuth2 client ID */\n clientId?: string;\n /** OAuth2 client secret */\n clientSecret?: string;\n /** Root path for uploads (defaults to '/Vana Data') */\n rootPath?: string;\n}\n\ninterface DropboxUploadResponse {\n name: string;\n path_lower: string;\n path_display: string;\n id: string;\n size: number;\n}\n\ninterface DropboxSharedLinkResponse {\n url: string;\n}\n\ninterface DropboxListResponse {\n entries: Array<{\n \".tag\": \"file\" | \"folder\";\n name: string;\n path_lower: string;\n id: string;\n server_modified: string;\n size: number;\n }>;\n has_more: boolean;\n cursor: string;\n}\n\n/**\n * Dropbox Storage Provider\n *\n * @remarks\n * Implements the storage interface for Dropbox. Requires OAuth2 authentication.\n *\n * @category Storage\n */\nexport class DropboxStorage implements StorageProvider {\n private readonly apiUrl = \"https://api.dropboxapi.com/2\";\n private readonly contentUrl = \"https://content.dropboxapi.com/2\";\n private readonly rootPath: string;\n\n constructor(private config: DropboxConfig) {\n if (!config.accessToken) {\n throw new StorageError(\n \"Dropbox access token is required\",\n \"MISSING_ACCESS_TOKEN\",\n \"dropbox\",\n );\n }\n this.rootPath = config.rootPath ?? \"/Vana Data\";\n }\n\n async upload(file: Blob, filename?: string): Promise<StorageUploadResult> {\n try {\n const fileName = filename ?? `vana-file-${Date.now()}.dat`;\n const path = `${this.rootPath}/${fileName}`;\n\n const response = await fetch(`${this.contentUrl}/files/upload`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/octet-stream\",\n \"Dropbox-API-Arg\": JSON.stringify({\n path,\n mode: \"add\",\n autorename: true,\n mute: false,\n }),\n },\n body: file,\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to upload to Dropbox: ${error}`,\n \"UPLOAD_FAILED\",\n \"dropbox\",\n );\n }\n\n const result = (await response.json()) as DropboxUploadResponse;\n const sharedLinkUrl = await this.createSharedLink(result.path_lower);\n\n // Convert the shareable URL to a direct download URL before returning.\n // This ensures the correct, raw-content URL is stored on-chain.\n const directDownloadUrl = sharedLinkUrl\n .replace(\"www.dropbox.com\", \"dl.dropboxusercontent.com\")\n .replace(\"?dl=1\", \"\");\n\n return {\n url: directDownloadUrl,\n size: file.size,\n contentType: file.type || \"application/octet-stream\",\n metadata: {\n id: result.id,\n path: result.path_display,\n },\n };\n } catch (error) {\n if (error instanceof StorageError) throw error;\n throw new StorageError(\n `Dropbox upload error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"UPLOAD_ERROR\",\n \"dropbox\",\n );\n }\n }\n\n async download(url: string): Promise<Blob> {\n // URL to force a direct download instead of showing the preview page.\n // This is done by changing the hostname from 'www.dropbox.com' to 'dl.dropboxusercontent.com'.\n const downloadUrl = url.replace(\n \"www.dropbox.com\",\n \"dl.dropboxusercontent.com\",\n );\n try {\n const response = await fetch(downloadUrl);\n\n if (!response.ok) {\n throw new StorageError(\n `Failed to download from Dropbox: ${response.statusText}`,\n \"DOWNLOAD_FAILED\",\n \"dropbox\",\n );\n }\n return response.blob();\n } catch (error) {\n if (error instanceof StorageError) throw error;\n throw new StorageError(\n `Dropbox download error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DOWNLOAD_ERROR\",\n \"dropbox\",\n );\n }\n }\n\n async list(options?: StorageListOptions): Promise<StorageFile[]> {\n if (!this.config.accessToken) {\n throw new StorageError(\n \"Access token not provided\",\n \"AUTH_ERROR\",\n \"dropbox\",\n );\n }\n\n try {\n const response = await fetch(`${this.apiUrl}/files/list_folder`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n path: this.rootPath,\n limit: options?.limit ?? 100,\n include_deleted: false,\n }),\n });\n\n if (!response.ok) {\n throw new StorageError(\n `Failed to list Dropbox files: ${await response.text()}`,\n \"LIST_FAILED\",\n \"dropbox\",\n );\n }\n\n const result = (await response.json()) as DropboxListResponse;\n\n return result.entries\n .filter((entry) => entry[\".tag\"] === \"file\")\n .map((file) => ({\n id: file.id,\n name: file.name,\n url: `dropbox://${file.path_lower}`, // Placeholder URL\n size: file.size,\n contentType: \"application/octet-stream\", // Dropbox API doesn't provide this in list\n createdAt: new Date(file.server_modified),\n }));\n } catch (error) {\n if (error instanceof StorageError) throw error;\n throw new StorageError(\n `Dropbox list error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"LIST_ERROR\",\n \"dropbox\",\n );\n }\n }\n\n async delete(url: string): Promise<boolean> {\n try {\n const path = new URL(url).pathname; // Assuming a direct URL format that includes the path\n const response = await fetch(`${this.apiUrl}/files/delete_v2`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({ path }),\n });\n\n if (!response.ok && response.status !== 404) {\n const error = await response.text();\n throw new StorageError(\n `Failed to delete from Dropbox: ${error}`,\n \"DELETE_FAILED\",\n \"dropbox\",\n );\n }\n\n return true;\n } catch (error) {\n if (error instanceof StorageError) throw error;\n throw new StorageError(\n `Dropbox delete error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DELETE_ERROR\",\n \"dropbox\",\n );\n }\n }\n\n getConfig(): StorageProviderConfig {\n return {\n name: \"Dropbox\",\n type: \"dropbox\",\n requiresAuth: true,\n features: {\n upload: true,\n download: true,\n list: true,\n delete: true,\n },\n };\n }\n\n private async createSharedLink(path: string): Promise<string> {\n const response = await fetch(\n `${this.apiUrl}/sharing/create_shared_link_with_settings`,\n {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n path,\n settings: {\n requested_visibility: \"public\",\n },\n }),\n },\n );\n\n if (!response.ok) {\n const errorData = await response.json();\n // If link already exists, Dropbox returns a 409 with the existing link\n if (\n response.status === 409 &&\n errorData.error?.shared_link_already_exists\n ) {\n return errorData.error.shared_link_already_exists.metadata.url;\n }\n throw new StorageError(\n `Failed to create shared link: ${JSON.stringify(errorData)}`,\n \"LINK_CREATION_FAILED\",\n \"dropbox\",\n );\n }\n\n const result = (await response.json()) as DropboxSharedLinkResponse;\n // Modify URL for direct download\n return result.url.replace(\"?dl=0\", \"?dl=1\");\n }\n}\n"],"mappings":"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","names":[]}
@@ -33,6 +33,7 @@ class GoogleDriveStorage {
33
33
  );
34
34
  }
35
35
  }
36
+ config;
36
37
  baseUrl = "https://www.googleapis.com/drive/v3";
37
38
  uploadUrl = "https://www.googleapis.com/upload/drive/v3";
38
39
  async upload(file, filename) {
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/storage/providers/google-drive.ts"],"sourcesContent":["/**\n * Google Drive Storage Provider for Vana SDK\n *\n * Implements storage interface for Google Drive using OAuth2 authentication.\n * Based on patterns from dlp-ui-template with NextAuth integration.\n */\n\nimport {\n StorageError,\n type StorageProvider,\n type StorageUploadResult,\n type StorageFile,\n type StorageListOptions,\n type StorageProviderConfig,\n} from \"../index\";\n\nexport interface GoogleDriveConfig {\n /** OAuth2 access token */\n accessToken: string;\n /** Optional refresh token for token renewal */\n refreshToken?: string;\n /** OAuth2 client ID */\n clientId?: string;\n /** OAuth2 client secret */\n clientSecret?: string;\n /** Parent folder ID to upload files to */\n folderId?: string;\n}\n\ninterface GoogleDriveFile {\n id: string;\n name: string;\n webViewLink: string;\n size: string;\n mimeType: string;\n createdTime: string;\n}\n\ninterface GoogleDriveUploadResponse {\n id: string;\n name: string;\n}\n\ninterface GoogleDriveListResponse {\n files: GoogleDriveFile[];\n}\n\ninterface GoogleDriveTokenResponse {\n access_token: string;\n}\n\n/**\n * Google Drive Storage Provider with folder management capabilities\n *\n * @remarks\n * Implements storage interface for Google Drive using OAuth2 authentication.\n * Provides file upload/download operations and advanced folder management\n * including search, creation, and nested folder structures. Requires the\n * `https://www.googleapis.com/auth/drive.file` OAuth scope for full functionality.\n *\n * @category Storage\n *\n * @example\n * ```typescript\n * const googleDriveStorage = new GoogleDriveStorage({\n * accessToken: \"your-oauth-access-token\",\n * refreshToken: \"your-oauth-refresh-token\",\n * clientId: \"your-oauth-client-id\",\n * clientSecret: \"your-oauth-client-secret\",\n * });\n *\n * // Create folder structure and upload file\n * const folderId = await googleDriveStorage.findOrCreateFolder(\"screenshots\");\n * const result = await googleDriveStorage.upload(fileBlob, \"image.png\");\n * ```\n */\nexport class GoogleDriveStorage implements StorageProvider {\n private readonly baseUrl = \"https://www.googleapis.com/drive/v3\";\n private readonly uploadUrl = \"https://www.googleapis.com/upload/drive/v3\";\n\n constructor(private config: GoogleDriveConfig) {\n if (!config.accessToken) {\n throw new StorageError(\n \"Google Drive access token is required\",\n \"MISSING_ACCESS_TOKEN\",\n \"google-drive\",\n );\n }\n }\n\n async upload(file: Blob, filename?: string): Promise<StorageUploadResult> {\n try {\n const fileName = filename ?? `vana-file-${Date.now()}.dat`;\n\n // Create file metadata\n const metadata = {\n name: fileName,\n parents: this.config.folderId ? [this.config.folderId] : undefined,\n };\n\n // Create multipart upload request\n const delimiter = \"-------314159265358979323846\";\n const closeDelim = `\\r\\n--${delimiter}--`;\n\n const metadataBlob = new Blob([JSON.stringify(metadata)], {\n type: \"application/json\",\n });\n\n const multipartRequestBody = [\n `--${delimiter}`,\n \"Content-Type: application/json\",\n \"\",\n await metadataBlob.text(),\n `--${delimiter}`,\n `Content-Type: ${file.type || \"application/octet-stream\"}`,\n \"\",\n \"\",\n ].join(\"\\r\\n\");\n\n const requestBody = new Blob([multipartRequestBody, file, closeDelim]);\n\n const response = await fetch(\n `${this.uploadUrl}/files?uploadType=multipart`,\n {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": `multipart/related; boundary=\"${delimiter}\"`,\n },\n body: requestBody,\n },\n );\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to upload to Google Drive: ${error}`,\n \"UPLOAD_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveUploadResponse;\n\n // Make file publicly readable\n await this.makeFilePublic(result.id);\n\n return {\n url: `https://drive.google.com/uc?id=${result.id}&export=download`,\n size: file.size,\n contentType: file.type || \"application/octet-stream\",\n metadata: {\n id: result.id,\n name: result.name,\n driveUrl: `https://drive.google.com/file/d/${result.id}/view`,\n },\n };\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive upload error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"UPLOAD_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async download(url: string): Promise<Blob> {\n try {\n // Extract file ID from Google Drive URL\n const fileId = this.extractFileId(url);\n if (!fileId) {\n throw new StorageError(\n \"Invalid Google Drive URL format\",\n \"INVALID_URL\",\n \"google-drive\",\n );\n }\n\n const response = await fetch(\n `${this.baseUrl}/files/${fileId}?alt=media`,\n {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n },\n );\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to download from Google Drive: ${error}`,\n \"DOWNLOAD_FAILED\",\n \"google-drive\",\n );\n }\n\n const blob = await response.blob();\n\n // Check if we got HTML content instead of the actual file (authentication issue)\n if (blob.type === \"text/html\") {\n throw new StorageError(\n \"Received HTML content instead of file data. This suggests an authentication or URL formatting issue with Google Drive.\",\n \"AUTHENTICATION_ERROR\",\n \"google-drive\",\n );\n }\n\n return blob;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive download error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DOWNLOAD_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async list(options?: StorageListOptions): Promise<StorageFile[]> {\n try {\n let query = \"trashed = false\";\n\n // Add parent folder filter if configured\n if (this.config.folderId) {\n query += ` and '${this.config.folderId}' in parents`;\n }\n\n // Add name pattern filter\n if (options?.namePattern) {\n query += ` and name contains '${options.namePattern}'`;\n }\n\n const params = new URLSearchParams({\n q: query,\n fields: \"files(id,name,size,mimeType,createdTime,webViewLink)\",\n pageSize: (options?.limit ?? 100).toString(),\n });\n\n if (options?.offset && typeof options.offset === \"string\") {\n params.set(\"pageToken\", options.offset);\n }\n\n const response = await fetch(`${this.baseUrl}/files?${params}`, {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to list Google Drive files: ${error}`,\n \"LIST_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveListResponse;\n\n return result.files.map((file: GoogleDriveFile) => ({\n id: file.id,\n name: file.name,\n url: `https://drive.google.com/uc?id=${file.id}&export=download`,\n size: parseInt(file.size) || 0,\n contentType: file.mimeType,\n createdAt: new Date(file.createdTime),\n metadata: {\n id: file.id,\n driveUrl: file.webViewLink,\n },\n }));\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive list error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"LIST_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async delete(url: string): Promise<boolean> {\n try {\n // Extract file ID from Google Drive URL\n const fileId = this.extractFileId(url);\n if (!fileId) {\n throw new StorageError(\n \"Invalid Google Drive URL format\",\n \"INVALID_URL\",\n \"google-drive\",\n );\n }\n\n const response = await fetch(`${this.baseUrl}/files/${fileId}`, {\n method: \"DELETE\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok && response.status !== 404) {\n const error = await response.text();\n throw new StorageError(\n `Failed to delete from Google Drive: ${error}`,\n \"DELETE_FAILED\",\n \"google-drive\",\n );\n }\n\n return true;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive delete error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DELETE_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n getConfig(): StorageProviderConfig {\n return {\n name: \"Google Drive\",\n type: \"google-drive\",\n requiresAuth: true,\n features: {\n upload: true,\n download: true,\n list: true,\n delete: true,\n },\n };\n }\n\n /**\n * Make a Google Drive file publicly readable\n *\n * @param fileId - Google Drive file ID\n */\n private async makeFilePublic(fileId: string): Promise<void> {\n try {\n await fetch(`${this.baseUrl}/files/${fileId}/permissions`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n role: \"reader\",\n type: \"anyone\",\n }),\n });\n } catch (error) {\n // Non-critical error - file upload succeeded but sharing failed\n console.warn(\"Failed to make Google Drive file public:\", error);\n }\n }\n\n /**\n * Extract file ID from various Google Drive URL formats\n *\n * @param url - Google Drive URL\n * @returns File ID or null if not found\n */\n private extractFileId(url: string): string | null {\n // Handle various Google Drive URL formats\n const patterns = [\n /\\/file\\/d\\/([a-zA-Z0-9-_]+)/, // https://drive.google.com/file/d/FILE_ID/view\n /id=([a-zA-Z0-9-_]+)/, // https://drive.google.com/uc?id=FILE_ID\n /^([a-zA-Z0-9-_]+)$/, // Just the file ID\n ];\n\n for (const pattern of patterns) {\n const match = url.match(pattern);\n if (match) {\n return match[1];\n }\n }\n\n return null;\n }\n\n /**\n * Searches for an existing folder by name within a specified parent folder\n *\n * @remarks\n * This method queries the Google Drive API to find a folder with the exact name\n * within the specified parent directory. Only searches for folders (not files)\n * and excludes trashed items.\n *\n * @param name - The exact name of the folder to search for\n * @param parentId - The ID of the parent folder to search within (defaults to 'root')\n * @returns Promise that resolves to the folder ID if found, or `null` if not found\n * @throws {StorageError} When the Google Drive API request fails\n *\n * @example\n * ```typescript\n * // Search for a folder in the root directory\n * const folderId = await googleDriveStorage.findFolder(\"screenshots\");\n * if (folderId) {\n * console.log(\"Found folder:\", folderId);\n * } else {\n * console.log(\"Folder not found\");\n * }\n *\n * // Search for a subfolder within another folder\n * const subFolderId = await googleDriveStorage.findFolder(\"roasts\", parentFolderId);\n * ```\n */\n async findFolder(\n name: string,\n parentId: string = \"root\",\n ): Promise<string | null> {\n try {\n const query = `name='${name}' and '${parentId}' in parents and mimeType='application/vnd.google-apps.folder' and trashed = false`;\n\n const params = new URLSearchParams({\n q: query,\n fields: \"files(id,name,mimeType)\",\n });\n\n const response = await fetch(`${this.baseUrl}/files?${params}`, {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to search Google Drive folders: ${error}`,\n \"FIND_FOLDER_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveListResponse;\n\n if (result.files && result.files.length > 0) {\n return result.files[0].id;\n }\n\n return null;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive findFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"FIND_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Creates a new folder within a specified parent folder\n *\n * @remarks\n * This method creates a new folder using the Google Drive API. The folder will be\n * created with the specified name as a child of the parent folder. Requires the\n * `https://www.googleapis.com/auth/drive.file` OAuth scope.\n *\n * @param name - The name for the new folder\n * @param parentId - The ID of the parent folder where the new folder will be created (defaults to 'root')\n * @returns Promise that resolves to the ID of the newly created folder\n * @throws {StorageError} When the Google Drive API request fails or folder creation is denied\n *\n * @example\n * ```typescript\n * // Create a folder in the root directory\n * const folderId = await googleDriveStorage.createFolder(\"my-documents\");\n * console.log(\"Created folder:\", folderId);\n *\n * // Create a subfolder within another folder\n * const subFolderId = await googleDriveStorage.createFolder(\"reports\", parentFolderId);\n * ```\n */\n async createFolder(name: string, parentId: string = \"root\"): Promise<string> {\n try {\n const metadata = {\n name,\n mimeType: \"application/vnd.google-apps.folder\",\n parents: [parentId],\n };\n\n const response = await fetch(`${this.baseUrl}/files`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify(metadata),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to create Google Drive folder: ${error}`,\n \"CREATE_FOLDER_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveUploadResponse;\n return result.id;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive createFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"CREATE_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Finds an existing folder by name, or creates it if it doesn't exist\n *\n * @remarks\n * This is a convenience method that combines `findFolder` and `createFolder`.\n * It first searches for an existing folder with the specified name. If found,\n * it returns the existing folder's ID. If not found, it creates a new folder\n * and returns the new folder's ID.\n *\n * @param name - The name of the folder to find or create\n * @param parentId - The ID of the parent folder to search within or create the folder in (defaults to 'root')\n * @returns Promise that resolves to the folder ID (either existing or newly created)\n * @throws {StorageError} When the Google Drive API request fails\n *\n * @example\n * ```typescript\n * // Ensure a folder exists, creating it if necessary\n * const folderId = await googleDriveStorage.findOrCreateFolder(\"screenshots\");\n * console.log(\"Folder ID:\", folderId); // Will be same ID if folder already existed\n *\n * // Create nested folder structure\n * const parentId = await googleDriveStorage.findOrCreateFolder(\"projects\");\n * const childId = await googleDriveStorage.findOrCreateFolder(\"vana-app\", parentId);\n * ```\n */\n async findOrCreateFolder(\n name: string,\n parentId: string = \"root\",\n ): Promise<string> {\n try {\n const existingFolderId = await this.findFolder(name, parentId);\n\n if (existingFolderId) {\n return existingFolderId;\n }\n\n return await this.createFolder(name, parentId);\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive findOrCreateFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"FIND_OR_CREATE_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Refresh the access token using refresh token\n *\n * @returns Promise with new access token\n */\n async refreshAccessToken(): Promise<string> {\n if (\n !this.config.refreshToken ||\n !this.config.clientId ||\n !this.config.clientSecret\n ) {\n throw new StorageError(\n \"Refresh token, client ID, and client secret are required for token refresh\",\n \"MISSING_REFRESH_CONFIG\",\n \"google-drive\",\n );\n }\n\n try {\n const response = await fetch(\"https://oauth2.googleapis.com/token\", {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\",\n },\n body: new URLSearchParams({\n client_id: this.config.clientId,\n client_secret: this.config.clientSecret,\n refresh_token: this.config.refreshToken,\n grant_type: \"refresh_token\",\n }),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to refresh Google Drive token: ${error}`,\n \"TOKEN_REFRESH_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveTokenResponse;\n this.config.accessToken = result.access_token;\n\n return result.access_token;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive token refresh error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"TOKEN_REFRESH_ERROR\",\n \"google-drive\",\n );\n }\n }\n}\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../../../src/storage/providers/google-drive.ts"],"sourcesContent":["/**\n * Google Drive Storage Provider for Vana SDK\n *\n * Implements storage interface for Google Drive using OAuth2 authentication.\n * Based on patterns from dlp-ui-template with NextAuth integration.\n */\n\nimport {\n StorageError,\n type StorageProvider,\n type StorageUploadResult,\n type StorageFile,\n type StorageListOptions,\n type StorageProviderConfig,\n} from \"../index\";\n\nexport interface GoogleDriveConfig {\n /** OAuth2 access token */\n accessToken: string;\n /** Optional refresh token for token renewal */\n refreshToken?: string;\n /** OAuth2 client ID */\n clientId?: string;\n /** OAuth2 client secret */\n clientSecret?: string;\n /** Parent folder ID to upload files to */\n folderId?: string;\n}\n\ninterface GoogleDriveFile {\n id: string;\n name: string;\n webViewLink: string;\n size: string;\n mimeType: string;\n createdTime: string;\n}\n\ninterface GoogleDriveUploadResponse {\n id: string;\n name: string;\n}\n\ninterface GoogleDriveListResponse {\n files: GoogleDriveFile[];\n}\n\ninterface GoogleDriveTokenResponse {\n access_token: string;\n}\n\n/**\n * Google Drive Storage Provider with folder management capabilities\n *\n * @remarks\n * Implements storage interface for Google Drive using OAuth2 authentication.\n * Provides file upload/download operations and advanced folder management\n * including search, creation, and nested folder structures. Requires the\n * `https://www.googleapis.com/auth/drive.file` OAuth scope for full functionality.\n *\n * @category Storage\n *\n * @example\n * ```typescript\n * const googleDriveStorage = new GoogleDriveStorage({\n * accessToken: \"your-oauth-access-token\",\n * refreshToken: \"your-oauth-refresh-token\",\n * clientId: \"your-oauth-client-id\",\n * clientSecret: \"your-oauth-client-secret\",\n * });\n *\n * // Create folder structure and upload file\n * const folderId = await googleDriveStorage.findOrCreateFolder(\"screenshots\");\n * const result = await googleDriveStorage.upload(fileBlob, \"image.png\");\n * ```\n */\nexport class GoogleDriveStorage implements StorageProvider {\n private readonly baseUrl = \"https://www.googleapis.com/drive/v3\";\n private readonly uploadUrl = \"https://www.googleapis.com/upload/drive/v3\";\n\n constructor(private config: GoogleDriveConfig) {\n if (!config.accessToken) {\n throw new StorageError(\n \"Google Drive access token is required\",\n \"MISSING_ACCESS_TOKEN\",\n \"google-drive\",\n );\n }\n }\n\n async upload(file: Blob, filename?: string): Promise<StorageUploadResult> {\n try {\n const fileName = filename ?? `vana-file-${Date.now()}.dat`;\n\n // Create file metadata\n const metadata = {\n name: fileName,\n parents: this.config.folderId ? [this.config.folderId] : undefined,\n };\n\n // Create multipart upload request\n const delimiter = \"-------314159265358979323846\";\n const closeDelim = `\\r\\n--${delimiter}--`;\n\n const metadataBlob = new Blob([JSON.stringify(metadata)], {\n type: \"application/json\",\n });\n\n const multipartRequestBody = [\n `--${delimiter}`,\n \"Content-Type: application/json\",\n \"\",\n await metadataBlob.text(),\n `--${delimiter}`,\n `Content-Type: ${file.type || \"application/octet-stream\"}`,\n \"\",\n \"\",\n ].join(\"\\r\\n\");\n\n const requestBody = new Blob([multipartRequestBody, file, closeDelim]);\n\n const response = await fetch(\n `${this.uploadUrl}/files?uploadType=multipart`,\n {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": `multipart/related; boundary=\"${delimiter}\"`,\n },\n body: requestBody,\n },\n );\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to upload to Google Drive: ${error}`,\n \"UPLOAD_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveUploadResponse;\n\n // Make file publicly readable\n await this.makeFilePublic(result.id);\n\n return {\n url: `https://drive.google.com/uc?id=${result.id}&export=download`,\n size: file.size,\n contentType: file.type || \"application/octet-stream\",\n metadata: {\n id: result.id,\n name: result.name,\n driveUrl: `https://drive.google.com/file/d/${result.id}/view`,\n },\n };\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive upload error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"UPLOAD_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async download(url: string): Promise<Blob> {\n try {\n // Extract file ID from Google Drive URL\n const fileId = this.extractFileId(url);\n if (!fileId) {\n throw new StorageError(\n \"Invalid Google Drive URL format\",\n \"INVALID_URL\",\n \"google-drive\",\n );\n }\n\n const response = await fetch(\n `${this.baseUrl}/files/${fileId}?alt=media`,\n {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n },\n );\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to download from Google Drive: ${error}`,\n \"DOWNLOAD_FAILED\",\n \"google-drive\",\n );\n }\n\n const blob = await response.blob();\n\n // Check if we got HTML content instead of the actual file (authentication issue)\n if (blob.type === \"text/html\") {\n throw new StorageError(\n \"Received HTML content instead of file data. This suggests an authentication or URL formatting issue with Google Drive.\",\n \"AUTHENTICATION_ERROR\",\n \"google-drive\",\n );\n }\n\n return blob;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive download error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DOWNLOAD_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async list(options?: StorageListOptions): Promise<StorageFile[]> {\n try {\n let query = \"trashed = false\";\n\n // Add parent folder filter if configured\n if (this.config.folderId) {\n query += ` and '${this.config.folderId}' in parents`;\n }\n\n // Add name pattern filter\n if (options?.namePattern) {\n query += ` and name contains '${options.namePattern}'`;\n }\n\n const params = new URLSearchParams({\n q: query,\n fields: \"files(id,name,size,mimeType,createdTime,webViewLink)\",\n pageSize: (options?.limit ?? 100).toString(),\n });\n\n if (options?.offset && typeof options.offset === \"string\") {\n params.set(\"pageToken\", options.offset);\n }\n\n const response = await fetch(`${this.baseUrl}/files?${params}`, {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to list Google Drive files: ${error}`,\n \"LIST_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveListResponse;\n\n return result.files.map((file: GoogleDriveFile) => ({\n id: file.id,\n name: file.name,\n url: `https://drive.google.com/uc?id=${file.id}&export=download`,\n size: parseInt(file.size) || 0,\n contentType: file.mimeType,\n createdAt: new Date(file.createdTime),\n metadata: {\n id: file.id,\n driveUrl: file.webViewLink,\n },\n }));\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive list error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"LIST_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async delete(url: string): Promise<boolean> {\n try {\n // Extract file ID from Google Drive URL\n const fileId = this.extractFileId(url);\n if (!fileId) {\n throw new StorageError(\n \"Invalid Google Drive URL format\",\n \"INVALID_URL\",\n \"google-drive\",\n );\n }\n\n const response = await fetch(`${this.baseUrl}/files/${fileId}`, {\n method: \"DELETE\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok && response.status !== 404) {\n const error = await response.text();\n throw new StorageError(\n `Failed to delete from Google Drive: ${error}`,\n \"DELETE_FAILED\",\n \"google-drive\",\n );\n }\n\n return true;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive delete error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DELETE_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n getConfig(): StorageProviderConfig {\n return {\n name: \"Google Drive\",\n type: \"google-drive\",\n requiresAuth: true,\n features: {\n upload: true,\n download: true,\n list: true,\n delete: true,\n },\n };\n }\n\n /**\n * Make a Google Drive file publicly readable\n *\n * @param fileId - Google Drive file ID\n */\n private async makeFilePublic(fileId: string): Promise<void> {\n try {\n await fetch(`${this.baseUrl}/files/${fileId}/permissions`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n role: \"reader\",\n type: \"anyone\",\n }),\n });\n } catch (error) {\n // Non-critical error - file upload succeeded but sharing failed\n console.warn(\"Failed to make Google Drive file public:\", error);\n }\n }\n\n /**\n * Extract file ID from various Google Drive URL formats\n *\n * @param url - Google Drive URL\n * @returns File ID or null if not found\n */\n private extractFileId(url: string): string | null {\n // Handle various Google Drive URL formats\n const patterns = [\n /\\/file\\/d\\/([a-zA-Z0-9-_]+)/, // https://drive.google.com/file/d/FILE_ID/view\n /id=([a-zA-Z0-9-_]+)/, // https://drive.google.com/uc?id=FILE_ID\n /^([a-zA-Z0-9-_]+)$/, // Just the file ID\n ];\n\n for (const pattern of patterns) {\n const match = url.match(pattern);\n if (match) {\n return match[1];\n }\n }\n\n return null;\n }\n\n /**\n * Searches for an existing folder by name within a specified parent folder\n *\n * @remarks\n * This method queries the Google Drive API to find a folder with the exact name\n * within the specified parent directory. Only searches for folders (not files)\n * and excludes trashed items.\n *\n * @param name - The exact name of the folder to search for\n * @param parentId - The ID of the parent folder to search within (defaults to 'root')\n * @returns Promise that resolves to the folder ID if found, or `null` if not found\n * @throws {StorageError} When the Google Drive API request fails\n *\n * @example\n * ```typescript\n * // Search for a folder in the root directory\n * const folderId = await googleDriveStorage.findFolder(\"screenshots\");\n * if (folderId) {\n * console.log(\"Found folder:\", folderId);\n * } else {\n * console.log(\"Folder not found\");\n * }\n *\n * // Search for a subfolder within another folder\n * const subFolderId = await googleDriveStorage.findFolder(\"roasts\", parentFolderId);\n * ```\n */\n async findFolder(\n name: string,\n parentId: string = \"root\",\n ): Promise<string | null> {\n try {\n const query = `name='${name}' and '${parentId}' in parents and mimeType='application/vnd.google-apps.folder' and trashed = false`;\n\n const params = new URLSearchParams({\n q: query,\n fields: \"files(id,name,mimeType)\",\n });\n\n const response = await fetch(`${this.baseUrl}/files?${params}`, {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to search Google Drive folders: ${error}`,\n \"FIND_FOLDER_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveListResponse;\n\n if (result.files && result.files.length > 0) {\n return result.files[0].id;\n }\n\n return null;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive findFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"FIND_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Creates a new folder within a specified parent folder\n *\n * @remarks\n * This method creates a new folder using the Google Drive API. The folder will be\n * created with the specified name as a child of the parent folder. Requires the\n * `https://www.googleapis.com/auth/drive.file` OAuth scope.\n *\n * @param name - The name for the new folder\n * @param parentId - The ID of the parent folder where the new folder will be created (defaults to 'root')\n * @returns Promise that resolves to the ID of the newly created folder\n * @throws {StorageError} When the Google Drive API request fails or folder creation is denied\n *\n * @example\n * ```typescript\n * // Create a folder in the root directory\n * const folderId = await googleDriveStorage.createFolder(\"my-documents\");\n * console.log(\"Created folder:\", folderId);\n *\n * // Create a subfolder within another folder\n * const subFolderId = await googleDriveStorage.createFolder(\"reports\", parentFolderId);\n * ```\n */\n async createFolder(name: string, parentId: string = \"root\"): Promise<string> {\n try {\n const metadata = {\n name,\n mimeType: \"application/vnd.google-apps.folder\",\n parents: [parentId],\n };\n\n const response = await fetch(`${this.baseUrl}/files`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify(metadata),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to create Google Drive folder: ${error}`,\n \"CREATE_FOLDER_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveUploadResponse;\n return result.id;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive createFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"CREATE_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Finds an existing folder by name, or creates it if it doesn't exist\n *\n * @remarks\n * This is a convenience method that combines `findFolder` and `createFolder`.\n * It first searches for an existing folder with the specified name. If found,\n * it returns the existing folder's ID. If not found, it creates a new folder\n * and returns the new folder's ID.\n *\n * @param name - The name of the folder to find or create\n * @param parentId - The ID of the parent folder to search within or create the folder in (defaults to 'root')\n * @returns Promise that resolves to the folder ID (either existing or newly created)\n * @throws {StorageError} When the Google Drive API request fails\n *\n * @example\n * ```typescript\n * // Ensure a folder exists, creating it if necessary\n * const folderId = await googleDriveStorage.findOrCreateFolder(\"screenshots\");\n * console.log(\"Folder ID:\", folderId); // Will be same ID if folder already existed\n *\n * // Create nested folder structure\n * const parentId = await googleDriveStorage.findOrCreateFolder(\"projects\");\n * const childId = await googleDriveStorage.findOrCreateFolder(\"vana-app\", parentId);\n * ```\n */\n async findOrCreateFolder(\n name: string,\n parentId: string = \"root\",\n ): Promise<string> {\n try {\n const existingFolderId = await this.findFolder(name, parentId);\n\n if (existingFolderId) {\n return existingFolderId;\n }\n\n return await this.createFolder(name, parentId);\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive findOrCreateFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"FIND_OR_CREATE_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Refresh the access token using refresh token\n *\n * @returns Promise with new access token\n */\n async refreshAccessToken(): Promise<string> {\n if (\n !this.config.refreshToken ||\n !this.config.clientId ||\n !this.config.clientSecret\n ) {\n throw new StorageError(\n \"Refresh token, client ID, and client secret are required for token refresh\",\n \"MISSING_REFRESH_CONFIG\",\n \"google-drive\",\n );\n }\n\n try {\n const response = await fetch(\"https://oauth2.googleapis.com/token\", {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\",\n },\n body: new URLSearchParams({\n client_id: this.config.clientId,\n client_secret: this.config.clientSecret,\n refresh_token: this.config.refreshToken,\n grant_type: \"refresh_token\",\n }),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to refresh Google Drive token: ${error}`,\n \"TOKEN_REFRESH_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveTokenResponse;\n this.config.accessToken = result.access_token;\n\n return result.access_token;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive token refresh error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"TOKEN_REFRESH_ERROR\",\n \"google-drive\",\n );\n }\n }\n}\n"],"mappings":"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","names":[]}
@@ -12,6 +12,7 @@ class GoogleDriveStorage {
12
12
  );
13
13
  }
14
14
  }
15
+ config;
15
16
  baseUrl = "https://www.googleapis.com/drive/v3";
16
17
  uploadUrl = "https://www.googleapis.com/upload/drive/v3";
17
18
  async upload(file, filename) {
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/storage/providers/google-drive.ts"],"sourcesContent":["/**\n * Google Drive Storage Provider for Vana SDK\n *\n * Implements storage interface for Google Drive using OAuth2 authentication.\n * Based on patterns from dlp-ui-template with NextAuth integration.\n */\n\nimport {\n StorageError,\n type StorageProvider,\n type StorageUploadResult,\n type StorageFile,\n type StorageListOptions,\n type StorageProviderConfig,\n} from \"../index\";\n\nexport interface GoogleDriveConfig {\n /** OAuth2 access token */\n accessToken: string;\n /** Optional refresh token for token renewal */\n refreshToken?: string;\n /** OAuth2 client ID */\n clientId?: string;\n /** OAuth2 client secret */\n clientSecret?: string;\n /** Parent folder ID to upload files to */\n folderId?: string;\n}\n\ninterface GoogleDriveFile {\n id: string;\n name: string;\n webViewLink: string;\n size: string;\n mimeType: string;\n createdTime: string;\n}\n\ninterface GoogleDriveUploadResponse {\n id: string;\n name: string;\n}\n\ninterface GoogleDriveListResponse {\n files: GoogleDriveFile[];\n}\n\ninterface GoogleDriveTokenResponse {\n access_token: string;\n}\n\n/**\n * Google Drive Storage Provider with folder management capabilities\n *\n * @remarks\n * Implements storage interface for Google Drive using OAuth2 authentication.\n * Provides file upload/download operations and advanced folder management\n * including search, creation, and nested folder structures. Requires the\n * `https://www.googleapis.com/auth/drive.file` OAuth scope for full functionality.\n *\n * @category Storage\n *\n * @example\n * ```typescript\n * const googleDriveStorage = new GoogleDriveStorage({\n * accessToken: \"your-oauth-access-token\",\n * refreshToken: \"your-oauth-refresh-token\",\n * clientId: \"your-oauth-client-id\",\n * clientSecret: \"your-oauth-client-secret\",\n * });\n *\n * // Create folder structure and upload file\n * const folderId = await googleDriveStorage.findOrCreateFolder(\"screenshots\");\n * const result = await googleDriveStorage.upload(fileBlob, \"image.png\");\n * ```\n */\nexport class GoogleDriveStorage implements StorageProvider {\n private readonly baseUrl = \"https://www.googleapis.com/drive/v3\";\n private readonly uploadUrl = \"https://www.googleapis.com/upload/drive/v3\";\n\n constructor(private config: GoogleDriveConfig) {\n if (!config.accessToken) {\n throw new StorageError(\n \"Google Drive access token is required\",\n \"MISSING_ACCESS_TOKEN\",\n \"google-drive\",\n );\n }\n }\n\n async upload(file: Blob, filename?: string): Promise<StorageUploadResult> {\n try {\n const fileName = filename ?? `vana-file-${Date.now()}.dat`;\n\n // Create file metadata\n const metadata = {\n name: fileName,\n parents: this.config.folderId ? [this.config.folderId] : undefined,\n };\n\n // Create multipart upload request\n const delimiter = \"-------314159265358979323846\";\n const closeDelim = `\\r\\n--${delimiter}--`;\n\n const metadataBlob = new Blob([JSON.stringify(metadata)], {\n type: \"application/json\",\n });\n\n const multipartRequestBody = [\n `--${delimiter}`,\n \"Content-Type: application/json\",\n \"\",\n await metadataBlob.text(),\n `--${delimiter}`,\n `Content-Type: ${file.type || \"application/octet-stream\"}`,\n \"\",\n \"\",\n ].join(\"\\r\\n\");\n\n const requestBody = new Blob([multipartRequestBody, file, closeDelim]);\n\n const response = await fetch(\n `${this.uploadUrl}/files?uploadType=multipart`,\n {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": `multipart/related; boundary=\"${delimiter}\"`,\n },\n body: requestBody,\n },\n );\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to upload to Google Drive: ${error}`,\n \"UPLOAD_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveUploadResponse;\n\n // Make file publicly readable\n await this.makeFilePublic(result.id);\n\n return {\n url: `https://drive.google.com/uc?id=${result.id}&export=download`,\n size: file.size,\n contentType: file.type || \"application/octet-stream\",\n metadata: {\n id: result.id,\n name: result.name,\n driveUrl: `https://drive.google.com/file/d/${result.id}/view`,\n },\n };\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive upload error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"UPLOAD_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async download(url: string): Promise<Blob> {\n try {\n // Extract file ID from Google Drive URL\n const fileId = this.extractFileId(url);\n if (!fileId) {\n throw new StorageError(\n \"Invalid Google Drive URL format\",\n \"INVALID_URL\",\n \"google-drive\",\n );\n }\n\n const response = await fetch(\n `${this.baseUrl}/files/${fileId}?alt=media`,\n {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n },\n );\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to download from Google Drive: ${error}`,\n \"DOWNLOAD_FAILED\",\n \"google-drive\",\n );\n }\n\n const blob = await response.blob();\n\n // Check if we got HTML content instead of the actual file (authentication issue)\n if (blob.type === \"text/html\") {\n throw new StorageError(\n \"Received HTML content instead of file data. This suggests an authentication or URL formatting issue with Google Drive.\",\n \"AUTHENTICATION_ERROR\",\n \"google-drive\",\n );\n }\n\n return blob;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive download error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DOWNLOAD_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async list(options?: StorageListOptions): Promise<StorageFile[]> {\n try {\n let query = \"trashed = false\";\n\n // Add parent folder filter if configured\n if (this.config.folderId) {\n query += ` and '${this.config.folderId}' in parents`;\n }\n\n // Add name pattern filter\n if (options?.namePattern) {\n query += ` and name contains '${options.namePattern}'`;\n }\n\n const params = new URLSearchParams({\n q: query,\n fields: \"files(id,name,size,mimeType,createdTime,webViewLink)\",\n pageSize: (options?.limit ?? 100).toString(),\n });\n\n if (options?.offset && typeof options.offset === \"string\") {\n params.set(\"pageToken\", options.offset);\n }\n\n const response = await fetch(`${this.baseUrl}/files?${params}`, {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to list Google Drive files: ${error}`,\n \"LIST_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveListResponse;\n\n return result.files.map((file: GoogleDriveFile) => ({\n id: file.id,\n name: file.name,\n url: `https://drive.google.com/uc?id=${file.id}&export=download`,\n size: parseInt(file.size) || 0,\n contentType: file.mimeType,\n createdAt: new Date(file.createdTime),\n metadata: {\n id: file.id,\n driveUrl: file.webViewLink,\n },\n }));\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive list error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"LIST_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async delete(url: string): Promise<boolean> {\n try {\n // Extract file ID from Google Drive URL\n const fileId = this.extractFileId(url);\n if (!fileId) {\n throw new StorageError(\n \"Invalid Google Drive URL format\",\n \"INVALID_URL\",\n \"google-drive\",\n );\n }\n\n const response = await fetch(`${this.baseUrl}/files/${fileId}`, {\n method: \"DELETE\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok && response.status !== 404) {\n const error = await response.text();\n throw new StorageError(\n `Failed to delete from Google Drive: ${error}`,\n \"DELETE_FAILED\",\n \"google-drive\",\n );\n }\n\n return true;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive delete error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DELETE_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n getConfig(): StorageProviderConfig {\n return {\n name: \"Google Drive\",\n type: \"google-drive\",\n requiresAuth: true,\n features: {\n upload: true,\n download: true,\n list: true,\n delete: true,\n },\n };\n }\n\n /**\n * Make a Google Drive file publicly readable\n *\n * @param fileId - Google Drive file ID\n */\n private async makeFilePublic(fileId: string): Promise<void> {\n try {\n await fetch(`${this.baseUrl}/files/${fileId}/permissions`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n role: \"reader\",\n type: \"anyone\",\n }),\n });\n } catch (error) {\n // Non-critical error - file upload succeeded but sharing failed\n console.warn(\"Failed to make Google Drive file public:\", error);\n }\n }\n\n /**\n * Extract file ID from various Google Drive URL formats\n *\n * @param url - Google Drive URL\n * @returns File ID or null if not found\n */\n private extractFileId(url: string): string | null {\n // Handle various Google Drive URL formats\n const patterns = [\n /\\/file\\/d\\/([a-zA-Z0-9-_]+)/, // https://drive.google.com/file/d/FILE_ID/view\n /id=([a-zA-Z0-9-_]+)/, // https://drive.google.com/uc?id=FILE_ID\n /^([a-zA-Z0-9-_]+)$/, // Just the file ID\n ];\n\n for (const pattern of patterns) {\n const match = url.match(pattern);\n if (match) {\n return match[1];\n }\n }\n\n return null;\n }\n\n /**\n * Searches for an existing folder by name within a specified parent folder\n *\n * @remarks\n * This method queries the Google Drive API to find a folder with the exact name\n * within the specified parent directory. Only searches for folders (not files)\n * and excludes trashed items.\n *\n * @param name - The exact name of the folder to search for\n * @param parentId - The ID of the parent folder to search within (defaults to 'root')\n * @returns Promise that resolves to the folder ID if found, or `null` if not found\n * @throws {StorageError} When the Google Drive API request fails\n *\n * @example\n * ```typescript\n * // Search for a folder in the root directory\n * const folderId = await googleDriveStorage.findFolder(\"screenshots\");\n * if (folderId) {\n * console.log(\"Found folder:\", folderId);\n * } else {\n * console.log(\"Folder not found\");\n * }\n *\n * // Search for a subfolder within another folder\n * const subFolderId = await googleDriveStorage.findFolder(\"roasts\", parentFolderId);\n * ```\n */\n async findFolder(\n name: string,\n parentId: string = \"root\",\n ): Promise<string | null> {\n try {\n const query = `name='${name}' and '${parentId}' in parents and mimeType='application/vnd.google-apps.folder' and trashed = false`;\n\n const params = new URLSearchParams({\n q: query,\n fields: \"files(id,name,mimeType)\",\n });\n\n const response = await fetch(`${this.baseUrl}/files?${params}`, {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to search Google Drive folders: ${error}`,\n \"FIND_FOLDER_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveListResponse;\n\n if (result.files && result.files.length > 0) {\n return result.files[0].id;\n }\n\n return null;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive findFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"FIND_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Creates a new folder within a specified parent folder\n *\n * @remarks\n * This method creates a new folder using the Google Drive API. The folder will be\n * created with the specified name as a child of the parent folder. Requires the\n * `https://www.googleapis.com/auth/drive.file` OAuth scope.\n *\n * @param name - The name for the new folder\n * @param parentId - The ID of the parent folder where the new folder will be created (defaults to 'root')\n * @returns Promise that resolves to the ID of the newly created folder\n * @throws {StorageError} When the Google Drive API request fails or folder creation is denied\n *\n * @example\n * ```typescript\n * // Create a folder in the root directory\n * const folderId = await googleDriveStorage.createFolder(\"my-documents\");\n * console.log(\"Created folder:\", folderId);\n *\n * // Create a subfolder within another folder\n * const subFolderId = await googleDriveStorage.createFolder(\"reports\", parentFolderId);\n * ```\n */\n async createFolder(name: string, parentId: string = \"root\"): Promise<string> {\n try {\n const metadata = {\n name,\n mimeType: \"application/vnd.google-apps.folder\",\n parents: [parentId],\n };\n\n const response = await fetch(`${this.baseUrl}/files`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify(metadata),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to create Google Drive folder: ${error}`,\n \"CREATE_FOLDER_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveUploadResponse;\n return result.id;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive createFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"CREATE_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Finds an existing folder by name, or creates it if it doesn't exist\n *\n * @remarks\n * This is a convenience method that combines `findFolder` and `createFolder`.\n * It first searches for an existing folder with the specified name. If found,\n * it returns the existing folder's ID. If not found, it creates a new folder\n * and returns the new folder's ID.\n *\n * @param name - The name of the folder to find or create\n * @param parentId - The ID of the parent folder to search within or create the folder in (defaults to 'root')\n * @returns Promise that resolves to the folder ID (either existing or newly created)\n * @throws {StorageError} When the Google Drive API request fails\n *\n * @example\n * ```typescript\n * // Ensure a folder exists, creating it if necessary\n * const folderId = await googleDriveStorage.findOrCreateFolder(\"screenshots\");\n * console.log(\"Folder ID:\", folderId); // Will be same ID if folder already existed\n *\n * // Create nested folder structure\n * const parentId = await googleDriveStorage.findOrCreateFolder(\"projects\");\n * const childId = await googleDriveStorage.findOrCreateFolder(\"vana-app\", parentId);\n * ```\n */\n async findOrCreateFolder(\n name: string,\n parentId: string = \"root\",\n ): Promise<string> {\n try {\n const existingFolderId = await this.findFolder(name, parentId);\n\n if (existingFolderId) {\n return existingFolderId;\n }\n\n return await this.createFolder(name, parentId);\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive findOrCreateFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"FIND_OR_CREATE_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Refresh the access token using refresh token\n *\n * @returns Promise with new access token\n */\n async refreshAccessToken(): Promise<string> {\n if (\n !this.config.refreshToken ||\n !this.config.clientId ||\n !this.config.clientSecret\n ) {\n throw new StorageError(\n \"Refresh token, client ID, and client secret are required for token refresh\",\n \"MISSING_REFRESH_CONFIG\",\n \"google-drive\",\n );\n }\n\n try {\n const response = await fetch(\"https://oauth2.googleapis.com/token\", {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\",\n },\n body: new URLSearchParams({\n client_id: this.config.clientId,\n client_secret: this.config.clientSecret,\n refresh_token: this.config.refreshToken,\n grant_type: \"refresh_token\",\n }),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to refresh Google Drive token: ${error}`,\n \"TOKEN_REFRESH_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveTokenResponse;\n this.config.accessToken = result.access_token;\n\n return result.access_token;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive token refresh error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"TOKEN_REFRESH_ERROR\",\n \"google-drive\",\n );\n }\n }\n}\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../../../src/storage/providers/google-drive.ts"],"sourcesContent":["/**\n * Google Drive Storage Provider for Vana SDK\n *\n * Implements storage interface for Google Drive using OAuth2 authentication.\n * Based on patterns from dlp-ui-template with NextAuth integration.\n */\n\nimport {\n StorageError,\n type StorageProvider,\n type StorageUploadResult,\n type StorageFile,\n type StorageListOptions,\n type StorageProviderConfig,\n} from \"../index\";\n\nexport interface GoogleDriveConfig {\n /** OAuth2 access token */\n accessToken: string;\n /** Optional refresh token for token renewal */\n refreshToken?: string;\n /** OAuth2 client ID */\n clientId?: string;\n /** OAuth2 client secret */\n clientSecret?: string;\n /** Parent folder ID to upload files to */\n folderId?: string;\n}\n\ninterface GoogleDriveFile {\n id: string;\n name: string;\n webViewLink: string;\n size: string;\n mimeType: string;\n createdTime: string;\n}\n\ninterface GoogleDriveUploadResponse {\n id: string;\n name: string;\n}\n\ninterface GoogleDriveListResponse {\n files: GoogleDriveFile[];\n}\n\ninterface GoogleDriveTokenResponse {\n access_token: string;\n}\n\n/**\n * Google Drive Storage Provider with folder management capabilities\n *\n * @remarks\n * Implements storage interface for Google Drive using OAuth2 authentication.\n * Provides file upload/download operations and advanced folder management\n * including search, creation, and nested folder structures. Requires the\n * `https://www.googleapis.com/auth/drive.file` OAuth scope for full functionality.\n *\n * @category Storage\n *\n * @example\n * ```typescript\n * const googleDriveStorage = new GoogleDriveStorage({\n * accessToken: \"your-oauth-access-token\",\n * refreshToken: \"your-oauth-refresh-token\",\n * clientId: \"your-oauth-client-id\",\n * clientSecret: \"your-oauth-client-secret\",\n * });\n *\n * // Create folder structure and upload file\n * const folderId = await googleDriveStorage.findOrCreateFolder(\"screenshots\");\n * const result = await googleDriveStorage.upload(fileBlob, \"image.png\");\n * ```\n */\nexport class GoogleDriveStorage implements StorageProvider {\n private readonly baseUrl = \"https://www.googleapis.com/drive/v3\";\n private readonly uploadUrl = \"https://www.googleapis.com/upload/drive/v3\";\n\n constructor(private config: GoogleDriveConfig) {\n if (!config.accessToken) {\n throw new StorageError(\n \"Google Drive access token is required\",\n \"MISSING_ACCESS_TOKEN\",\n \"google-drive\",\n );\n }\n }\n\n async upload(file: Blob, filename?: string): Promise<StorageUploadResult> {\n try {\n const fileName = filename ?? `vana-file-${Date.now()}.dat`;\n\n // Create file metadata\n const metadata = {\n name: fileName,\n parents: this.config.folderId ? [this.config.folderId] : undefined,\n };\n\n // Create multipart upload request\n const delimiter = \"-------314159265358979323846\";\n const closeDelim = `\\r\\n--${delimiter}--`;\n\n const metadataBlob = new Blob([JSON.stringify(metadata)], {\n type: \"application/json\",\n });\n\n const multipartRequestBody = [\n `--${delimiter}`,\n \"Content-Type: application/json\",\n \"\",\n await metadataBlob.text(),\n `--${delimiter}`,\n `Content-Type: ${file.type || \"application/octet-stream\"}`,\n \"\",\n \"\",\n ].join(\"\\r\\n\");\n\n const requestBody = new Blob([multipartRequestBody, file, closeDelim]);\n\n const response = await fetch(\n `${this.uploadUrl}/files?uploadType=multipart`,\n {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": `multipart/related; boundary=\"${delimiter}\"`,\n },\n body: requestBody,\n },\n );\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to upload to Google Drive: ${error}`,\n \"UPLOAD_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveUploadResponse;\n\n // Make file publicly readable\n await this.makeFilePublic(result.id);\n\n return {\n url: `https://drive.google.com/uc?id=${result.id}&export=download`,\n size: file.size,\n contentType: file.type || \"application/octet-stream\",\n metadata: {\n id: result.id,\n name: result.name,\n driveUrl: `https://drive.google.com/file/d/${result.id}/view`,\n },\n };\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive upload error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"UPLOAD_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async download(url: string): Promise<Blob> {\n try {\n // Extract file ID from Google Drive URL\n const fileId = this.extractFileId(url);\n if (!fileId) {\n throw new StorageError(\n \"Invalid Google Drive URL format\",\n \"INVALID_URL\",\n \"google-drive\",\n );\n }\n\n const response = await fetch(\n `${this.baseUrl}/files/${fileId}?alt=media`,\n {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n },\n );\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to download from Google Drive: ${error}`,\n \"DOWNLOAD_FAILED\",\n \"google-drive\",\n );\n }\n\n const blob = await response.blob();\n\n // Check if we got HTML content instead of the actual file (authentication issue)\n if (blob.type === \"text/html\") {\n throw new StorageError(\n \"Received HTML content instead of file data. This suggests an authentication or URL formatting issue with Google Drive.\",\n \"AUTHENTICATION_ERROR\",\n \"google-drive\",\n );\n }\n\n return blob;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive download error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DOWNLOAD_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async list(options?: StorageListOptions): Promise<StorageFile[]> {\n try {\n let query = \"trashed = false\";\n\n // Add parent folder filter if configured\n if (this.config.folderId) {\n query += ` and '${this.config.folderId}' in parents`;\n }\n\n // Add name pattern filter\n if (options?.namePattern) {\n query += ` and name contains '${options.namePattern}'`;\n }\n\n const params = new URLSearchParams({\n q: query,\n fields: \"files(id,name,size,mimeType,createdTime,webViewLink)\",\n pageSize: (options?.limit ?? 100).toString(),\n });\n\n if (options?.offset && typeof options.offset === \"string\") {\n params.set(\"pageToken\", options.offset);\n }\n\n const response = await fetch(`${this.baseUrl}/files?${params}`, {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to list Google Drive files: ${error}`,\n \"LIST_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveListResponse;\n\n return result.files.map((file: GoogleDriveFile) => ({\n id: file.id,\n name: file.name,\n url: `https://drive.google.com/uc?id=${file.id}&export=download`,\n size: parseInt(file.size) || 0,\n contentType: file.mimeType,\n createdAt: new Date(file.createdTime),\n metadata: {\n id: file.id,\n driveUrl: file.webViewLink,\n },\n }));\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive list error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"LIST_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n async delete(url: string): Promise<boolean> {\n try {\n // Extract file ID from Google Drive URL\n const fileId = this.extractFileId(url);\n if (!fileId) {\n throw new StorageError(\n \"Invalid Google Drive URL format\",\n \"INVALID_URL\",\n \"google-drive\",\n );\n }\n\n const response = await fetch(`${this.baseUrl}/files/${fileId}`, {\n method: \"DELETE\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok && response.status !== 404) {\n const error = await response.text();\n throw new StorageError(\n `Failed to delete from Google Drive: ${error}`,\n \"DELETE_FAILED\",\n \"google-drive\",\n );\n }\n\n return true;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive delete error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DELETE_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n getConfig(): StorageProviderConfig {\n return {\n name: \"Google Drive\",\n type: \"google-drive\",\n requiresAuth: true,\n features: {\n upload: true,\n download: true,\n list: true,\n delete: true,\n },\n };\n }\n\n /**\n * Make a Google Drive file publicly readable\n *\n * @param fileId - Google Drive file ID\n */\n private async makeFilePublic(fileId: string): Promise<void> {\n try {\n await fetch(`${this.baseUrl}/files/${fileId}/permissions`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n role: \"reader\",\n type: \"anyone\",\n }),\n });\n } catch (error) {\n // Non-critical error - file upload succeeded but sharing failed\n console.warn(\"Failed to make Google Drive file public:\", error);\n }\n }\n\n /**\n * Extract file ID from various Google Drive URL formats\n *\n * @param url - Google Drive URL\n * @returns File ID or null if not found\n */\n private extractFileId(url: string): string | null {\n // Handle various Google Drive URL formats\n const patterns = [\n /\\/file\\/d\\/([a-zA-Z0-9-_]+)/, // https://drive.google.com/file/d/FILE_ID/view\n /id=([a-zA-Z0-9-_]+)/, // https://drive.google.com/uc?id=FILE_ID\n /^([a-zA-Z0-9-_]+)$/, // Just the file ID\n ];\n\n for (const pattern of patterns) {\n const match = url.match(pattern);\n if (match) {\n return match[1];\n }\n }\n\n return null;\n }\n\n /**\n * Searches for an existing folder by name within a specified parent folder\n *\n * @remarks\n * This method queries the Google Drive API to find a folder with the exact name\n * within the specified parent directory. Only searches for folders (not files)\n * and excludes trashed items.\n *\n * @param name - The exact name of the folder to search for\n * @param parentId - The ID of the parent folder to search within (defaults to 'root')\n * @returns Promise that resolves to the folder ID if found, or `null` if not found\n * @throws {StorageError} When the Google Drive API request fails\n *\n * @example\n * ```typescript\n * // Search for a folder in the root directory\n * const folderId = await googleDriveStorage.findFolder(\"screenshots\");\n * if (folderId) {\n * console.log(\"Found folder:\", folderId);\n * } else {\n * console.log(\"Folder not found\");\n * }\n *\n * // Search for a subfolder within another folder\n * const subFolderId = await googleDriveStorage.findFolder(\"roasts\", parentFolderId);\n * ```\n */\n async findFolder(\n name: string,\n parentId: string = \"root\",\n ): Promise<string | null> {\n try {\n const query = `name='${name}' and '${parentId}' in parents and mimeType='application/vnd.google-apps.folder' and trashed = false`;\n\n const params = new URLSearchParams({\n q: query,\n fields: \"files(id,name,mimeType)\",\n });\n\n const response = await fetch(`${this.baseUrl}/files?${params}`, {\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n },\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to search Google Drive folders: ${error}`,\n \"FIND_FOLDER_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveListResponse;\n\n if (result.files && result.files.length > 0) {\n return result.files[0].id;\n }\n\n return null;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive findFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"FIND_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Creates a new folder within a specified parent folder\n *\n * @remarks\n * This method creates a new folder using the Google Drive API. The folder will be\n * created with the specified name as a child of the parent folder. Requires the\n * `https://www.googleapis.com/auth/drive.file` OAuth scope.\n *\n * @param name - The name for the new folder\n * @param parentId - The ID of the parent folder where the new folder will be created (defaults to 'root')\n * @returns Promise that resolves to the ID of the newly created folder\n * @throws {StorageError} When the Google Drive API request fails or folder creation is denied\n *\n * @example\n * ```typescript\n * // Create a folder in the root directory\n * const folderId = await googleDriveStorage.createFolder(\"my-documents\");\n * console.log(\"Created folder:\", folderId);\n *\n * // Create a subfolder within another folder\n * const subFolderId = await googleDriveStorage.createFolder(\"reports\", parentFolderId);\n * ```\n */\n async createFolder(name: string, parentId: string = \"root\"): Promise<string> {\n try {\n const metadata = {\n name,\n mimeType: \"application/vnd.google-apps.folder\",\n parents: [parentId],\n };\n\n const response = await fetch(`${this.baseUrl}/files`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${this.config.accessToken}`,\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify(metadata),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to create Google Drive folder: ${error}`,\n \"CREATE_FOLDER_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveUploadResponse;\n return result.id;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive createFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"CREATE_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Finds an existing folder by name, or creates it if it doesn't exist\n *\n * @remarks\n * This is a convenience method that combines `findFolder` and `createFolder`.\n * It first searches for an existing folder with the specified name. If found,\n * it returns the existing folder's ID. If not found, it creates a new folder\n * and returns the new folder's ID.\n *\n * @param name - The name of the folder to find or create\n * @param parentId - The ID of the parent folder to search within or create the folder in (defaults to 'root')\n * @returns Promise that resolves to the folder ID (either existing or newly created)\n * @throws {StorageError} When the Google Drive API request fails\n *\n * @example\n * ```typescript\n * // Ensure a folder exists, creating it if necessary\n * const folderId = await googleDriveStorage.findOrCreateFolder(\"screenshots\");\n * console.log(\"Folder ID:\", folderId); // Will be same ID if folder already existed\n *\n * // Create nested folder structure\n * const parentId = await googleDriveStorage.findOrCreateFolder(\"projects\");\n * const childId = await googleDriveStorage.findOrCreateFolder(\"vana-app\", parentId);\n * ```\n */\n async findOrCreateFolder(\n name: string,\n parentId: string = \"root\",\n ): Promise<string> {\n try {\n const existingFolderId = await this.findFolder(name, parentId);\n\n if (existingFolderId) {\n return existingFolderId;\n }\n\n return await this.createFolder(name, parentId);\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive findOrCreateFolder error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"FIND_OR_CREATE_FOLDER_ERROR\",\n \"google-drive\",\n );\n }\n }\n\n /**\n * Refresh the access token using refresh token\n *\n * @returns Promise with new access token\n */\n async refreshAccessToken(): Promise<string> {\n if (\n !this.config.refreshToken ||\n !this.config.clientId ||\n !this.config.clientSecret\n ) {\n throw new StorageError(\n \"Refresh token, client ID, and client secret are required for token refresh\",\n \"MISSING_REFRESH_CONFIG\",\n \"google-drive\",\n );\n }\n\n try {\n const response = await fetch(\"https://oauth2.googleapis.com/token\", {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\",\n },\n body: new URLSearchParams({\n client_id: this.config.clientId,\n client_secret: this.config.clientSecret,\n refresh_token: this.config.refreshToken,\n grant_type: \"refresh_token\",\n }),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to refresh Google Drive token: ${error}`,\n \"TOKEN_REFRESH_FAILED\",\n \"google-drive\",\n );\n }\n\n const result = (await response.json()) as GoogleDriveTokenResponse;\n this.config.accessToken = result.access_token;\n\n return result.access_token;\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `Google Drive token refresh error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"TOKEN_REFRESH_ERROR\",\n \"google-drive\",\n );\n }\n }\n}\n"],"mappings":"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","names":[]}
@@ -36,6 +36,7 @@ class IpfsStorage {
36
36
  this.gatewayUrl = config.gatewayUrl ?? "https://gateway.pinata.cloud/ipfs";
37
37
  this.hasAuth = !!(config.headers && Object.keys(config.headers).length > 0);
38
38
  }
39
+ config;
39
40
  gatewayUrl;
40
41
  hasAuth;
41
42
  /**
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/storage/providers/ipfs.ts"],"sourcesContent":["import {\n StorageError,\n type StorageProvider,\n type StorageUploadResult,\n type StorageFile,\n type StorageListOptions,\n type StorageProviderConfig,\n} from \"../index\";\nimport { toBase64 } from \"../../utils/encoding\";\n\nexport interface IpfsConfig {\n /** IPFS API endpoint for uploads */\n apiEndpoint: string;\n /** Gateway URL for downloads (optional, defaults to public gateway) */\n gatewayUrl?: string;\n /** Additional headers for API requests */\n headers?: Record<string, string>;\n}\n\ninterface IpfsUploadResponse {\n Hash?: string;\n Size?: number;\n}\n\n/**\n * Connects to any standard IPFS node or service provider\n *\n * @remarks\n * This provider implements the standard IPFS HTTP API (`/api/v0/add`) and works\n * with any IPFS-compatible service. It provides the essential IPFS operations\n * (upload/download) while maintaining the immutable, content-addressed nature\n * of IPFS. Use static factory methods for common providers like Infura or local nodes.\n *\n * @category Storage\n *\n * @example\n * ```typescript\n * // Use with Infura (recommended for production)\n * const ipfsStorage = IpfsStorage.forInfura({\n * projectId: \"your-project-id\",\n * projectSecret: \"your-project-secret\"\n * });\n *\n * // Use with local IPFS node\n * const localStorage = IpfsStorage.forLocalNode();\n *\n * // Upload file and get CID\n * const result = await ipfsStorage.upload(fileBlob, \"document.pdf\");\n * console.log(\"Uploaded to IPFS:\", result.url);\n * ```\n */\nexport class IpfsStorage implements StorageProvider {\n private readonly gatewayUrl: string;\n private readonly hasAuth: boolean;\n\n constructor(private config: IpfsConfig) {\n if (!config.apiEndpoint) {\n throw new StorageError(\n \"IPFS API endpoint is required\",\n \"MISSING_API_ENDPOINT\",\n \"ipfs\",\n );\n }\n\n this.gatewayUrl = config.gatewayUrl ?? \"https://gateway.pinata.cloud/ipfs\";\n this.hasAuth = !!(config.headers && Object.keys(config.headers).length > 0);\n }\n\n /**\n * Creates an IPFS storage instance configured for Infura\n *\n * @remarks\n * Infura provides reliable, scalable IPFS infrastructure with global availability.\n * This factory method automatically configures the correct endpoints and authentication\n * for Infura's IPFS service.\n *\n * @param credentials - Infura project credentials\n * @param credentials.projectId - Your Infura project ID\n * @param credentials.projectSecret - Your Infura project secret\n * @returns Configured IpfsStorage instance for Infura\n *\n * @example\n * ```typescript\n * const ipfsStorage = IpfsStorage.forInfura({\n * projectId: \"2FVGj8UJP5v8ZcnX9K5L7M8c\",\n * projectSecret: \"a7f2c1e5b8d9f3a6e4c8b2d7f9e1a4c3\"\n * });\n *\n * const result = await ipfsStorage.upload(fileBlob);\n * ```\n */\n static forInfura(credentials: {\n projectId: string;\n projectSecret: string;\n }): IpfsStorage {\n const encoder = new TextEncoder();\n const auth = toBase64(\n encoder.encode(`${credentials.projectId}:${credentials.projectSecret}`),\n );\n return new IpfsStorage({\n apiEndpoint: \"https://ipfs.infura.io:5001/api/v0/add\",\n gatewayUrl: \"https://ipfs.infura.io/ipfs\",\n headers: {\n Authorization: `Basic ${auth}`,\n },\n });\n }\n\n /**\n * Creates an IPFS storage instance configured for a local IPFS node\n *\n * @remarks\n * This factory method configures the storage provider to connect to a local IPFS node,\n * typically running on your development machine or server. Assumes standard ports\n * (5001 for API, 8080 for gateway) unless otherwise specified.\n *\n * @param options - Local node configuration options\n * @param options.url - Base URL of the local IPFS node (defaults to http://localhost:5001)\n * @returns Configured IpfsStorage instance for local node\n *\n * @example\n * ```typescript\n * // Use default localhost configuration\n * const localStorage = IpfsStorage.forLocalNode();\n *\n * // Use custom local node URL\n * const customStorage = IpfsStorage.forLocalNode({\n * url: \"http://192.168.1.100:5001\"\n * });\n *\n * const result = await localStorage.upload(fileBlob, \"local-file.txt\");\n * ```\n */\n static forLocalNode(options?: { url?: string }): IpfsStorage {\n const baseUrl = options?.url ?? \"http://localhost:5001\";\n return new IpfsStorage({\n apiEndpoint: `${baseUrl}/api/v0/add`,\n gatewayUrl: `${baseUrl.replace(\":5001\", \":8080\")}/ipfs`,\n });\n }\n\n /**\n * Uploads a file to IPFS and returns the content identifier (CID)\n *\n * @remarks\n * This method uploads the file to the configured IPFS endpoint using the standard\n * `/api/v0/add` API. The file is content-addressed, meaning the same file will\n * always produce the same CID regardless of when or where it's uploaded.\n *\n * @param file - The file to upload to IPFS\n * @param filename - Optional filename (for metadata purposes only)\n * @returns Promise that resolves to StorageUploadResult with IPFS gateway URL\n * @throws {StorageError} When the upload fails or no CID is returned\n *\n * @example\n * ```typescript\n * const result = await ipfsStorage.upload(fileBlob, \"report.pdf\");\n * console.log(\"File uploaded to IPFS:\", result.url);\n * // Example URL: \"https://gateway.pinata.cloud/ipfs/QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\"\n * ```\n */\n async upload(file: Blob, filename?: string): Promise<StorageUploadResult> {\n try {\n const fileName = filename ?? `ipfs-file-${Date.now()}.dat`;\n\n // Create FormData for IPFS upload\n const formData = new FormData();\n formData.append(\"file\", file, fileName);\n\n const response = await fetch(this.config.apiEndpoint, {\n method: \"POST\",\n headers: this.config.headers ?? {},\n body: formData,\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to upload to IPFS: ${error}`,\n \"UPLOAD_FAILED\",\n \"ipfs\",\n );\n }\n\n const result = (await response.json()) as IpfsUploadResponse;\n const hash = result.Hash;\n\n if (!hash) {\n throw new StorageError(\n \"IPFS upload succeeded but no hash returned\",\n \"NO_HASH_RETURNED\",\n \"ipfs\",\n );\n }\n\n return {\n url: `ipfs://${hash}`,\n size: file.size,\n contentType: file.type ?? \"application/octet-stream\",\n };\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `IPFS upload error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"UPLOAD_ERROR\",\n \"ipfs\",\n );\n }\n }\n\n /**\n * Downloads a file from IPFS using its content identifier (CID)\n *\n * @remarks\n * This method retrieves the file from IPFS using the configured gateway.\n * It accepts various formats including raw CIDs, ipfs:// URLs, and gateway URLs.\n * The file is downloaded from the globally distributed IPFS network.\n *\n * @param cid - The IPFS content identifier, ipfs:// URL, or gateway URL\n * @returns Promise that resolves to the downloaded file content\n * @throws {StorageError} When the download fails or CID format is invalid\n *\n * @example\n * ```typescript\n * // Download using raw CID\n * const file = await ipfsStorage.download(\"QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\");\n *\n * // Download using ipfs:// URL\n * const file2 = await ipfsStorage.download(\"ipfs://QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\");\n *\n * // Create download link\n * const url = URL.createObjectURL(file);\n * ```\n */\n async download(cid: string): Promise<Blob> {\n try {\n const downloadUrl = this.buildDownloadUrl(cid);\n\n const response = await fetch(downloadUrl);\n\n if (!response.ok) {\n throw new StorageError(\n `Failed to download from IPFS: ${response.statusText}`,\n \"DOWNLOAD_FAILED\",\n \"ipfs\",\n );\n }\n\n return await response.blob();\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `IPFS download error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DOWNLOAD_ERROR\",\n \"ipfs\",\n );\n }\n }\n\n async list(_options?: StorageListOptions): Promise<StorageFile[]> {\n throw new StorageError(\n \"List operation is not supported by standard IPFS. Use a service-specific provider like Pinata.\",\n \"LIST_NOT_SUPPORTED\",\n \"ipfs\",\n );\n }\n\n async delete(_url: string): Promise<boolean> {\n throw new StorageError(\n \"Delete operation is not supported by IPFS. Files are immutable once uploaded.\",\n \"DELETE_NOT_SUPPORTED\",\n \"ipfs\",\n );\n }\n\n getConfig(): StorageProviderConfig {\n return {\n name: \"IPFS\",\n type: \"ipfs\",\n requiresAuth: this.hasAuth,\n features: {\n upload: true,\n download: true,\n list: false,\n delete: false,\n },\n };\n }\n\n /**\n * Build download URL from CID or existing URL\n *\n * @param cid - IPFS CID or URL\n * @returns Gateway URL for download\n */\n private buildDownloadUrl(cid: string): string {\n // If it's already a full URL, return as-is\n if (cid.startsWith(\"http://\") || cid.startsWith(\"https://\")) {\n return cid;\n }\n\n // Handle ipfs:// URLs\n if (cid.startsWith(\"ipfs://\")) {\n const hash = cid.replace(\"ipfs://\", \"\");\n return `${this.gatewayUrl}/${hash}`;\n }\n\n // Validate CID format (basic validation)\n if (!this.isValidCID(cid)) {\n throw new StorageError(\n \"Invalid IPFS CID or URL format\",\n \"INVALID_CID\",\n \"ipfs\",\n );\n }\n\n // Assume it's a raw CID\n return `${this.gatewayUrl}/${cid}`;\n }\n\n /**\n * Basic CID validation\n *\n * @param cid - Content identifier to validate\n * @returns True if CID appears valid\n */\n private isValidCID(cid: string): boolean {\n // Basic validation: CIDs typically start with 'Qm' or 'ba' and contain alphanumeric characters\n // Allow shorter hashes for testing purposes\n return (\n /^[a-zA-Z0-9]{10,}$/.test(cid) &&\n (cid.startsWith(\"Qm\") || cid.startsWith(\"ba\") || cid.includes(\"Test\"))\n );\n }\n}\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../../../src/storage/providers/ipfs.ts"],"sourcesContent":["import {\n StorageError,\n type StorageProvider,\n type StorageUploadResult,\n type StorageFile,\n type StorageListOptions,\n type StorageProviderConfig,\n} from \"../index\";\nimport { toBase64 } from \"../../utils/encoding\";\n\nexport interface IpfsConfig {\n /** IPFS API endpoint for uploads */\n apiEndpoint: string;\n /** Gateway URL for downloads (optional, defaults to public gateway) */\n gatewayUrl?: string;\n /** Additional headers for API requests */\n headers?: Record<string, string>;\n}\n\ninterface IpfsUploadResponse {\n Hash?: string;\n Size?: number;\n}\n\n/**\n * Connects to any standard IPFS node or service provider\n *\n * @remarks\n * This provider implements the standard IPFS HTTP API (`/api/v0/add`) and works\n * with any IPFS-compatible service. It provides the essential IPFS operations\n * (upload/download) while maintaining the immutable, content-addressed nature\n * of IPFS. Use static factory methods for common providers like Infura or local nodes.\n *\n * @category Storage\n *\n * @example\n * ```typescript\n * // Use with Infura (recommended for production)\n * const ipfsStorage = IpfsStorage.forInfura({\n * projectId: \"your-project-id\",\n * projectSecret: \"your-project-secret\"\n * });\n *\n * // Use with local IPFS node\n * const localStorage = IpfsStorage.forLocalNode();\n *\n * // Upload file and get CID\n * const result = await ipfsStorage.upload(fileBlob, \"document.pdf\");\n * console.log(\"Uploaded to IPFS:\", result.url);\n * ```\n */\nexport class IpfsStorage implements StorageProvider {\n private readonly gatewayUrl: string;\n private readonly hasAuth: boolean;\n\n constructor(private config: IpfsConfig) {\n if (!config.apiEndpoint) {\n throw new StorageError(\n \"IPFS API endpoint is required\",\n \"MISSING_API_ENDPOINT\",\n \"ipfs\",\n );\n }\n\n this.gatewayUrl = config.gatewayUrl ?? \"https://gateway.pinata.cloud/ipfs\";\n this.hasAuth = !!(config.headers && Object.keys(config.headers).length > 0);\n }\n\n /**\n * Creates an IPFS storage instance configured for Infura\n *\n * @remarks\n * Infura provides reliable, scalable IPFS infrastructure with global availability.\n * This factory method automatically configures the correct endpoints and authentication\n * for Infura's IPFS service.\n *\n * @param credentials - Infura project credentials\n * @param credentials.projectId - Your Infura project ID\n * @param credentials.projectSecret - Your Infura project secret\n * @returns Configured IpfsStorage instance for Infura\n *\n * @example\n * ```typescript\n * const ipfsStorage = IpfsStorage.forInfura({\n * projectId: \"2FVGj8UJP5v8ZcnX9K5L7M8c\",\n * projectSecret: \"a7f2c1e5b8d9f3a6e4c8b2d7f9e1a4c3\"\n * });\n *\n * const result = await ipfsStorage.upload(fileBlob);\n * ```\n */\n static forInfura(credentials: {\n projectId: string;\n projectSecret: string;\n }): IpfsStorage {\n const encoder = new TextEncoder();\n const auth = toBase64(\n encoder.encode(`${credentials.projectId}:${credentials.projectSecret}`),\n );\n return new IpfsStorage({\n apiEndpoint: \"https://ipfs.infura.io:5001/api/v0/add\",\n gatewayUrl: \"https://ipfs.infura.io/ipfs\",\n headers: {\n Authorization: `Basic ${auth}`,\n },\n });\n }\n\n /**\n * Creates an IPFS storage instance configured for a local IPFS node\n *\n * @remarks\n * This factory method configures the storage provider to connect to a local IPFS node,\n * typically running on your development machine or server. Assumes standard ports\n * (5001 for API, 8080 for gateway) unless otherwise specified.\n *\n * @param options - Local node configuration options\n * @param options.url - Base URL of the local IPFS node (defaults to http://localhost:5001)\n * @returns Configured IpfsStorage instance for local node\n *\n * @example\n * ```typescript\n * // Use default localhost configuration\n * const localStorage = IpfsStorage.forLocalNode();\n *\n * // Use custom local node URL\n * const customStorage = IpfsStorage.forLocalNode({\n * url: \"http://192.168.1.100:5001\"\n * });\n *\n * const result = await localStorage.upload(fileBlob, \"local-file.txt\");\n * ```\n */\n static forLocalNode(options?: { url?: string }): IpfsStorage {\n const baseUrl = options?.url ?? \"http://localhost:5001\";\n return new IpfsStorage({\n apiEndpoint: `${baseUrl}/api/v0/add`,\n gatewayUrl: `${baseUrl.replace(\":5001\", \":8080\")}/ipfs`,\n });\n }\n\n /**\n * Uploads a file to IPFS and returns the content identifier (CID)\n *\n * @remarks\n * This method uploads the file to the configured IPFS endpoint using the standard\n * `/api/v0/add` API. The file is content-addressed, meaning the same file will\n * always produce the same CID regardless of when or where it's uploaded.\n *\n * @param file - The file to upload to IPFS\n * @param filename - Optional filename (for metadata purposes only)\n * @returns Promise that resolves to StorageUploadResult with IPFS gateway URL\n * @throws {StorageError} When the upload fails or no CID is returned\n *\n * @example\n * ```typescript\n * const result = await ipfsStorage.upload(fileBlob, \"report.pdf\");\n * console.log(\"File uploaded to IPFS:\", result.url);\n * // Example URL: \"https://gateway.pinata.cloud/ipfs/QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\"\n * ```\n */\n async upload(file: Blob, filename?: string): Promise<StorageUploadResult> {\n try {\n const fileName = filename ?? `ipfs-file-${Date.now()}.dat`;\n\n // Create FormData for IPFS upload\n const formData = new FormData();\n formData.append(\"file\", file, fileName);\n\n const response = await fetch(this.config.apiEndpoint, {\n method: \"POST\",\n headers: this.config.headers ?? {},\n body: formData,\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to upload to IPFS: ${error}`,\n \"UPLOAD_FAILED\",\n \"ipfs\",\n );\n }\n\n const result = (await response.json()) as IpfsUploadResponse;\n const hash = result.Hash;\n\n if (!hash) {\n throw new StorageError(\n \"IPFS upload succeeded but no hash returned\",\n \"NO_HASH_RETURNED\",\n \"ipfs\",\n );\n }\n\n return {\n url: `ipfs://${hash}`,\n size: file.size,\n contentType: file.type ?? \"application/octet-stream\",\n };\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `IPFS upload error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"UPLOAD_ERROR\",\n \"ipfs\",\n );\n }\n }\n\n /**\n * Downloads a file from IPFS using its content identifier (CID)\n *\n * @remarks\n * This method retrieves the file from IPFS using the configured gateway.\n * It accepts various formats including raw CIDs, ipfs:// URLs, and gateway URLs.\n * The file is downloaded from the globally distributed IPFS network.\n *\n * @param cid - The IPFS content identifier, ipfs:// URL, or gateway URL\n * @returns Promise that resolves to the downloaded file content\n * @throws {StorageError} When the download fails or CID format is invalid\n *\n * @example\n * ```typescript\n * // Download using raw CID\n * const file = await ipfsStorage.download(\"QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\");\n *\n * // Download using ipfs:// URL\n * const file2 = await ipfsStorage.download(\"ipfs://QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\");\n *\n * // Create download link\n * const url = URL.createObjectURL(file);\n * ```\n */\n async download(cid: string): Promise<Blob> {\n try {\n const downloadUrl = this.buildDownloadUrl(cid);\n\n const response = await fetch(downloadUrl);\n\n if (!response.ok) {\n throw new StorageError(\n `Failed to download from IPFS: ${response.statusText}`,\n \"DOWNLOAD_FAILED\",\n \"ipfs\",\n );\n }\n\n return await response.blob();\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `IPFS download error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DOWNLOAD_ERROR\",\n \"ipfs\",\n );\n }\n }\n\n async list(_options?: StorageListOptions): Promise<StorageFile[]> {\n throw new StorageError(\n \"List operation is not supported by standard IPFS. Use a service-specific provider like Pinata.\",\n \"LIST_NOT_SUPPORTED\",\n \"ipfs\",\n );\n }\n\n async delete(_url: string): Promise<boolean> {\n throw new StorageError(\n \"Delete operation is not supported by IPFS. Files are immutable once uploaded.\",\n \"DELETE_NOT_SUPPORTED\",\n \"ipfs\",\n );\n }\n\n getConfig(): StorageProviderConfig {\n return {\n name: \"IPFS\",\n type: \"ipfs\",\n requiresAuth: this.hasAuth,\n features: {\n upload: true,\n download: true,\n list: false,\n delete: false,\n },\n };\n }\n\n /**\n * Build download URL from CID or existing URL\n *\n * @param cid - IPFS CID or URL\n * @returns Gateway URL for download\n */\n private buildDownloadUrl(cid: string): string {\n // If it's already a full URL, return as-is\n if (cid.startsWith(\"http://\") || cid.startsWith(\"https://\")) {\n return cid;\n }\n\n // Handle ipfs:// URLs\n if (cid.startsWith(\"ipfs://\")) {\n const hash = cid.replace(\"ipfs://\", \"\");\n return `${this.gatewayUrl}/${hash}`;\n }\n\n // Validate CID format (basic validation)\n if (!this.isValidCID(cid)) {\n throw new StorageError(\n \"Invalid IPFS CID or URL format\",\n \"INVALID_CID\",\n \"ipfs\",\n );\n }\n\n // Assume it's a raw CID\n return `${this.gatewayUrl}/${cid}`;\n }\n\n /**\n * Basic CID validation\n *\n * @param cid - Content identifier to validate\n * @returns True if CID appears valid\n */\n private isValidCID(cid: string): boolean {\n // Basic validation: CIDs typically start with 'Qm' or 'ba' and contain alphanumeric characters\n // Allow shorter hashes for testing purposes\n return (\n /^[a-zA-Z0-9]{10,}$/.test(cid) &&\n (cid.startsWith(\"Qm\") || cid.startsWith(\"ba\") || cid.includes(\"Test\"))\n );\n }\n}\n"],"mappings":"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","names":[]}
@@ -15,6 +15,7 @@ class IpfsStorage {
15
15
  this.gatewayUrl = config.gatewayUrl ?? "https://gateway.pinata.cloud/ipfs";
16
16
  this.hasAuth = !!(config.headers && Object.keys(config.headers).length > 0);
17
17
  }
18
+ config;
18
19
  gatewayUrl;
19
20
  hasAuth;
20
21
  /**
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/storage/providers/ipfs.ts"],"sourcesContent":["import {\n StorageError,\n type StorageProvider,\n type StorageUploadResult,\n type StorageFile,\n type StorageListOptions,\n type StorageProviderConfig,\n} from \"../index\";\nimport { toBase64 } from \"../../utils/encoding\";\n\nexport interface IpfsConfig {\n /** IPFS API endpoint for uploads */\n apiEndpoint: string;\n /** Gateway URL for downloads (optional, defaults to public gateway) */\n gatewayUrl?: string;\n /** Additional headers for API requests */\n headers?: Record<string, string>;\n}\n\ninterface IpfsUploadResponse {\n Hash?: string;\n Size?: number;\n}\n\n/**\n * Connects to any standard IPFS node or service provider\n *\n * @remarks\n * This provider implements the standard IPFS HTTP API (`/api/v0/add`) and works\n * with any IPFS-compatible service. It provides the essential IPFS operations\n * (upload/download) while maintaining the immutable, content-addressed nature\n * of IPFS. Use static factory methods for common providers like Infura or local nodes.\n *\n * @category Storage\n *\n * @example\n * ```typescript\n * // Use with Infura (recommended for production)\n * const ipfsStorage = IpfsStorage.forInfura({\n * projectId: \"your-project-id\",\n * projectSecret: \"your-project-secret\"\n * });\n *\n * // Use with local IPFS node\n * const localStorage = IpfsStorage.forLocalNode();\n *\n * // Upload file and get CID\n * const result = await ipfsStorage.upload(fileBlob, \"document.pdf\");\n * console.log(\"Uploaded to IPFS:\", result.url);\n * ```\n */\nexport class IpfsStorage implements StorageProvider {\n private readonly gatewayUrl: string;\n private readonly hasAuth: boolean;\n\n constructor(private config: IpfsConfig) {\n if (!config.apiEndpoint) {\n throw new StorageError(\n \"IPFS API endpoint is required\",\n \"MISSING_API_ENDPOINT\",\n \"ipfs\",\n );\n }\n\n this.gatewayUrl = config.gatewayUrl ?? \"https://gateway.pinata.cloud/ipfs\";\n this.hasAuth = !!(config.headers && Object.keys(config.headers).length > 0);\n }\n\n /**\n * Creates an IPFS storage instance configured for Infura\n *\n * @remarks\n * Infura provides reliable, scalable IPFS infrastructure with global availability.\n * This factory method automatically configures the correct endpoints and authentication\n * for Infura's IPFS service.\n *\n * @param credentials - Infura project credentials\n * @param credentials.projectId - Your Infura project ID\n * @param credentials.projectSecret - Your Infura project secret\n * @returns Configured IpfsStorage instance for Infura\n *\n * @example\n * ```typescript\n * const ipfsStorage = IpfsStorage.forInfura({\n * projectId: \"2FVGj8UJP5v8ZcnX9K5L7M8c\",\n * projectSecret: \"a7f2c1e5b8d9f3a6e4c8b2d7f9e1a4c3\"\n * });\n *\n * const result = await ipfsStorage.upload(fileBlob);\n * ```\n */\n static forInfura(credentials: {\n projectId: string;\n projectSecret: string;\n }): IpfsStorage {\n const encoder = new TextEncoder();\n const auth = toBase64(\n encoder.encode(`${credentials.projectId}:${credentials.projectSecret}`),\n );\n return new IpfsStorage({\n apiEndpoint: \"https://ipfs.infura.io:5001/api/v0/add\",\n gatewayUrl: \"https://ipfs.infura.io/ipfs\",\n headers: {\n Authorization: `Basic ${auth}`,\n },\n });\n }\n\n /**\n * Creates an IPFS storage instance configured for a local IPFS node\n *\n * @remarks\n * This factory method configures the storage provider to connect to a local IPFS node,\n * typically running on your development machine or server. Assumes standard ports\n * (5001 for API, 8080 for gateway) unless otherwise specified.\n *\n * @param options - Local node configuration options\n * @param options.url - Base URL of the local IPFS node (defaults to http://localhost:5001)\n * @returns Configured IpfsStorage instance for local node\n *\n * @example\n * ```typescript\n * // Use default localhost configuration\n * const localStorage = IpfsStorage.forLocalNode();\n *\n * // Use custom local node URL\n * const customStorage = IpfsStorage.forLocalNode({\n * url: \"http://192.168.1.100:5001\"\n * });\n *\n * const result = await localStorage.upload(fileBlob, \"local-file.txt\");\n * ```\n */\n static forLocalNode(options?: { url?: string }): IpfsStorage {\n const baseUrl = options?.url ?? \"http://localhost:5001\";\n return new IpfsStorage({\n apiEndpoint: `${baseUrl}/api/v0/add`,\n gatewayUrl: `${baseUrl.replace(\":5001\", \":8080\")}/ipfs`,\n });\n }\n\n /**\n * Uploads a file to IPFS and returns the content identifier (CID)\n *\n * @remarks\n * This method uploads the file to the configured IPFS endpoint using the standard\n * `/api/v0/add` API. The file is content-addressed, meaning the same file will\n * always produce the same CID regardless of when or where it's uploaded.\n *\n * @param file - The file to upload to IPFS\n * @param filename - Optional filename (for metadata purposes only)\n * @returns Promise that resolves to StorageUploadResult with IPFS gateway URL\n * @throws {StorageError} When the upload fails or no CID is returned\n *\n * @example\n * ```typescript\n * const result = await ipfsStorage.upload(fileBlob, \"report.pdf\");\n * console.log(\"File uploaded to IPFS:\", result.url);\n * // Example URL: \"https://gateway.pinata.cloud/ipfs/QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\"\n * ```\n */\n async upload(file: Blob, filename?: string): Promise<StorageUploadResult> {\n try {\n const fileName = filename ?? `ipfs-file-${Date.now()}.dat`;\n\n // Create FormData for IPFS upload\n const formData = new FormData();\n formData.append(\"file\", file, fileName);\n\n const response = await fetch(this.config.apiEndpoint, {\n method: \"POST\",\n headers: this.config.headers ?? {},\n body: formData,\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to upload to IPFS: ${error}`,\n \"UPLOAD_FAILED\",\n \"ipfs\",\n );\n }\n\n const result = (await response.json()) as IpfsUploadResponse;\n const hash = result.Hash;\n\n if (!hash) {\n throw new StorageError(\n \"IPFS upload succeeded but no hash returned\",\n \"NO_HASH_RETURNED\",\n \"ipfs\",\n );\n }\n\n return {\n url: `ipfs://${hash}`,\n size: file.size,\n contentType: file.type ?? \"application/octet-stream\",\n };\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `IPFS upload error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"UPLOAD_ERROR\",\n \"ipfs\",\n );\n }\n }\n\n /**\n * Downloads a file from IPFS using its content identifier (CID)\n *\n * @remarks\n * This method retrieves the file from IPFS using the configured gateway.\n * It accepts various formats including raw CIDs, ipfs:// URLs, and gateway URLs.\n * The file is downloaded from the globally distributed IPFS network.\n *\n * @param cid - The IPFS content identifier, ipfs:// URL, or gateway URL\n * @returns Promise that resolves to the downloaded file content\n * @throws {StorageError} When the download fails or CID format is invalid\n *\n * @example\n * ```typescript\n * // Download using raw CID\n * const file = await ipfsStorage.download(\"QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\");\n *\n * // Download using ipfs:// URL\n * const file2 = await ipfsStorage.download(\"ipfs://QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\");\n *\n * // Create download link\n * const url = URL.createObjectURL(file);\n * ```\n */\n async download(cid: string): Promise<Blob> {\n try {\n const downloadUrl = this.buildDownloadUrl(cid);\n\n const response = await fetch(downloadUrl);\n\n if (!response.ok) {\n throw new StorageError(\n `Failed to download from IPFS: ${response.statusText}`,\n \"DOWNLOAD_FAILED\",\n \"ipfs\",\n );\n }\n\n return await response.blob();\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `IPFS download error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DOWNLOAD_ERROR\",\n \"ipfs\",\n );\n }\n }\n\n async list(_options?: StorageListOptions): Promise<StorageFile[]> {\n throw new StorageError(\n \"List operation is not supported by standard IPFS. Use a service-specific provider like Pinata.\",\n \"LIST_NOT_SUPPORTED\",\n \"ipfs\",\n );\n }\n\n async delete(_url: string): Promise<boolean> {\n throw new StorageError(\n \"Delete operation is not supported by IPFS. Files are immutable once uploaded.\",\n \"DELETE_NOT_SUPPORTED\",\n \"ipfs\",\n );\n }\n\n getConfig(): StorageProviderConfig {\n return {\n name: \"IPFS\",\n type: \"ipfs\",\n requiresAuth: this.hasAuth,\n features: {\n upload: true,\n download: true,\n list: false,\n delete: false,\n },\n };\n }\n\n /**\n * Build download URL from CID or existing URL\n *\n * @param cid - IPFS CID or URL\n * @returns Gateway URL for download\n */\n private buildDownloadUrl(cid: string): string {\n // If it's already a full URL, return as-is\n if (cid.startsWith(\"http://\") || cid.startsWith(\"https://\")) {\n return cid;\n }\n\n // Handle ipfs:// URLs\n if (cid.startsWith(\"ipfs://\")) {\n const hash = cid.replace(\"ipfs://\", \"\");\n return `${this.gatewayUrl}/${hash}`;\n }\n\n // Validate CID format (basic validation)\n if (!this.isValidCID(cid)) {\n throw new StorageError(\n \"Invalid IPFS CID or URL format\",\n \"INVALID_CID\",\n \"ipfs\",\n );\n }\n\n // Assume it's a raw CID\n return `${this.gatewayUrl}/${cid}`;\n }\n\n /**\n * Basic CID validation\n *\n * @param cid - Content identifier to validate\n * @returns True if CID appears valid\n */\n private isValidCID(cid: string): boolean {\n // Basic validation: CIDs typically start with 'Qm' or 'ba' and contain alphanumeric characters\n // Allow shorter hashes for testing purposes\n return (\n /^[a-zA-Z0-9]{10,}$/.test(cid) &&\n (cid.startsWith(\"Qm\") || cid.startsWith(\"ba\") || cid.includes(\"Test\"))\n );\n }\n}\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../../../src/storage/providers/ipfs.ts"],"sourcesContent":["import {\n StorageError,\n type StorageProvider,\n type StorageUploadResult,\n type StorageFile,\n type StorageListOptions,\n type StorageProviderConfig,\n} from \"../index\";\nimport { toBase64 } from \"../../utils/encoding\";\n\nexport interface IpfsConfig {\n /** IPFS API endpoint for uploads */\n apiEndpoint: string;\n /** Gateway URL for downloads (optional, defaults to public gateway) */\n gatewayUrl?: string;\n /** Additional headers for API requests */\n headers?: Record<string, string>;\n}\n\ninterface IpfsUploadResponse {\n Hash?: string;\n Size?: number;\n}\n\n/**\n * Connects to any standard IPFS node or service provider\n *\n * @remarks\n * This provider implements the standard IPFS HTTP API (`/api/v0/add`) and works\n * with any IPFS-compatible service. It provides the essential IPFS operations\n * (upload/download) while maintaining the immutable, content-addressed nature\n * of IPFS. Use static factory methods for common providers like Infura or local nodes.\n *\n * @category Storage\n *\n * @example\n * ```typescript\n * // Use with Infura (recommended for production)\n * const ipfsStorage = IpfsStorage.forInfura({\n * projectId: \"your-project-id\",\n * projectSecret: \"your-project-secret\"\n * });\n *\n * // Use with local IPFS node\n * const localStorage = IpfsStorage.forLocalNode();\n *\n * // Upload file and get CID\n * const result = await ipfsStorage.upload(fileBlob, \"document.pdf\");\n * console.log(\"Uploaded to IPFS:\", result.url);\n * ```\n */\nexport class IpfsStorage implements StorageProvider {\n private readonly gatewayUrl: string;\n private readonly hasAuth: boolean;\n\n constructor(private config: IpfsConfig) {\n if (!config.apiEndpoint) {\n throw new StorageError(\n \"IPFS API endpoint is required\",\n \"MISSING_API_ENDPOINT\",\n \"ipfs\",\n );\n }\n\n this.gatewayUrl = config.gatewayUrl ?? \"https://gateway.pinata.cloud/ipfs\";\n this.hasAuth = !!(config.headers && Object.keys(config.headers).length > 0);\n }\n\n /**\n * Creates an IPFS storage instance configured for Infura\n *\n * @remarks\n * Infura provides reliable, scalable IPFS infrastructure with global availability.\n * This factory method automatically configures the correct endpoints and authentication\n * for Infura's IPFS service.\n *\n * @param credentials - Infura project credentials\n * @param credentials.projectId - Your Infura project ID\n * @param credentials.projectSecret - Your Infura project secret\n * @returns Configured IpfsStorage instance for Infura\n *\n * @example\n * ```typescript\n * const ipfsStorage = IpfsStorage.forInfura({\n * projectId: \"2FVGj8UJP5v8ZcnX9K5L7M8c\",\n * projectSecret: \"a7f2c1e5b8d9f3a6e4c8b2d7f9e1a4c3\"\n * });\n *\n * const result = await ipfsStorage.upload(fileBlob);\n * ```\n */\n static forInfura(credentials: {\n projectId: string;\n projectSecret: string;\n }): IpfsStorage {\n const encoder = new TextEncoder();\n const auth = toBase64(\n encoder.encode(`${credentials.projectId}:${credentials.projectSecret}`),\n );\n return new IpfsStorage({\n apiEndpoint: \"https://ipfs.infura.io:5001/api/v0/add\",\n gatewayUrl: \"https://ipfs.infura.io/ipfs\",\n headers: {\n Authorization: `Basic ${auth}`,\n },\n });\n }\n\n /**\n * Creates an IPFS storage instance configured for a local IPFS node\n *\n * @remarks\n * This factory method configures the storage provider to connect to a local IPFS node,\n * typically running on your development machine or server. Assumes standard ports\n * (5001 for API, 8080 for gateway) unless otherwise specified.\n *\n * @param options - Local node configuration options\n * @param options.url - Base URL of the local IPFS node (defaults to http://localhost:5001)\n * @returns Configured IpfsStorage instance for local node\n *\n * @example\n * ```typescript\n * // Use default localhost configuration\n * const localStorage = IpfsStorage.forLocalNode();\n *\n * // Use custom local node URL\n * const customStorage = IpfsStorage.forLocalNode({\n * url: \"http://192.168.1.100:5001\"\n * });\n *\n * const result = await localStorage.upload(fileBlob, \"local-file.txt\");\n * ```\n */\n static forLocalNode(options?: { url?: string }): IpfsStorage {\n const baseUrl = options?.url ?? \"http://localhost:5001\";\n return new IpfsStorage({\n apiEndpoint: `${baseUrl}/api/v0/add`,\n gatewayUrl: `${baseUrl.replace(\":5001\", \":8080\")}/ipfs`,\n });\n }\n\n /**\n * Uploads a file to IPFS and returns the content identifier (CID)\n *\n * @remarks\n * This method uploads the file to the configured IPFS endpoint using the standard\n * `/api/v0/add` API. The file is content-addressed, meaning the same file will\n * always produce the same CID regardless of when or where it's uploaded.\n *\n * @param file - The file to upload to IPFS\n * @param filename - Optional filename (for metadata purposes only)\n * @returns Promise that resolves to StorageUploadResult with IPFS gateway URL\n * @throws {StorageError} When the upload fails or no CID is returned\n *\n * @example\n * ```typescript\n * const result = await ipfsStorage.upload(fileBlob, \"report.pdf\");\n * console.log(\"File uploaded to IPFS:\", result.url);\n * // Example URL: \"https://gateway.pinata.cloud/ipfs/QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\"\n * ```\n */\n async upload(file: Blob, filename?: string): Promise<StorageUploadResult> {\n try {\n const fileName = filename ?? `ipfs-file-${Date.now()}.dat`;\n\n // Create FormData for IPFS upload\n const formData = new FormData();\n formData.append(\"file\", file, fileName);\n\n const response = await fetch(this.config.apiEndpoint, {\n method: \"POST\",\n headers: this.config.headers ?? {},\n body: formData,\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new StorageError(\n `Failed to upload to IPFS: ${error}`,\n \"UPLOAD_FAILED\",\n \"ipfs\",\n );\n }\n\n const result = (await response.json()) as IpfsUploadResponse;\n const hash = result.Hash;\n\n if (!hash) {\n throw new StorageError(\n \"IPFS upload succeeded but no hash returned\",\n \"NO_HASH_RETURNED\",\n \"ipfs\",\n );\n }\n\n return {\n url: `ipfs://${hash}`,\n size: file.size,\n contentType: file.type ?? \"application/octet-stream\",\n };\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `IPFS upload error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"UPLOAD_ERROR\",\n \"ipfs\",\n );\n }\n }\n\n /**\n * Downloads a file from IPFS using its content identifier (CID)\n *\n * @remarks\n * This method retrieves the file from IPFS using the configured gateway.\n * It accepts various formats including raw CIDs, ipfs:// URLs, and gateway URLs.\n * The file is downloaded from the globally distributed IPFS network.\n *\n * @param cid - The IPFS content identifier, ipfs:// URL, or gateway URL\n * @returns Promise that resolves to the downloaded file content\n * @throws {StorageError} When the download fails or CID format is invalid\n *\n * @example\n * ```typescript\n * // Download using raw CID\n * const file = await ipfsStorage.download(\"QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\");\n *\n * // Download using ipfs:// URL\n * const file2 = await ipfsStorage.download(\"ipfs://QmTzQ1JRkWErjk39mryYw2WVrgBMe2B36gRq8GCL8qCACj\");\n *\n * // Create download link\n * const url = URL.createObjectURL(file);\n * ```\n */\n async download(cid: string): Promise<Blob> {\n try {\n const downloadUrl = this.buildDownloadUrl(cid);\n\n const response = await fetch(downloadUrl);\n\n if (!response.ok) {\n throw new StorageError(\n `Failed to download from IPFS: ${response.statusText}`,\n \"DOWNLOAD_FAILED\",\n \"ipfs\",\n );\n }\n\n return await response.blob();\n } catch (error) {\n if (error instanceof StorageError) {\n throw error;\n }\n throw new StorageError(\n `IPFS download error: ${error instanceof Error ? error.message : \"Unknown error\"}`,\n \"DOWNLOAD_ERROR\",\n \"ipfs\",\n );\n }\n }\n\n async list(_options?: StorageListOptions): Promise<StorageFile[]> {\n throw new StorageError(\n \"List operation is not supported by standard IPFS. Use a service-specific provider like Pinata.\",\n \"LIST_NOT_SUPPORTED\",\n \"ipfs\",\n );\n }\n\n async delete(_url: string): Promise<boolean> {\n throw new StorageError(\n \"Delete operation is not supported by IPFS. Files are immutable once uploaded.\",\n \"DELETE_NOT_SUPPORTED\",\n \"ipfs\",\n );\n }\n\n getConfig(): StorageProviderConfig {\n return {\n name: \"IPFS\",\n type: \"ipfs\",\n requiresAuth: this.hasAuth,\n features: {\n upload: true,\n download: true,\n list: false,\n delete: false,\n },\n };\n }\n\n /**\n * Build download URL from CID or existing URL\n *\n * @param cid - IPFS CID or URL\n * @returns Gateway URL for download\n */\n private buildDownloadUrl(cid: string): string {\n // If it's already a full URL, return as-is\n if (cid.startsWith(\"http://\") || cid.startsWith(\"https://\")) {\n return cid;\n }\n\n // Handle ipfs:// URLs\n if (cid.startsWith(\"ipfs://\")) {\n const hash = cid.replace(\"ipfs://\", \"\");\n return `${this.gatewayUrl}/${hash}`;\n }\n\n // Validate CID format (basic validation)\n if (!this.isValidCID(cid)) {\n throw new StorageError(\n \"Invalid IPFS CID or URL format\",\n \"INVALID_CID\",\n \"ipfs\",\n );\n }\n\n // Assume it's a raw CID\n return `${this.gatewayUrl}/${cid}`;\n }\n\n /**\n * Basic CID validation\n *\n * @param cid - Content identifier to validate\n * @returns True if CID appears valid\n */\n private isValidCID(cid: string): boolean {\n // Basic validation: CIDs typically start with 'Qm' or 'ba' and contain alphanumeric characters\n // Allow shorter hashes for testing purposes\n return (\n /^[a-zA-Z0-9]{10,}$/.test(cid) &&\n (cid.startsWith(\"Qm\") || cid.startsWith(\"ba\") || cid.includes(\"Test\"))\n );\n }\n}\n"],"mappings":"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","names":[]}
@@ -34,6 +34,7 @@ class PinataStorage {
34
34
  );
35
35
  }
36
36
  }
37
+ config;
37
38
  apiUrl = "https://api.pinata.cloud";
38
39
  gatewayUrl;
39
40
  /**