@hammadj/better-auth-core 1.5.0-beta.9

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 (353) hide show
  1. package/.turbo/turbo-build.log +266 -0
  2. package/.turbo/turbo-test.log +2 -0
  3. package/LICENSE.md +20 -0
  4. package/dist/api/index.d.mts +181 -0
  5. package/dist/api/index.mjs +34 -0
  6. package/dist/api/index.mjs.map +1 -0
  7. package/dist/async_hooks/index.d.mts +7 -0
  8. package/dist/async_hooks/index.mjs +22 -0
  9. package/dist/async_hooks/index.mjs.map +1 -0
  10. package/dist/async_hooks/pure.index.d.mts +7 -0
  11. package/dist/async_hooks/pure.index.mjs +35 -0
  12. package/dist/async_hooks/pure.index.mjs.map +1 -0
  13. package/dist/context/endpoint-context.d.mts +19 -0
  14. package/dist/context/endpoint-context.mjs +32 -0
  15. package/dist/context/endpoint-context.mjs.map +1 -0
  16. package/dist/context/global.d.mts +7 -0
  17. package/dist/context/global.mjs +38 -0
  18. package/dist/context/global.mjs.map +1 -0
  19. package/dist/context/index.d.mts +5 -0
  20. package/dist/context/index.mjs +6 -0
  21. package/dist/context/request-state.d.mts +26 -0
  22. package/dist/context/request-state.mjs +50 -0
  23. package/dist/context/request-state.mjs.map +1 -0
  24. package/dist/context/transaction.d.mts +25 -0
  25. package/dist/context/transaction.mjs +96 -0
  26. package/dist/context/transaction.mjs.map +1 -0
  27. package/dist/db/adapter/factory.d.mts +28 -0
  28. package/dist/db/adapter/factory.mjs +716 -0
  29. package/dist/db/adapter/factory.mjs.map +1 -0
  30. package/dist/db/adapter/get-default-field-name.d.mts +19 -0
  31. package/dist/db/adapter/get-default-field-name.mjs +39 -0
  32. package/dist/db/adapter/get-default-field-name.mjs.map +1 -0
  33. package/dist/db/adapter/get-default-model-name.d.mts +13 -0
  34. package/dist/db/adapter/get-default-model-name.mjs +33 -0
  35. package/dist/db/adapter/get-default-model-name.mjs.map +1 -0
  36. package/dist/db/adapter/get-field-attributes.d.mts +30 -0
  37. package/dist/db/adapter/get-field-attributes.mjs +40 -0
  38. package/dist/db/adapter/get-field-attributes.mjs.map +1 -0
  39. package/dist/db/adapter/get-field-name.d.mts +19 -0
  40. package/dist/db/adapter/get-field-name.mjs +34 -0
  41. package/dist/db/adapter/get-field-name.mjs.map +1 -0
  42. package/dist/db/adapter/get-id-field.d.mts +40 -0
  43. package/dist/db/adapter/get-id-field.mjs +68 -0
  44. package/dist/db/adapter/get-id-field.mjs.map +1 -0
  45. package/dist/db/adapter/get-model-name.d.mts +13 -0
  46. package/dist/db/adapter/get-model-name.mjs +24 -0
  47. package/dist/db/adapter/get-model-name.mjs.map +1 -0
  48. package/dist/db/adapter/index.d.mts +515 -0
  49. package/dist/db/adapter/index.mjs +10 -0
  50. package/dist/db/adapter/types.d.mts +140 -0
  51. package/dist/db/adapter/utils.d.mts +8 -0
  52. package/dist/db/adapter/utils.mjs +39 -0
  53. package/dist/db/adapter/utils.mjs.map +1 -0
  54. package/dist/db/get-tables.d.mts +9 -0
  55. package/dist/db/get-tables.mjs +267 -0
  56. package/dist/db/get-tables.mjs.map +1 -0
  57. package/dist/db/index.d.mts +10 -0
  58. package/dist/db/index.mjs +9 -0
  59. package/dist/db/plugin.d.mts +13 -0
  60. package/dist/db/schema/account.d.mts +27 -0
  61. package/dist/db/schema/account.mjs +20 -0
  62. package/dist/db/schema/account.mjs.map +1 -0
  63. package/dist/db/schema/rate-limit.d.mts +15 -0
  64. package/dist/db/schema/rate-limit.mjs +12 -0
  65. package/dist/db/schema/rate-limit.mjs.map +1 -0
  66. package/dist/db/schema/session.d.mts +22 -0
  67. package/dist/db/schema/session.mjs +15 -0
  68. package/dist/db/schema/session.mjs.map +1 -0
  69. package/dist/db/schema/shared.d.mts +11 -0
  70. package/dist/db/schema/shared.mjs +12 -0
  71. package/dist/db/schema/shared.mjs.map +1 -0
  72. package/dist/db/schema/user.d.mts +21 -0
  73. package/dist/db/schema/user.mjs +14 -0
  74. package/dist/db/schema/user.mjs.map +1 -0
  75. package/dist/db/schema/verification.d.mts +20 -0
  76. package/dist/db/schema/verification.mjs +13 -0
  77. package/dist/db/schema/verification.mjs.map +1 -0
  78. package/dist/db/type.d.mts +147 -0
  79. package/dist/env/color-depth.d.mts +5 -0
  80. package/dist/env/color-depth.mjs +89 -0
  81. package/dist/env/color-depth.mjs.map +1 -0
  82. package/dist/env/env-impl.d.mts +33 -0
  83. package/dist/env/env-impl.mjs +83 -0
  84. package/dist/env/env-impl.mjs.map +1 -0
  85. package/dist/env/index.d.mts +4 -0
  86. package/dist/env/index.mjs +5 -0
  87. package/dist/env/logger.d.mts +49 -0
  88. package/dist/env/logger.mjs +82 -0
  89. package/dist/env/logger.mjs.map +1 -0
  90. package/dist/error/codes.d.mts +199 -0
  91. package/dist/error/codes.mjs +57 -0
  92. package/dist/error/codes.mjs.map +1 -0
  93. package/dist/error/index.d.mts +20 -0
  94. package/dist/error/index.mjs +30 -0
  95. package/dist/error/index.mjs.map +1 -0
  96. package/dist/index.d.mts +8 -0
  97. package/dist/index.mjs +1 -0
  98. package/dist/oauth2/client-credentials-token.d.mts +37 -0
  99. package/dist/oauth2/client-credentials-token.mjs +55 -0
  100. package/dist/oauth2/client-credentials-token.mjs.map +1 -0
  101. package/dist/oauth2/create-authorization-url.d.mts +46 -0
  102. package/dist/oauth2/create-authorization-url.mjs +43 -0
  103. package/dist/oauth2/create-authorization-url.mjs.map +1 -0
  104. package/dist/oauth2/index.d.mts +8 -0
  105. package/dist/oauth2/index.mjs +8 -0
  106. package/dist/oauth2/oauth-provider.d.mts +195 -0
  107. package/dist/oauth2/refresh-access-token.d.mts +36 -0
  108. package/dist/oauth2/refresh-access-token.mjs +59 -0
  109. package/dist/oauth2/refresh-access-token.mjs.map +1 -0
  110. package/dist/oauth2/utils.d.mts +8 -0
  111. package/dist/oauth2/utils.mjs +28 -0
  112. package/dist/oauth2/utils.mjs.map +1 -0
  113. package/dist/oauth2/validate-authorization-code.d.mts +56 -0
  114. package/dist/oauth2/validate-authorization-code.mjs +72 -0
  115. package/dist/oauth2/validate-authorization-code.mjs.map +1 -0
  116. package/dist/oauth2/verify.d.mts +43 -0
  117. package/dist/oauth2/verify.mjs +96 -0
  118. package/dist/oauth2/verify.mjs.map +1 -0
  119. package/dist/social-providers/apple.d.mts +120 -0
  120. package/dist/social-providers/apple.mjs +105 -0
  121. package/dist/social-providers/apple.mjs.map +1 -0
  122. package/dist/social-providers/atlassian.d.mts +73 -0
  123. package/dist/social-providers/atlassian.mjs +84 -0
  124. package/dist/social-providers/atlassian.mjs.map +1 -0
  125. package/dist/social-providers/cognito.d.mts +88 -0
  126. package/dist/social-providers/cognito.mjs +166 -0
  127. package/dist/social-providers/cognito.mjs.map +1 -0
  128. package/dist/social-providers/discord.d.mts +127 -0
  129. package/dist/social-providers/discord.mjs +65 -0
  130. package/dist/social-providers/discord.mjs.map +1 -0
  131. package/dist/social-providers/dropbox.d.mts +72 -0
  132. package/dist/social-providers/dropbox.mjs +76 -0
  133. package/dist/social-providers/dropbox.mjs.map +1 -0
  134. package/dist/social-providers/facebook.d.mts +82 -0
  135. package/dist/social-providers/facebook.mjs +121 -0
  136. package/dist/social-providers/facebook.mjs.map +1 -0
  137. package/dist/social-providers/figma.d.mts +64 -0
  138. package/dist/social-providers/figma.mjs +87 -0
  139. package/dist/social-providers/figma.mjs.map +1 -0
  140. package/dist/social-providers/github.d.mts +105 -0
  141. package/dist/social-providers/github.mjs +97 -0
  142. package/dist/social-providers/github.mjs.map +1 -0
  143. package/dist/social-providers/gitlab.d.mts +126 -0
  144. package/dist/social-providers/gitlab.mjs +83 -0
  145. package/dist/social-providers/gitlab.mjs.map +1 -0
  146. package/dist/social-providers/google.d.mts +100 -0
  147. package/dist/social-providers/google.mjs +109 -0
  148. package/dist/social-providers/google.mjs.map +1 -0
  149. package/dist/social-providers/huggingface.d.mts +86 -0
  150. package/dist/social-providers/huggingface.mjs +76 -0
  151. package/dist/social-providers/huggingface.mjs.map +1 -0
  152. package/dist/social-providers/index.d.mts +1725 -0
  153. package/dist/social-providers/index.mjs +77 -0
  154. package/dist/social-providers/index.mjs.map +1 -0
  155. package/dist/social-providers/kakao.d.mts +164 -0
  156. package/dist/social-providers/kakao.mjs +73 -0
  157. package/dist/social-providers/kakao.mjs.map +1 -0
  158. package/dist/social-providers/kick.d.mts +76 -0
  159. package/dist/social-providers/kick.mjs +72 -0
  160. package/dist/social-providers/kick.mjs.map +1 -0
  161. package/dist/social-providers/line.d.mts +108 -0
  162. package/dist/social-providers/line.mjs +114 -0
  163. package/dist/social-providers/line.mjs.map +1 -0
  164. package/dist/social-providers/linear.d.mts +71 -0
  165. package/dist/social-providers/linear.mjs +89 -0
  166. package/dist/social-providers/linear.mjs.map +1 -0
  167. package/dist/social-providers/linkedin.d.mts +70 -0
  168. package/dist/social-providers/linkedin.mjs +77 -0
  169. package/dist/social-providers/linkedin.mjs.map +1 -0
  170. package/dist/social-providers/microsoft-entra-id.d.mts +175 -0
  171. package/dist/social-providers/microsoft-entra-id.mjs +107 -0
  172. package/dist/social-providers/microsoft-entra-id.mjs.map +1 -0
  173. package/dist/social-providers/naver.d.mts +95 -0
  174. package/dist/social-providers/naver.mjs +68 -0
  175. package/dist/social-providers/naver.mjs.map +1 -0
  176. package/dist/social-providers/notion.d.mts +67 -0
  177. package/dist/social-providers/notion.mjs +76 -0
  178. package/dist/social-providers/notion.mjs.map +1 -0
  179. package/dist/social-providers/paybin.d.mts +74 -0
  180. package/dist/social-providers/paybin.mjs +86 -0
  181. package/dist/social-providers/paybin.mjs.map +1 -0
  182. package/dist/social-providers/paypal.d.mts +132 -0
  183. package/dist/social-providers/paypal.mjs +145 -0
  184. package/dist/social-providers/paypal.mjs.map +1 -0
  185. package/dist/social-providers/polar.d.mts +77 -0
  186. package/dist/social-providers/polar.mjs +74 -0
  187. package/dist/social-providers/polar.mjs.map +1 -0
  188. package/dist/social-providers/reddit.d.mts +65 -0
  189. package/dist/social-providers/reddit.mjs +84 -0
  190. package/dist/social-providers/reddit.mjs.map +1 -0
  191. package/dist/social-providers/roblox.d.mts +73 -0
  192. package/dist/social-providers/roblox.mjs +60 -0
  193. package/dist/social-providers/roblox.mjs.map +1 -0
  194. package/dist/social-providers/salesforce.d.mts +82 -0
  195. package/dist/social-providers/salesforce.mjs +92 -0
  196. package/dist/social-providers/salesforce.mjs.map +1 -0
  197. package/dist/social-providers/slack.d.mts +86 -0
  198. package/dist/social-providers/slack.mjs +69 -0
  199. package/dist/social-providers/slack.mjs.map +1 -0
  200. package/dist/social-providers/spotify.d.mts +66 -0
  201. package/dist/social-providers/spotify.mjs +72 -0
  202. package/dist/social-providers/spotify.mjs.map +1 -0
  203. package/dist/social-providers/tiktok.d.mts +171 -0
  204. package/dist/social-providers/tiktok.mjs +63 -0
  205. package/dist/social-providers/tiktok.mjs.map +1 -0
  206. package/dist/social-providers/twitch.d.mts +82 -0
  207. package/dist/social-providers/twitch.mjs +79 -0
  208. package/dist/social-providers/twitch.mjs.map +1 -0
  209. package/dist/social-providers/twitter.d.mts +129 -0
  210. package/dist/social-providers/twitter.mjs +88 -0
  211. package/dist/social-providers/twitter.mjs.map +1 -0
  212. package/dist/social-providers/vercel.d.mts +65 -0
  213. package/dist/social-providers/vercel.mjs +62 -0
  214. package/dist/social-providers/vercel.mjs.map +1 -0
  215. package/dist/social-providers/vk.d.mts +73 -0
  216. package/dist/social-providers/vk.mjs +84 -0
  217. package/dist/social-providers/vk.mjs.map +1 -0
  218. package/dist/social-providers/zoom.d.mts +173 -0
  219. package/dist/social-providers/zoom.mjs +73 -0
  220. package/dist/social-providers/zoom.mjs.map +1 -0
  221. package/dist/types/context.d.mts +267 -0
  222. package/dist/types/cookie.d.mts +16 -0
  223. package/dist/types/helper.d.mts +10 -0
  224. package/dist/types/index.d.mts +8 -0
  225. package/dist/types/init-options.d.mts +1314 -0
  226. package/dist/types/plugin-client.d.mts +112 -0
  227. package/dist/types/plugin.d.mts +125 -0
  228. package/dist/utils/db.d.mts +12 -0
  229. package/dist/utils/db.mjs +17 -0
  230. package/dist/utils/db.mjs.map +1 -0
  231. package/dist/utils/deprecate.d.mts +10 -0
  232. package/dist/utils/deprecate.mjs +18 -0
  233. package/dist/utils/deprecate.mjs.map +1 -0
  234. package/dist/utils/error-codes.d.mts +13 -0
  235. package/dist/utils/error-codes.mjs +12 -0
  236. package/dist/utils/error-codes.mjs.map +1 -0
  237. package/dist/utils/id.d.mts +5 -0
  238. package/dist/utils/id.mjs +10 -0
  239. package/dist/utils/id.mjs.map +1 -0
  240. package/dist/utils/ip.d.mts +55 -0
  241. package/dist/utils/ip.mjs +119 -0
  242. package/dist/utils/ip.mjs.map +1 -0
  243. package/dist/utils/json.d.mts +5 -0
  244. package/dist/utils/json.mjs +26 -0
  245. package/dist/utils/json.mjs.map +1 -0
  246. package/dist/utils/string.d.mts +5 -0
  247. package/dist/utils/string.mjs +8 -0
  248. package/dist/utils/string.mjs.map +1 -0
  249. package/dist/utils/url.d.mts +21 -0
  250. package/dist/utils/url.mjs +33 -0
  251. package/dist/utils/url.mjs.map +1 -0
  252. package/package.json +147 -0
  253. package/src/api/index.ts +106 -0
  254. package/src/async_hooks/index.ts +40 -0
  255. package/src/async_hooks/pure.index.ts +46 -0
  256. package/src/context/endpoint-context.ts +50 -0
  257. package/src/context/global.ts +57 -0
  258. package/src/context/index.ts +23 -0
  259. package/src/context/request-state.test.ts +94 -0
  260. package/src/context/request-state.ts +91 -0
  261. package/src/context/transaction.ts +136 -0
  262. package/src/db/adapter/factory.ts +1362 -0
  263. package/src/db/adapter/get-default-field-name.ts +59 -0
  264. package/src/db/adapter/get-default-model-name.ts +51 -0
  265. package/src/db/adapter/get-field-attributes.ts +62 -0
  266. package/src/db/adapter/get-field-name.ts +43 -0
  267. package/src/db/adapter/get-id-field.ts +141 -0
  268. package/src/db/adapter/get-model-name.ts +36 -0
  269. package/src/db/adapter/index.ts +554 -0
  270. package/src/db/adapter/types.ts +171 -0
  271. package/src/db/adapter/utils.ts +61 -0
  272. package/src/db/get-tables.ts +296 -0
  273. package/src/db/index.ts +18 -0
  274. package/src/db/plugin.ts +11 -0
  275. package/src/db/schema/account.ts +34 -0
  276. package/src/db/schema/rate-limit.ts +21 -0
  277. package/src/db/schema/session.ts +17 -0
  278. package/src/db/schema/shared.ts +7 -0
  279. package/src/db/schema/user.ts +16 -0
  280. package/src/db/schema/verification.ts +15 -0
  281. package/src/db/test/get-tables.test.ts +116 -0
  282. package/src/db/type.ts +180 -0
  283. package/src/env/color-depth.ts +172 -0
  284. package/src/env/env-impl.ts +124 -0
  285. package/src/env/index.ts +23 -0
  286. package/src/env/logger.test.ts +34 -0
  287. package/src/env/logger.ts +145 -0
  288. package/src/error/codes.ts +58 -0
  289. package/src/error/index.ts +35 -0
  290. package/src/index.ts +1 -0
  291. package/src/oauth2/client-credentials-token.ts +102 -0
  292. package/src/oauth2/create-authorization-url.ts +87 -0
  293. package/src/oauth2/index.ts +26 -0
  294. package/src/oauth2/oauth-provider.ts +222 -0
  295. package/src/oauth2/refresh-access-token.ts +124 -0
  296. package/src/oauth2/utils.ts +38 -0
  297. package/src/oauth2/validate-authorization-code.ts +149 -0
  298. package/src/oauth2/validate-token.test.ts +174 -0
  299. package/src/oauth2/verify.ts +221 -0
  300. package/src/social-providers/apple.ts +223 -0
  301. package/src/social-providers/atlassian.ts +132 -0
  302. package/src/social-providers/cognito.ts +279 -0
  303. package/src/social-providers/discord.ts +169 -0
  304. package/src/social-providers/dropbox.ts +112 -0
  305. package/src/social-providers/facebook.ts +206 -0
  306. package/src/social-providers/figma.ts +117 -0
  307. package/src/social-providers/github.ts +184 -0
  308. package/src/social-providers/gitlab.ts +155 -0
  309. package/src/social-providers/google.ts +199 -0
  310. package/src/social-providers/huggingface.ts +118 -0
  311. package/src/social-providers/index.ts +127 -0
  312. package/src/social-providers/kakao.ts +178 -0
  313. package/src/social-providers/kick.ts +109 -0
  314. package/src/social-providers/line.ts +169 -0
  315. package/src/social-providers/linear.ts +121 -0
  316. package/src/social-providers/linkedin.ts +110 -0
  317. package/src/social-providers/microsoft-entra-id.ts +259 -0
  318. package/src/social-providers/naver.ts +112 -0
  319. package/src/social-providers/notion.ts +108 -0
  320. package/src/social-providers/paybin.ts +122 -0
  321. package/src/social-providers/paypal.ts +263 -0
  322. package/src/social-providers/polar.ts +110 -0
  323. package/src/social-providers/reddit.ts +122 -0
  324. package/src/social-providers/roblox.ts +111 -0
  325. package/src/social-providers/salesforce.ts +159 -0
  326. package/src/social-providers/slack.ts +111 -0
  327. package/src/social-providers/spotify.ts +93 -0
  328. package/src/social-providers/tiktok.ts +209 -0
  329. package/src/social-providers/twitch.ts +111 -0
  330. package/src/social-providers/twitter.ts +198 -0
  331. package/src/social-providers/vercel.ts +87 -0
  332. package/src/social-providers/vk.ts +124 -0
  333. package/src/social-providers/zoom.ts +238 -0
  334. package/src/types/context.ts +396 -0
  335. package/src/types/cookie.ts +10 -0
  336. package/src/types/helper.ts +26 -0
  337. package/src/types/index.ts +32 -0
  338. package/src/types/init-options.ts +1529 -0
  339. package/src/types/plugin-client.ts +127 -0
  340. package/src/types/plugin.ts +157 -0
  341. package/src/utils/db.ts +20 -0
  342. package/src/utils/deprecate.test.ts +72 -0
  343. package/src/utils/deprecate.ts +21 -0
  344. package/src/utils/error-codes.ts +65 -0
  345. package/src/utils/id.ts +5 -0
  346. package/src/utils/ip.test.ts +255 -0
  347. package/src/utils/ip.ts +211 -0
  348. package/src/utils/json.ts +25 -0
  349. package/src/utils/string.ts +3 -0
  350. package/src/utils/url.ts +43 -0
  351. package/tsconfig.json +7 -0
  352. package/tsdown.config.ts +35 -0
  353. package/vitest.config.ts +3 -0
@@ -0,0 +1,169 @@
1
+ import { betterFetch } from "@better-fetch/fetch";
2
+ import type { OAuthProvider, ProviderOptions } from "../oauth2";
3
+ import { refreshAccessToken, validateAuthorizationCode } from "../oauth2";
4
+ export interface DiscordProfile extends Record<string, any> {
5
+ /** the user's id (i.e. the numerical snowflake) */
6
+ id: string;
7
+ /** the user's username, not unique across the platform */
8
+ username: string;
9
+ /** the user's Discord-tag */
10
+ discriminator: string;
11
+ /** the user's display name, if it is set */
12
+ global_name: string | null;
13
+ /**
14
+ * the user's avatar hash:
15
+ * https://discord.com/developers/docs/reference#image-formatting
16
+ */
17
+ avatar: string | null;
18
+ /** whether the user belongs to an OAuth2 application */
19
+ bot?: boolean | undefined;
20
+ /**
21
+ * whether the user is an Official Discord System user (part of the urgent
22
+ * message system)
23
+ */
24
+ system?: boolean | undefined;
25
+ /** whether the user has two factor enabled on their account */
26
+ mfa_enabled: boolean;
27
+ /**
28
+ * the user's banner hash:
29
+ * https://discord.com/developers/docs/reference#image-formatting
30
+ */
31
+ banner: string | null;
32
+
33
+ /** the user's banner color encoded as an integer representation of hexadecimal color code */
34
+ accent_color: number | null;
35
+
36
+ /**
37
+ * the user's chosen language option:
38
+ * https://discord.com/developers/docs/reference#locales
39
+ */
40
+ locale: string;
41
+ /** whether the email on this account has been verified */
42
+ verified: boolean;
43
+ /** the user's email */
44
+ email: string;
45
+ /**
46
+ * the flags on a user's account:
47
+ * https://discord.com/developers/docs/resources/user#user-object-user-flags
48
+ */
49
+ flags: number;
50
+ /**
51
+ * the type of Nitro subscription on a user's account:
52
+ * https://discord.com/developers/docs/resources/user#user-object-premium-types
53
+ */
54
+ premium_type: number;
55
+ /**
56
+ * the public flags on a user's account:
57
+ * https://discord.com/developers/docs/resources/user#user-object-user-flags
58
+ */
59
+ public_flags: number;
60
+ /** undocumented field; corresponds to the user's custom nickname */
61
+ display_name: string | null;
62
+ /**
63
+ * undocumented field; corresponds to the Discord feature where you can e.g.
64
+ * put your avatar inside of an ice cube
65
+ */
66
+ avatar_decoration: string | null;
67
+ /**
68
+ * undocumented field; corresponds to the premium feature where you can
69
+ * select a custom banner color
70
+ */
71
+ banner_color: string | null;
72
+ /** undocumented field; the CDN URL of their profile picture */
73
+ image_url: string;
74
+ }
75
+
76
+ export interface DiscordOptions extends ProviderOptions<DiscordProfile> {
77
+ clientId: string;
78
+ prompt?: ("none" | "consent") | undefined;
79
+ permissions?: number | undefined;
80
+ }
81
+
82
+ export const discord = (options: DiscordOptions) => {
83
+ return {
84
+ id: "discord",
85
+ name: "Discord",
86
+ createAuthorizationURL({ state, scopes, redirectURI }) {
87
+ const _scopes = options.disableDefaultScope ? [] : ["identify", "email"];
88
+ if (scopes) _scopes.push(...scopes);
89
+ if (options.scope) _scopes.push(...options.scope);
90
+ const hasBotScope = _scopes.includes("bot");
91
+ const permissionsParam =
92
+ hasBotScope && options.permissions !== undefined
93
+ ? `&permissions=${options.permissions}`
94
+ : "";
95
+ return new URL(
96
+ `https://discord.com/api/oauth2/authorize?scope=${_scopes.join(
97
+ "+",
98
+ )}&response_type=code&client_id=${
99
+ options.clientId
100
+ }&redirect_uri=${encodeURIComponent(
101
+ options.redirectURI || redirectURI,
102
+ )}&state=${state}&prompt=${
103
+ options.prompt || "none"
104
+ }${permissionsParam}`,
105
+ );
106
+ },
107
+ validateAuthorizationCode: async ({ code, redirectURI }) => {
108
+ return validateAuthorizationCode({
109
+ code,
110
+ redirectURI,
111
+ options,
112
+ tokenEndpoint: "https://discord.com/api/oauth2/token",
113
+ });
114
+ },
115
+ refreshAccessToken: options.refreshAccessToken
116
+ ? options.refreshAccessToken
117
+ : async (refreshToken) => {
118
+ return refreshAccessToken({
119
+ refreshToken,
120
+ options: {
121
+ clientId: options.clientId,
122
+ clientKey: options.clientKey,
123
+ clientSecret: options.clientSecret,
124
+ },
125
+ tokenEndpoint: "https://discord.com/api/oauth2/token",
126
+ });
127
+ },
128
+ async getUserInfo(token) {
129
+ if (options.getUserInfo) {
130
+ return options.getUserInfo(token);
131
+ }
132
+ const { data: profile, error } = await betterFetch<DiscordProfile>(
133
+ "https://discord.com/api/users/@me",
134
+ {
135
+ headers: {
136
+ authorization: `Bearer ${token.accessToken}`,
137
+ },
138
+ },
139
+ );
140
+
141
+ if (error) {
142
+ return null;
143
+ }
144
+ if (profile.avatar === null) {
145
+ const defaultAvatarNumber =
146
+ profile.discriminator === "0"
147
+ ? Number(BigInt(profile.id) >> BigInt(22)) % 6
148
+ : parseInt(profile.discriminator) % 5;
149
+ profile.image_url = `https://cdn.discordapp.com/embed/avatars/${defaultAvatarNumber}.png`;
150
+ } else {
151
+ const format = profile.avatar.startsWith("a_") ? "gif" : "png";
152
+ profile.image_url = `https://cdn.discordapp.com/avatars/${profile.id}/${profile.avatar}.${format}`;
153
+ }
154
+ const userMap = await options.mapProfileToUser?.(profile);
155
+ return {
156
+ user: {
157
+ id: profile.id,
158
+ name: profile.global_name || profile.username || "",
159
+ email: profile.email,
160
+ emailVerified: profile.verified,
161
+ image: profile.image_url,
162
+ ...userMap,
163
+ },
164
+ data: profile,
165
+ };
166
+ },
167
+ options,
168
+ } satisfies OAuthProvider<DiscordProfile>;
169
+ };
@@ -0,0 +1,112 @@
1
+ import { betterFetch } from "@better-fetch/fetch";
2
+ import type { OAuthProvider, ProviderOptions } from "../oauth2";
3
+ import {
4
+ createAuthorizationURL,
5
+ refreshAccessToken,
6
+ validateAuthorizationCode,
7
+ } from "../oauth2";
8
+
9
+ export interface DropboxProfile {
10
+ account_id: string;
11
+ name: {
12
+ given_name: string;
13
+ surname: string;
14
+ familiar_name: string;
15
+ display_name: string;
16
+ abbreviated_name: string;
17
+ };
18
+ email: string;
19
+ email_verified: boolean;
20
+ profile_photo_url: string;
21
+ }
22
+
23
+ export interface DropboxOptions extends ProviderOptions<DropboxProfile> {
24
+ clientId: string;
25
+ accessType?: ("offline" | "online" | "legacy") | undefined;
26
+ }
27
+
28
+ export const dropbox = (options: DropboxOptions) => {
29
+ const tokenEndpoint = "https://api.dropboxapi.com/oauth2/token";
30
+
31
+ return {
32
+ id: "dropbox",
33
+ name: "Dropbox",
34
+ createAuthorizationURL: async ({
35
+ state,
36
+ scopes,
37
+ codeVerifier,
38
+ redirectURI,
39
+ }) => {
40
+ const _scopes = options.disableDefaultScope ? [] : ["account_info.read"];
41
+ if (options.scope) _scopes.push(...options.scope);
42
+ if (scopes) _scopes.push(...scopes);
43
+ const additionalParams: Record<string, string> = {};
44
+ if (options.accessType) {
45
+ additionalParams.token_access_type = options.accessType;
46
+ }
47
+ return await createAuthorizationURL({
48
+ id: "dropbox",
49
+ options,
50
+ authorizationEndpoint: "https://www.dropbox.com/oauth2/authorize",
51
+ scopes: _scopes,
52
+ state,
53
+ redirectURI,
54
+ codeVerifier,
55
+ additionalParams,
56
+ });
57
+ },
58
+ validateAuthorizationCode: async ({ code, codeVerifier, redirectURI }) => {
59
+ return await validateAuthorizationCode({
60
+ code,
61
+ codeVerifier,
62
+ redirectURI,
63
+ options,
64
+ tokenEndpoint,
65
+ });
66
+ },
67
+ refreshAccessToken: options.refreshAccessToken
68
+ ? options.refreshAccessToken
69
+ : async (refreshToken) => {
70
+ return refreshAccessToken({
71
+ refreshToken,
72
+ options: {
73
+ clientId: options.clientId,
74
+ clientKey: options.clientKey,
75
+ clientSecret: options.clientSecret,
76
+ },
77
+ tokenEndpoint,
78
+ });
79
+ },
80
+ async getUserInfo(token) {
81
+ if (options.getUserInfo) {
82
+ return options.getUserInfo(token);
83
+ }
84
+ const { data: profile, error } = await betterFetch<DropboxProfile>(
85
+ "https://api.dropboxapi.com/2/users/get_current_account",
86
+ {
87
+ method: "POST",
88
+ headers: {
89
+ Authorization: `Bearer ${token.accessToken}`,
90
+ },
91
+ },
92
+ );
93
+
94
+ if (error) {
95
+ return null;
96
+ }
97
+ const userMap = await options.mapProfileToUser?.(profile);
98
+ return {
99
+ user: {
100
+ id: profile.account_id,
101
+ name: profile.name?.display_name,
102
+ email: profile.email,
103
+ emailVerified: profile.email_verified || false,
104
+ image: profile.profile_photo_url,
105
+ ...userMap,
106
+ },
107
+ data: profile,
108
+ };
109
+ },
110
+ options,
111
+ } satisfies OAuthProvider<DropboxProfile>;
112
+ };
@@ -0,0 +1,206 @@
1
+ import { betterFetch } from "@better-fetch/fetch";
2
+ import { createRemoteJWKSet, decodeJwt, jwtVerify } from "jose";
3
+ import type { OAuthProvider, ProviderOptions } from "../oauth2";
4
+ import {
5
+ createAuthorizationURL,
6
+ refreshAccessToken,
7
+ validateAuthorizationCode,
8
+ } from "../oauth2";
9
+ export interface FacebookProfile {
10
+ id: string;
11
+ name: string;
12
+ email: string;
13
+ email_verified: boolean;
14
+ picture: {
15
+ data: {
16
+ height: number;
17
+ is_silhouette: boolean;
18
+ url: string;
19
+ width: number;
20
+ };
21
+ };
22
+ }
23
+
24
+ export interface FacebookOptions extends ProviderOptions<FacebookProfile> {
25
+ clientId: string;
26
+ /**
27
+ * Extend list of fields to retrieve from the Facebook user profile.
28
+ *
29
+ * @default ["id", "name", "email", "picture"]
30
+ */
31
+ fields?: string[] | undefined;
32
+
33
+ /**
34
+ * The config id to use when undergoing oauth
35
+ */
36
+ configId?: string | undefined;
37
+ }
38
+
39
+ export const facebook = (options: FacebookOptions) => {
40
+ return {
41
+ id: "facebook",
42
+ name: "Facebook",
43
+ async createAuthorizationURL({ state, scopes, redirectURI, loginHint }) {
44
+ const _scopes = options.disableDefaultScope
45
+ ? []
46
+ : ["email", "public_profile"];
47
+ if (options.scope) _scopes.push(...options.scope);
48
+ if (scopes) _scopes.push(...scopes);
49
+ return await createAuthorizationURL({
50
+ id: "facebook",
51
+ options,
52
+ authorizationEndpoint: "https://www.facebook.com/v24.0/dialog/oauth",
53
+ scopes: _scopes,
54
+ state,
55
+ redirectURI,
56
+ loginHint,
57
+ additionalParams: options.configId
58
+ ? {
59
+ config_id: options.configId,
60
+ }
61
+ : {},
62
+ });
63
+ },
64
+ validateAuthorizationCode: async ({ code, redirectURI }) => {
65
+ return validateAuthorizationCode({
66
+ code,
67
+ redirectURI,
68
+ options,
69
+ tokenEndpoint: "https://graph.facebook.com/v24.0/oauth/access_token",
70
+ });
71
+ },
72
+ async verifyIdToken(token, nonce) {
73
+ if (options.disableIdTokenSignIn) {
74
+ return false;
75
+ }
76
+
77
+ if (options.verifyIdToken) {
78
+ return options.verifyIdToken(token, nonce);
79
+ }
80
+
81
+ /* limited login */
82
+ // check is limited token
83
+ if (token.split(".").length === 3) {
84
+ try {
85
+ const { payload: jwtClaims } = await jwtVerify(
86
+ token,
87
+ createRemoteJWKSet(
88
+ // https://developers.facebook.com/docs/facebook-login/limited-login/token/#jwks
89
+ new URL(
90
+ "https://limited.facebook.com/.well-known/oauth/openid/jwks/",
91
+ ),
92
+ ),
93
+ {
94
+ algorithms: ["RS256"],
95
+ audience: options.clientId,
96
+ issuer: "https://www.facebook.com",
97
+ },
98
+ );
99
+
100
+ if (nonce && jwtClaims.nonce !== nonce) {
101
+ return false;
102
+ }
103
+
104
+ return !!jwtClaims;
105
+ } catch {
106
+ return false;
107
+ }
108
+ }
109
+
110
+ /* access_token */
111
+ return true;
112
+ },
113
+ refreshAccessToken: options.refreshAccessToken
114
+ ? options.refreshAccessToken
115
+ : async (refreshToken) => {
116
+ return refreshAccessToken({
117
+ refreshToken,
118
+ options: {
119
+ clientId: options.clientId,
120
+ clientKey: options.clientKey,
121
+ clientSecret: options.clientSecret,
122
+ },
123
+ tokenEndpoint:
124
+ "https://graph.facebook.com/v24.0/oauth/access_token",
125
+ });
126
+ },
127
+ async getUserInfo(token) {
128
+ if (options.getUserInfo) {
129
+ return options.getUserInfo(token);
130
+ }
131
+
132
+ if (token.idToken && token.idToken.split(".").length === 3) {
133
+ const profile = decodeJwt(token.idToken) as {
134
+ sub: string;
135
+ email: string;
136
+ name: string;
137
+ picture: string;
138
+ };
139
+
140
+ const user = {
141
+ id: profile.sub,
142
+ name: profile.name,
143
+ email: profile.email,
144
+ picture: {
145
+ data: {
146
+ url: profile.picture,
147
+ height: 100,
148
+ width: 100,
149
+ is_silhouette: false,
150
+ },
151
+ },
152
+ };
153
+
154
+ // https://developers.facebook.com/docs/facebook-login/limited-login/permissions
155
+ // Facebook ID token does not include email_verified claim.
156
+ // We default to false for security consistency.
157
+ const userMap = await options.mapProfileToUser?.({
158
+ ...user,
159
+ email_verified: false,
160
+ });
161
+
162
+ return {
163
+ user: {
164
+ ...user,
165
+ emailVerified: false,
166
+ ...userMap,
167
+ },
168
+ data: profile,
169
+ };
170
+ }
171
+
172
+ const fields = [
173
+ "id",
174
+ "name",
175
+ "email",
176
+ "picture",
177
+ ...(options?.fields || []),
178
+ ];
179
+ const { data: profile, error } = await betterFetch<FacebookProfile>(
180
+ "https://graph.facebook.com/me?fields=" + fields.join(","),
181
+ {
182
+ auth: {
183
+ type: "Bearer",
184
+ token: token.accessToken,
185
+ },
186
+ },
187
+ );
188
+ if (error) {
189
+ return null;
190
+ }
191
+ const userMap = await options.mapProfileToUser?.(profile);
192
+ return {
193
+ user: {
194
+ id: profile.id,
195
+ name: profile.name,
196
+ email: profile.email,
197
+ image: profile.picture.data.url,
198
+ emailVerified: profile.email_verified,
199
+ ...userMap,
200
+ },
201
+ data: profile,
202
+ };
203
+ },
204
+ options,
205
+ } satisfies OAuthProvider<FacebookProfile>;
206
+ };
@@ -0,0 +1,117 @@
1
+ import { betterFetch } from "@better-fetch/fetch";
2
+ import { logger } from "../env";
3
+ import { BetterAuthError } from "../error";
4
+ import type { OAuthProvider, ProviderOptions } from "../oauth2";
5
+ import {
6
+ createAuthorizationURL,
7
+ refreshAccessToken,
8
+ validateAuthorizationCode,
9
+ } from "../oauth2";
10
+
11
+ export interface FigmaProfile {
12
+ id: string;
13
+ email: string;
14
+ handle: string;
15
+ img_url: string;
16
+ }
17
+
18
+ export interface FigmaOptions extends ProviderOptions<FigmaProfile> {
19
+ clientId: string;
20
+ }
21
+
22
+ export const figma = (options: FigmaOptions) => {
23
+ return {
24
+ id: "figma",
25
+ name: "Figma",
26
+ async createAuthorizationURL({ state, scopes, codeVerifier, redirectURI }) {
27
+ if (!options.clientId || !options.clientSecret) {
28
+ logger.error(
29
+ "Client Id and Client Secret are required for Figma. Make sure to provide them in the options.",
30
+ );
31
+ throw new BetterAuthError("CLIENT_ID_AND_SECRET_REQUIRED");
32
+ }
33
+ if (!codeVerifier) {
34
+ throw new BetterAuthError("codeVerifier is required for Figma");
35
+ }
36
+
37
+ const _scopes = options.disableDefaultScope ? [] : ["current_user:read"];
38
+ if (options.scope) _scopes.push(...options.scope);
39
+ if (scopes) _scopes.push(...scopes);
40
+
41
+ const url = await createAuthorizationURL({
42
+ id: "figma",
43
+ options,
44
+ authorizationEndpoint: "https://www.figma.com/oauth",
45
+ scopes: _scopes,
46
+ state,
47
+ codeVerifier,
48
+ redirectURI,
49
+ });
50
+
51
+ return url;
52
+ },
53
+ validateAuthorizationCode: async ({ code, codeVerifier, redirectURI }) => {
54
+ return validateAuthorizationCode({
55
+ code,
56
+ codeVerifier,
57
+ redirectURI,
58
+ options,
59
+ tokenEndpoint: "https://api.figma.com/v1/oauth/token",
60
+ authentication: "basic",
61
+ });
62
+ },
63
+ refreshAccessToken: options.refreshAccessToken
64
+ ? options.refreshAccessToken
65
+ : async (refreshToken) => {
66
+ return refreshAccessToken({
67
+ refreshToken,
68
+ options: {
69
+ clientId: options.clientId,
70
+ clientKey: options.clientKey,
71
+ clientSecret: options.clientSecret,
72
+ },
73
+ tokenEndpoint: "https://api.figma.com/v1/oauth/token",
74
+ authentication: "basic",
75
+ });
76
+ },
77
+ async getUserInfo(token) {
78
+ if (options.getUserInfo) {
79
+ return options.getUserInfo(token);
80
+ }
81
+
82
+ try {
83
+ const { data: profile } = await betterFetch<FigmaProfile>(
84
+ "https://api.figma.com/v1/me",
85
+ {
86
+ headers: {
87
+ Authorization: `Bearer ${token.accessToken}`,
88
+ },
89
+ },
90
+ );
91
+
92
+ if (!profile) {
93
+ logger.error("Failed to fetch user from Figma");
94
+ return null;
95
+ }
96
+
97
+ const userMap = await options.mapProfileToUser?.(profile);
98
+
99
+ return {
100
+ user: {
101
+ id: profile.id,
102
+ name: profile.handle,
103
+ email: profile.email,
104
+ image: profile.img_url,
105
+ emailVerified: false,
106
+ ...userMap,
107
+ },
108
+ data: profile,
109
+ };
110
+ } catch (error) {
111
+ logger.error("Failed to fetch user info from Figma:", error);
112
+ return null;
113
+ }
114
+ },
115
+ options,
116
+ } satisfies OAuthProvider<FigmaProfile>;
117
+ };