payload-auth 1.6.0 → 1.6.1-canary.1

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 (143) hide show
  1. package/dist/better-auth/adapter/index.d.ts.map +1 -1
  2. package/dist/better-auth/adapter/index.js +30 -30
  3. package/dist/better-auth/adapter/transform/index.d.ts +8 -7
  4. package/dist/better-auth/adapter/transform/index.d.ts.map +1 -1
  5. package/dist/better-auth/adapter/transform/index.js +2 -2
  6. package/dist/better-auth/adapter/types.d.ts +5 -4
  7. package/dist/better-auth/adapter/types.d.ts.map +1 -1
  8. package/dist/better-auth/adapter/types.js +1 -1
  9. package/dist/better-auth/generated-types.d.ts +21 -0
  10. package/dist/better-auth/generated-types.d.ts.map +1 -1
  11. package/dist/better-auth/generated-types.js +1 -1
  12. package/dist/better-auth/plugin/helpers/get-better-auth-schema.d.ts +14 -17
  13. package/dist/better-auth/plugin/helpers/get-better-auth-schema.d.ts.map +1 -1
  14. package/dist/better-auth/plugin/helpers/get-better-auth-schema.js +51 -33
  15. package/dist/better-auth/plugin/helpers/get-collection.d.ts +11 -36
  16. package/dist/better-auth/plugin/helpers/get-collection.d.ts.map +1 -1
  17. package/dist/better-auth/plugin/helpers/get-collection.js +20 -53
  18. package/dist/better-auth/plugin/helpers/prepare-session-data.d.ts.map +1 -1
  19. package/dist/better-auth/plugin/helpers/prepare-session-data.js +1 -1
  20. package/dist/better-auth/plugin/helpers/sync-resolved-schema-with-collection-map.d.ts +21 -0
  21. package/dist/better-auth/plugin/helpers/sync-resolved-schema-with-collection-map.d.ts.map +1 -0
  22. package/dist/better-auth/plugin/helpers/sync-resolved-schema-with-collection-map.js +62 -0
  23. package/dist/better-auth/plugin/index.d.ts +8 -1
  24. package/dist/better-auth/plugin/index.d.ts.map +1 -1
  25. package/dist/better-auth/plugin/index.js +67 -91
  26. package/dist/better-auth/plugin/lib/apply-disabled-default-auth-config.d.ts +13 -0
  27. package/dist/better-auth/plugin/lib/apply-disabled-default-auth-config.d.ts.map +1 -0
  28. package/dist/better-auth/plugin/lib/apply-disabled-default-auth-config.js +80 -0
  29. package/dist/better-auth/plugin/lib/build-collections/accounts/hooks/sync-password-to-user.d.ts +2 -2
  30. package/dist/better-auth/plugin/lib/build-collections/accounts/hooks/sync-password-to-user.d.ts.map +1 -1
  31. package/dist/better-auth/plugin/lib/build-collections/accounts/hooks/sync-password-to-user.js +6 -16
  32. package/dist/better-auth/plugin/lib/build-collections/accounts/index.d.ts +2 -2
  33. package/dist/better-auth/plugin/lib/build-collections/accounts/index.d.ts.map +1 -1
  34. package/dist/better-auth/plugin/lib/build-collections/accounts/index.js +10 -12
  35. package/dist/better-auth/plugin/lib/build-collections/api-keys.d.ts +1 -1
  36. package/dist/better-auth/plugin/lib/build-collections/api-keys.d.ts.map +1 -1
  37. package/dist/better-auth/plugin/lib/build-collections/api-keys.js +8 -11
  38. package/dist/better-auth/plugin/lib/build-collections/index.d.ts +3 -4
  39. package/dist/better-auth/plugin/lib/build-collections/index.d.ts.map +1 -1
  40. package/dist/better-auth/plugin/lib/build-collections/index.js +15 -28
  41. package/dist/better-auth/plugin/lib/build-collections/invitations.d.ts +1 -1
  42. package/dist/better-auth/plugin/lib/build-collections/invitations.d.ts.map +1 -1
  43. package/dist/better-auth/plugin/lib/build-collections/invitations.js +8 -11
  44. package/dist/better-auth/plugin/lib/build-collections/jwks.d.ts +1 -1
  45. package/dist/better-auth/plugin/lib/build-collections/jwks.d.ts.map +1 -1
  46. package/dist/better-auth/plugin/lib/build-collections/jwks.js +8 -11
  47. package/dist/better-auth/plugin/lib/build-collections/members.d.ts +1 -1
  48. package/dist/better-auth/plugin/lib/build-collections/members.d.ts.map +1 -1
  49. package/dist/better-auth/plugin/lib/build-collections/members.js +8 -10
  50. package/dist/better-auth/plugin/lib/build-collections/oauth-access-tokens.d.ts +1 -1
  51. package/dist/better-auth/plugin/lib/build-collections/oauth-access-tokens.d.ts.map +1 -1
  52. package/dist/better-auth/plugin/lib/build-collections/oauth-access-tokens.js +8 -11
  53. package/dist/better-auth/plugin/lib/build-collections/oauth-applications.d.ts +1 -1
  54. package/dist/better-auth/plugin/lib/build-collections/oauth-applications.d.ts.map +1 -1
  55. package/dist/better-auth/plugin/lib/build-collections/oauth-applications.js +8 -11
  56. package/dist/better-auth/plugin/lib/build-collections/oauth-consents.d.ts +1 -1
  57. package/dist/better-auth/plugin/lib/build-collections/oauth-consents.d.ts.map +1 -1
  58. package/dist/better-auth/plugin/lib/build-collections/oauth-consents.js +7 -9
  59. package/dist/better-auth/plugin/lib/build-collections/organizations.d.ts +1 -1
  60. package/dist/better-auth/plugin/lib/build-collections/organizations.d.ts.map +1 -1
  61. package/dist/better-auth/plugin/lib/build-collections/organizations.js +8 -10
  62. package/dist/better-auth/plugin/lib/build-collections/passkeys.d.ts +1 -1
  63. package/dist/better-auth/plugin/lib/build-collections/passkeys.d.ts.map +1 -1
  64. package/dist/better-auth/plugin/lib/build-collections/passkeys.js +9 -11
  65. package/dist/better-auth/plugin/lib/build-collections/sessions.d.ts +1 -1
  66. package/dist/better-auth/plugin/lib/build-collections/sessions.d.ts.map +1 -1
  67. package/dist/better-auth/plugin/lib/build-collections/sessions.js +7 -8
  68. package/dist/better-auth/plugin/lib/build-collections/sso-providers.d.ts +1 -1
  69. package/dist/better-auth/plugin/lib/build-collections/sso-providers.d.ts.map +1 -1
  70. package/dist/better-auth/plugin/lib/build-collections/sso-providers.js +8 -10
  71. package/dist/better-auth/plugin/lib/build-collections/subscriptions.d.ts +1 -1
  72. package/dist/better-auth/plugin/lib/build-collections/subscriptions.d.ts.map +1 -1
  73. package/dist/better-auth/plugin/lib/build-collections/subscriptions.js +8 -10
  74. package/dist/better-auth/plugin/lib/build-collections/teams.d.ts +1 -1
  75. package/dist/better-auth/plugin/lib/build-collections/teams.d.ts.map +1 -1
  76. package/dist/better-auth/plugin/lib/build-collections/teams.js +8 -11
  77. package/dist/better-auth/plugin/lib/build-collections/two-factors.d.ts +1 -1
  78. package/dist/better-auth/plugin/lib/build-collections/two-factors.d.ts.map +1 -1
  79. package/dist/better-auth/plugin/lib/build-collections/two-factors.js +8 -11
  80. package/dist/better-auth/plugin/lib/build-collections/users/hooks/after-login.d.ts.map +1 -1
  81. package/dist/better-auth/plugin/lib/build-collections/users/hooks/after-login.js +4 -11
  82. package/dist/better-auth/plugin/lib/build-collections/users/hooks/after-logout.d.ts.map +1 -1
  83. package/dist/better-auth/plugin/lib/build-collections/users/hooks/after-logout.js +3 -8
  84. package/dist/better-auth/plugin/lib/build-collections/users/hooks/before-delete.d.ts.map +1 -1
  85. package/dist/better-auth/plugin/lib/build-collections/users/hooks/before-delete.js +5 -15
  86. package/dist/better-auth/plugin/lib/build-collections/users/hooks/sync-account.d.ts.map +1 -1
  87. package/dist/better-auth/plugin/lib/build-collections/users/hooks/sync-account.js +19 -22
  88. package/dist/better-auth/plugin/lib/build-collections/users/index.d.ts +1 -1
  89. package/dist/better-auth/plugin/lib/build-collections/users/index.d.ts.map +1 -1
  90. package/dist/better-auth/plugin/lib/build-collections/users/index.js +9 -15
  91. package/dist/better-auth/plugin/lib/build-collections/utils/collection-schema.d.ts +6 -6
  92. package/dist/better-auth/plugin/lib/build-collections/utils/collection-schema.d.ts.map +1 -1
  93. package/dist/better-auth/plugin/lib/build-collections/utils/collection-schema.js +9 -17
  94. package/dist/better-auth/plugin/lib/build-collections/utils/transform-schema-fields-to-payload.d.ts +2 -2
  95. package/dist/better-auth/plugin/lib/build-collections/utils/transform-schema-fields-to-payload.d.ts.map +1 -1
  96. package/dist/better-auth/plugin/lib/build-collections/utils/transform-schema-fields-to-payload.js +1 -1
  97. package/dist/better-auth/plugin/lib/build-collections/verifications.d.ts +1 -1
  98. package/dist/better-auth/plugin/lib/build-collections/verifications.d.ts.map +1 -1
  99. package/dist/better-auth/plugin/lib/build-collections/verifications.js +8 -11
  100. package/dist/better-auth/plugin/lib/init-better-auth.d.ts +1 -1
  101. package/dist/better-auth/plugin/lib/init-better-auth.d.ts.map +1 -1
  102. package/dist/better-auth/plugin/lib/init-better-auth.js +3 -2
  103. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/api-key-plugin.d.ts +2 -2
  104. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/api-key-plugin.d.ts.map +1 -1
  105. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/api-key-plugin.js +4 -4
  106. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/index.d.ts +3 -4
  107. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/index.d.ts.map +1 -1
  108. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/index.js +15 -15
  109. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/oidc-plugin.d.ts +2 -2
  110. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/oidc-plugin.d.ts.map +1 -1
  111. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/oidc-plugin.js +8 -8
  112. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/organizations-plugin.d.ts +2 -2
  113. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/organizations-plugin.d.ts.map +1 -1
  114. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/organizations-plugin.js +12 -12
  115. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/passkey-plugin.d.ts +2 -2
  116. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/passkey-plugin.d.ts.map +1 -1
  117. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/passkey-plugin.js +4 -4
  118. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/sso-plugin.d.ts +2 -2
  119. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/sso-plugin.d.ts.map +1 -1
  120. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/sso-plugin.js +6 -6
  121. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/two-factor-plugin.d.ts +2 -2
  122. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/two-factor-plugin.d.ts.map +1 -1
  123. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/two-factor-plugin.js +4 -4
  124. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/utils/save-to-jwt-middleware.d.ts +3 -4
  125. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/utils/save-to-jwt-middleware.d.ts.map +1 -1
  126. package/dist/better-auth/plugin/lib/sanitize-better-auth-options/utils/save-to-jwt-middleware.js +4 -4
  127. package/dist/better-auth/plugin/payload/components/passkeys/types.d.ts +2 -2
  128. package/dist/better-auth/plugin/payload/components/passkeys/types.d.ts.map +1 -1
  129. package/dist/better-auth/plugin/payload/components/passkeys/types.js +1 -1
  130. package/dist/better-auth/plugin/types.d.ts +6 -5
  131. package/dist/better-auth/plugin/types.d.ts.map +1 -1
  132. package/dist/better-auth/plugin/types.js +1 -1
  133. package/dist/better-auth/scripts/generate-types.js +15 -1
  134. package/package.json +1 -1
  135. package/dist/better-auth/plugin/helpers/get-collection-schema-map.d.ts +0 -16
  136. package/dist/better-auth/plugin/helpers/get-collection-schema-map.d.ts.map +0 -1
  137. package/dist/better-auth/plugin/helpers/get-collection-schema-map.js +0 -108
  138. package/dist/better-auth/plugin/helpers/get-requst-collection.d.ts +0 -3
  139. package/dist/better-auth/plugin/helpers/get-requst-collection.d.ts.map +0 -1
  140. package/dist/better-auth/plugin/helpers/get-requst-collection.js +0 -14
  141. package/dist/better-auth/plugin/helpers/serialize-cookie.d.ts +0 -105
  142. package/dist/better-auth/plugin/helpers/serialize-cookie.d.ts.map +0 -1
  143. package/dist/better-auth/plugin/helpers/serialize-cookie.js +0 -186
@@ -1,108 +0,0 @@
1
- import { baModelFieldKeysToFieldNames, baModelKey } from "../constants";
2
- import { getAuthTables } from "better-auth/db";
3
- import { flattenAllFields } from "payload/shared";
4
- import { getDeafultCollectionSlug } from "./get-collection-slug";
5
- export function getDefaultCollectionSchemaMap(pluginOptions) {
6
- const tables = getAuthTables(pluginOptions.betterAuthOptions ?? {});
7
- const map = {};
8
- Object.entries(tables).forEach(([key, table])=>{
9
- const fieldNames = Object.entries(table.fields).map(([k, v])=>k);
10
- const typedKey = key;
11
- const value = {
12
- collectionSlug: getDeafultCollectionSlug({
13
- modelKey: typedKey,
14
- pluginOptions
15
- }),
16
- fields: fieldNames.reduce((acc, field)=>{
17
- if (typedKey in baModelFieldKeysToFieldNames && field in baModelFieldKeysToFieldNames[typedKey]) {
18
- const fieldMapping = baModelFieldKeysToFieldNames[typedKey];
19
- const typedField = field;
20
- acc[field] = fieldMapping[typedField] || field;
21
- } else {
22
- acc[field] = field;
23
- }
24
- return acc;
25
- }, {})
26
- };
27
- map[typedKey] = value;
28
- });
29
- // always add the role field to the user collection if not already present
30
- if (!map[baModelKey.user].fields.role) {
31
- map[baModelKey.user].fields.role = baModelFieldKeysToFieldNames.user.role;
32
- }
33
- return map;
34
- }
35
- /**
36
- * Builds a collection schema map by analyzing collection overrides and extracting
37
- * BetterAuth model keys and field mappings
38
- *
39
- * @param collectionOverrides - Collection override functions provided in plugin options
40
- * @returns A collection schema map with default values merged with any overrides
41
- */ export function buildCollectionSchemaMap(pluginOptions) {
42
- const collectionOverrides = {
43
- users: pluginOptions.users?.collectionOverrides,
44
- accounts: pluginOptions.accounts?.collectionOverrides,
45
- sessions: pluginOptions.sessions?.collectionOverrides,
46
- verifications: pluginOptions.verifications?.collectionOverrides,
47
- ...pluginOptions.pluginCollectionOverrides
48
- };
49
- const defaultCollectionSchemaMap = getDefaultCollectionSchemaMap(pluginOptions);
50
- if (!collectionOverrides || Object.keys(collectionOverrides).length === 0) {
51
- return {
52
- ...defaultCollectionSchemaMap
53
- };
54
- }
55
- const schemaMap = {
56
- ...defaultCollectionSchemaMap
57
- };
58
- Object.entries(collectionOverrides).forEach(([collectionSlug, overrideFunction])=>{
59
- if (!overrideFunction) return;
60
- const modifiedCollection = overrideFunction({
61
- collection: {
62
- slug: collectionSlug,
63
- fields: []
64
- }
65
- });
66
- const modelKey = assertModelKey(modifiedCollection);
67
- schemaMap[modelKey].collectionSlug = modifiedCollection.slug;
68
- const flattenedFields = flattenAllFields(modifiedCollection);
69
- if (flattenedFields && Array.isArray(flattenedFields)) {
70
- flattenedFields.forEach((field)=>{
71
- // we only want to process fields that have a betterAuthFieldKey
72
- // because its possible the user has added additional fields
73
- if (field.custom?.betterAuthFieldKey) {
74
- const fieldKey = assertFieldKey(field, modelKey, modifiedCollection.slug);
75
- schemaMap[modelKey].fields[fieldKey] = field.name;
76
- }
77
- });
78
- }
79
- });
80
- return schemaMap;
81
- }
82
- function assertFieldKey(field, modelKey, collectionSlug) {
83
- const fieldKey = field.custom?.betterAuthFieldKey;
84
- // Assert that fieldKey is a valid key in the baModelFieldKeysToFieldNames mapping
85
- if (!Object.values(baModelFieldKeysToFieldNames).some((fieldMap)=>Object.keys(fieldMap).includes(fieldKey))) {
86
- throw new Error(`Invalid custom.betterAuthFieldKey: ${fieldKey} for field ${field.name} in collection ${collectionSlug},
87
- must be one of ${Object.keys(baModelFieldKeysToFieldNames[modelKey]).join(', ')}`);
88
- }
89
- return fieldKey;
90
- }
91
- /**
92
- * Asserts that a collection has a valid BetterAuth model key and returns it
93
- *
94
- * @param collection - The collection to check for a BetterAuth model key
95
- * @returns The BetterAuth model key
96
- * @throws Error if the collection does not have a valid BetterAuth model key
97
- */ function assertModelKey(collection) {
98
- if (!collection.custom?.betterAuthModelKey) {
99
- throw new Error(`Collection ${collection.slug} is missing a betterAuthModelKey in its custom field`);
100
- }
101
- const modelKey = collection.custom.betterAuthModelKey;
102
- if (!Object.values(baModelKey).includes(modelKey)) {
103
- throw new Error(`Collection ${collection.slug} has an invalid betterAuthModelKey: ${modelKey}`);
104
- }
105
- return modelKey;
106
- }
107
-
108
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../src/better-auth/plugin/helpers/get-collection-schema-map.ts"],"sourcesContent":["import { baModelFieldKeysToFieldNames, baModelKey } from '../constants'\nimport { getAuthTables } from 'better-auth/db'\nimport { BetterAuthOptions } from 'better-auth/types'\nimport { baModelKeyToSlug } from '../constants'\nimport { FlattenedField } from 'payload'\nimport { flattenAllFields } from 'payload/shared'\nimport { CollectionConfig } from 'payload'\nimport { getDeafultCollectionSlug } from './get-collection-slug'\nimport { BetterAuthPluginOptions } from '../types'\n\nexport type CollectionSchemaMap = Record<\n  keyof typeof baModelKey,\n  {\n    collectionSlug: string\n    fields: Record<string, string>\n  }\n>\n\nexport function getDefaultCollectionSchemaMap(pluginOptions: BetterAuthPluginOptions): CollectionSchemaMap {\n  const tables = getAuthTables(pluginOptions.betterAuthOptions ?? {})\n  const map = {} as CollectionSchemaMap\n  Object.entries(tables).forEach(([key, table]) => {\n    const fieldNames = Object.entries(table.fields).map(([k, v]) => k)\n    const typedKey = key as keyof typeof baModelKeyToSlug\n    const value = {\n      collectionSlug: getDeafultCollectionSlug({ modelKey: typedKey, pluginOptions }),\n      fields: fieldNames.reduce(\n        (acc, field) => {\n          if (\n            typedKey in baModelFieldKeysToFieldNames &&\n            field in baModelFieldKeysToFieldNames[typedKey as keyof typeof baModelFieldKeysToFieldNames]\n          ) {\n            const fieldMapping = baModelFieldKeysToFieldNames[typedKey as keyof typeof baModelFieldKeysToFieldNames]\n            const typedField = field as keyof typeof fieldMapping\n            acc[field] = fieldMapping[typedField] || field\n          } else {\n            acc[field] = field\n          }\n          return acc\n        },\n        {} as Record<string, string>\n      )\n    }\n\n    map[typedKey] = value\n  })\n\n  // always add the role field to the user collection if not already present\n  if (!map[baModelKey.user].fields.role) {\n    map[baModelKey.user].fields.role = baModelFieldKeysToFieldNames.user.role\n  }\n\n  return map\n}\n\ntype CollectionOverride = ((options: { collection: CollectionConfig }) => CollectionConfig) | undefined\n\ntype CollectionOverrides = Record<string, CollectionOverride>\n\n/**\n * Builds a collection schema map by analyzing collection overrides and extracting\n * BetterAuth model keys and field mappings\n *\n * @param collectionOverrides - Collection override functions provided in plugin options\n * @returns A collection schema map with default values merged with any overrides\n */\nexport function buildCollectionSchemaMap(pluginOptions: BetterAuthPluginOptions): CollectionSchemaMap {\n  const collectionOverrides = {\n    users: pluginOptions.users?.collectionOverrides,\n    accounts: pluginOptions.accounts?.collectionOverrides,\n    sessions: pluginOptions.sessions?.collectionOverrides,\n    verifications: pluginOptions.verifications?.collectionOverrides,\n    ...pluginOptions.pluginCollectionOverrides\n  }\n\n  const defaultCollectionSchemaMap = getDefaultCollectionSchemaMap(pluginOptions)\n\n  if (!collectionOverrides || Object.keys(collectionOverrides).length === 0) {\n    return { ...defaultCollectionSchemaMap }\n  }\n\n  const schemaMap = { ...defaultCollectionSchemaMap }\n\n  Object.entries(collectionOverrides).forEach(([collectionSlug, overrideFunction]) => {\n    if (!overrideFunction) return\n\n    const modifiedCollection = overrideFunction({\n      collection: {\n        slug: collectionSlug,\n        fields: []\n      }\n    })\n    const modelKey = assertModelKey(modifiedCollection)\n    schemaMap[modelKey].collectionSlug = modifiedCollection.slug\n    const flattenedFields = flattenAllFields(modifiedCollection)\n    if (flattenedFields && Array.isArray(flattenedFields)) {\n      flattenedFields.forEach((field: FlattenedField) => {\n        // we only want to process fields that have a betterAuthFieldKey\n        // because its possible the user has added additional fields\n        if (field.custom?.betterAuthFieldKey) {\n          const fieldKey = assertFieldKey(field, modelKey, modifiedCollection.slug)\n          schemaMap[modelKey].fields[fieldKey] = field.name\n        }\n      })\n    }\n  })\n\n  return schemaMap\n}\n\nfunction assertFieldKey(field: FlattenedField, modelKey: keyof typeof baModelKey, collectionSlug: string) {\n  const fieldKey = field.custom?.betterAuthFieldKey as string\n\n  // Assert that fieldKey is a valid key in the baModelFieldKeysToFieldNames mapping\n  if (!Object.values(baModelFieldKeysToFieldNames).some((fieldMap) => Object.keys(fieldMap).includes(fieldKey))) {\n    throw new Error(\n      `Invalid custom.betterAuthFieldKey: ${fieldKey} for field ${field.name} in collection ${collectionSlug}, \n        must be one of ${Object.keys(baModelFieldKeysToFieldNames[modelKey as keyof typeof baModelFieldKeysToFieldNames]).join(', ')}`\n    )\n  }\n\n  return fieldKey\n}\n\n/**\n * Asserts that a collection has a valid BetterAuth model key and returns it\n *\n * @param collection - The collection to check for a BetterAuth model key\n * @returns The BetterAuth model key\n * @throws Error if the collection does not have a valid BetterAuth model key\n */\nfunction assertModelKey(collection: CollectionConfig): keyof typeof baModelKey {\n  if (!collection.custom?.betterAuthModelKey) {\n    throw new Error(`Collection ${collection.slug} is missing a betterAuthModelKey in its custom field`)\n  }\n\n  const modelKey = collection.custom.betterAuthModelKey as keyof typeof baModelKey\n\n  if (!Object.values(baModelKey).includes(modelKey as any)) {\n    throw new Error(`Collection ${collection.slug} has an invalid betterAuthModelKey: ${modelKey}`)\n  }\n\n  return modelKey\n}\n"],"names":["baModelFieldKeysToFieldNames","baModelKey","getAuthTables","flattenAllFields","getDeafultCollectionSlug","getDefaultCollectionSchemaMap","pluginOptions","tables","betterAuthOptions","map","Object","entries","forEach","key","table","fieldNames","fields","k","v","typedKey","value","collectionSlug","modelKey","reduce","acc","field","fieldMapping","typedField","user","role","buildCollectionSchemaMap","collectionOverrides","users","accounts","sessions","verifications","pluginCollectionOverrides","defaultCollectionSchemaMap","keys","length","schemaMap","overrideFunction","modifiedCollection","collection","slug","assertModelKey","flattenedFields","Array","isArray","custom","betterAuthFieldKey","fieldKey","assertFieldKey","name","values","some","fieldMap","includes","Error","join","betterAuthModelKey"],"mappings":"AAAA,SAASA,4BAA4B,EAAEC,UAAU,QAAQ,eAAc;AACvE,SAASC,aAAa,QAAQ,iBAAgB;AAI9C,SAASC,gBAAgB,QAAQ,iBAAgB;AAEjD,SAASC,wBAAwB,QAAQ,wBAAuB;AAWhE,OAAO,SAASC,8BAA8BC,aAAsC;IAClF,MAAMC,SAASL,cAAcI,cAAcE,iBAAiB,IAAI,CAAC;IACjE,MAAMC,MAAM,CAAC;IACbC,OAAOC,OAAO,CAACJ,QAAQK,OAAO,CAAC,CAAC,CAACC,KAAKC,MAAM;QAC1C,MAAMC,aAAaL,OAAOC,OAAO,CAACG,MAAME,MAAM,EAAEP,GAAG,CAAC,CAAC,CAACQ,GAAGC,EAAE,GAAKD;QAChE,MAAME,WAAWN;QACjB,MAAMO,QAAQ;YACZC,gBAAgBjB,yBAAyB;gBAAEkB,UAAUH;gBAAUb;YAAc;YAC7EU,QAAQD,WAAWQ,MAAM,CACvB,CAACC,KAAKC;gBACJ,IACEN,YAAYnB,gCACZyB,SAASzB,4BAA4B,CAACmB,SAAsD,EAC5F;oBACA,MAAMO,eAAe1B,4BAA4B,CAACmB,SAAsD;oBACxG,MAAMQ,aAAaF;oBACnBD,GAAG,CAACC,MAAM,GAAGC,YAAY,CAACC,WAAW,IAAIF;gBAC3C,OAAO;oBACLD,GAAG,CAACC,MAAM,GAAGA;gBACf;gBACA,OAAOD;YACT,GACA,CAAC;QAEL;QAEAf,GAAG,CAACU,SAAS,GAAGC;IAClB;IAEA,0EAA0E;IAC1E,IAAI,CAACX,GAAG,CAACR,WAAW2B,IAAI,CAAC,CAACZ,MAAM,CAACa,IAAI,EAAE;QACrCpB,GAAG,CAACR,WAAW2B,IAAI,CAAC,CAACZ,MAAM,CAACa,IAAI,GAAG7B,6BAA6B4B,IAAI,CAACC,IAAI;IAC3E;IAEA,OAAOpB;AACT;AAMA;;;;;;CAMC,GACD,OAAO,SAASqB,yBAAyBxB,aAAsC;IAC7E,MAAMyB,sBAAsB;QAC1BC,OAAO1B,cAAc0B,KAAK,EAAED;QAC5BE,UAAU3B,cAAc2B,QAAQ,EAAEF;QAClCG,UAAU5B,cAAc4B,QAAQ,EAAEH;QAClCI,eAAe7B,cAAc6B,aAAa,EAAEJ;QAC5C,GAAGzB,cAAc8B,yBAAyB;IAC5C;IAEA,MAAMC,6BAA6BhC,8BAA8BC;IAEjE,IAAI,CAACyB,uBAAuBrB,OAAO4B,IAAI,CAACP,qBAAqBQ,MAAM,KAAK,GAAG;QACzE,OAAO;YAAE,GAAGF,0BAA0B;QAAC;IACzC;IAEA,MAAMG,YAAY;QAAE,GAAGH,0BAA0B;IAAC;IAElD3B,OAAOC,OAAO,CAACoB,qBAAqBnB,OAAO,CAAC,CAAC,CAACS,gBAAgBoB,iBAAiB;QAC7E,IAAI,CAACA,kBAAkB;QAEvB,MAAMC,qBAAqBD,iBAAiB;YAC1CE,YAAY;gBACVC,MAAMvB;gBACNL,QAAQ,EAAE;YACZ;QACF;QACA,MAAMM,WAAWuB,eAAeH;QAChCF,SAAS,CAAClB,SAAS,CAACD,cAAc,GAAGqB,mBAAmBE,IAAI;QAC5D,MAAME,kBAAkB3C,iBAAiBuC;QACzC,IAAII,mBAAmBC,MAAMC,OAAO,CAACF,kBAAkB;YACrDA,gBAAgBlC,OAAO,CAAC,CAACa;gBACvB,gEAAgE;gBAChE,4DAA4D;gBAC5D,IAAIA,MAAMwB,MAAM,EAAEC,oBAAoB;oBACpC,MAAMC,WAAWC,eAAe3B,OAAOH,UAAUoB,mBAAmBE,IAAI;oBACxEJ,SAAS,CAAClB,SAAS,CAACN,MAAM,CAACmC,SAAS,GAAG1B,MAAM4B,IAAI;gBACnD;YACF;QACF;IACF;IAEA,OAAOb;AACT;AAEA,SAASY,eAAe3B,KAAqB,EAAEH,QAAiC,EAAED,cAAsB;IACtG,MAAM8B,WAAW1B,MAAMwB,MAAM,EAAEC;IAE/B,kFAAkF;IAClF,IAAI,CAACxC,OAAO4C,MAAM,CAACtD,8BAA8BuD,IAAI,CAAC,CAACC,WAAa9C,OAAO4B,IAAI,CAACkB,UAAUC,QAAQ,CAACN,YAAY;QAC7G,MAAM,IAAIO,MACR,CAAC,mCAAmC,EAAEP,SAAS,WAAW,EAAE1B,MAAM4B,IAAI,CAAC,eAAe,EAAEhC,eAAe;uBACtF,EAAEX,OAAO4B,IAAI,CAACtC,4BAA4B,CAACsB,SAAsD,EAAEqC,IAAI,CAAC,OAAO;IAEpI;IAEA,OAAOR;AACT;AAEA;;;;;;CAMC,GACD,SAASN,eAAeF,UAA4B;IAClD,IAAI,CAACA,WAAWM,MAAM,EAAEW,oBAAoB;QAC1C,MAAM,IAAIF,MAAM,CAAC,WAAW,EAAEf,WAAWC,IAAI,CAAC,oDAAoD,CAAC;IACrG;IAEA,MAAMtB,WAAWqB,WAAWM,MAAM,CAACW,kBAAkB;IAErD,IAAI,CAAClD,OAAO4C,MAAM,CAACrD,YAAYwD,QAAQ,CAACnC,WAAkB;QACxD,MAAM,IAAIoC,MAAM,CAAC,WAAW,EAAEf,WAAWC,IAAI,CAAC,oCAAoC,EAAEtB,UAAU;IAChG;IAEA,OAAOA;AACT"}
@@ -1,3 +0,0 @@
1
- import { type Collection, type PayloadRequest } from 'payload';
2
- export declare const getRequestCollection: (req: PayloadRequest) => Collection;
3
- //# sourceMappingURL=get-requst-collection.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"get-requst-collection.d.ts","sourceRoot":"","sources":["../../../../src/better-auth/plugin/helpers/get-requst-collection.ts"],"names":[],"mappings":"AAAA,OAAO,EAAY,KAAK,UAAU,EAAE,KAAK,cAAc,EAAE,MAAM,SAAS,CAAA;AAExE,eAAO,MAAM,oBAAoB,GAAI,KAAK,cAAc,KAAG,UAc1D,CAAA"}
@@ -1,14 +0,0 @@
1
- import { APIError } from "payload";
2
- export const getRequestCollection = (req)=>{
3
- const collectionSlug = req.routeParams?.collection;
4
- if (typeof collectionSlug !== 'string') {
5
- throw new APIError(`No collection was specified`, 400);
6
- }
7
- const collection = req.payload.collections[collectionSlug];
8
- if (!collection) {
9
- throw new APIError(`Collection with the slug ${collectionSlug} was not found`, 404);
10
- }
11
- return collection;
12
- };
13
-
14
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9iZXR0ZXItYXV0aC9wbHVnaW4vaGVscGVycy9nZXQtcmVxdXN0LWNvbGxlY3Rpb24udHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQVBJRXJyb3IsIHR5cGUgQ29sbGVjdGlvbiwgdHlwZSBQYXlsb2FkUmVxdWVzdCB9IGZyb20gJ3BheWxvYWQnXG5cbmV4cG9ydCBjb25zdCBnZXRSZXF1ZXN0Q29sbGVjdGlvbiA9IChyZXE6IFBheWxvYWRSZXF1ZXN0KTogQ29sbGVjdGlvbiA9PiB7XG4gIGNvbnN0IGNvbGxlY3Rpb25TbHVnID0gcmVxLnJvdXRlUGFyYW1zPy5jb2xsZWN0aW9uXG5cbiAgaWYgKHR5cGVvZiBjb2xsZWN0aW9uU2x1ZyAhPT0gJ3N0cmluZycpIHtcbiAgICB0aHJvdyBuZXcgQVBJRXJyb3IoYE5vIGNvbGxlY3Rpb24gd2FzIHNwZWNpZmllZGAsIDQwMClcbiAgfVxuXG4gIGNvbnN0IGNvbGxlY3Rpb24gPSByZXEucGF5bG9hZC5jb2xsZWN0aW9uc1tjb2xsZWN0aW9uU2x1Z11cblxuICBpZiAoIWNvbGxlY3Rpb24pIHtcbiAgICB0aHJvdyBuZXcgQVBJRXJyb3IoYENvbGxlY3Rpb24gd2l0aCB0aGUgc2x1ZyAke2NvbGxlY3Rpb25TbHVnfSB3YXMgbm90IGZvdW5kYCwgNDA0KVxuICB9XG5cbiAgcmV0dXJuIGNvbGxlY3Rpb25cbn1cbiJdLCJuYW1lcyI6WyJBUElFcnJvciIsImdldFJlcXVlc3RDb2xsZWN0aW9uIiwicmVxIiwiY29sbGVjdGlvblNsdWciLCJyb3V0ZVBhcmFtcyIsImNvbGxlY3Rpb24iLCJwYXlsb2FkIiwiY29sbGVjdGlvbnMiXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLFFBQVEsUUFBOEMsVUFBUztBQUV4RSxPQUFPLE1BQU1DLHVCQUF1QixDQUFDQztJQUNuQyxNQUFNQyxpQkFBaUJELElBQUlFLFdBQVcsRUFBRUM7SUFFeEMsSUFBSSxPQUFPRixtQkFBbUIsVUFBVTtRQUN0QyxNQUFNLElBQUlILFNBQVMsQ0FBQywyQkFBMkIsQ0FBQyxFQUFFO0lBQ3BEO0lBRUEsTUFBTUssYUFBYUgsSUFBSUksT0FBTyxDQUFDQyxXQUFXLENBQUNKLGVBQWU7SUFFMUQsSUFBSSxDQUFDRSxZQUFZO1FBQ2YsTUFBTSxJQUFJTCxTQUFTLENBQUMseUJBQXlCLEVBQUVHLGVBQWUsY0FBYyxDQUFDLEVBQUU7SUFDakY7SUFFQSxPQUFPRTtBQUNULEVBQUMifQ==
@@ -1,105 +0,0 @@
1
- type CookiePrefixOptions = 'host' | 'secure';
2
- type CookieOptions = {
3
- /**
4
- * Domain of the cookie
5
- *
6
- * The Domain attribute specifies which server can receive a cookie. If specified, cookies are
7
- * available on the specified server and its subdomains. If the it is not
8
- * specified, the cookies are available on the server that sets it but not on
9
- * its subdomains.
10
- *
11
- * @example
12
- * `domain: "example.com"`
13
- */
14
- domain?: string;
15
- /**
16
- * A lifetime of a cookie. Permanent cookies are deleted after the date specified in the
17
- * Expires attribute:
18
- *
19
- * Expires has been available for longer than Max-Age, however Max-Age is less error-prone, and
20
- * takes precedence when both are set. The rationale behind this is that when you set an
21
- * Expires date and time, they're relative to the client the cookie is being set on. If the
22
- * server is set to a different time, this could cause errors
23
- */
24
- expires?: Date;
25
- /**
26
- * Forbids JavaScript from accessing the cookie, for example, through the Document.cookie
27
- * property. Note that a cookie that has been created with HttpOnly will still be sent with
28
- * JavaScript-initiated requests, for example, when calling XMLHttpRequest.send() or fetch().
29
- * This mitigates attacks against cross-site scripting
30
- */
31
- httpOnly?: boolean;
32
- /**
33
- * Indicates the number of seconds until the cookie expires. A zero or negative number will
34
- * expire the cookie immediately. If both Expires and Max-Age are set, Max-Age has precedence.
35
- *
36
- * @example 604800 - 7 days
37
- */
38
- maxAge?: number;
39
- /**
40
- * Indicates the path that must exist in the requested URL for the browser to send the Cookie
41
- * header.
42
- *
43
- * @example
44
- * "/docs"
45
- * // -> the request paths /docs, /docs/, /docs/Web/, and /docs/Web/HTTP will all match. the request paths /, /fr/docs will not match.
46
- */
47
- path?: string;
48
- /**
49
- * Indicates that the cookie is sent to the server only when a request is made with the https:
50
- * scheme (except on localhost), and therefore, is more resistant to man-in-the-middle attacks.
51
- */
52
- secure?: boolean;
53
- /**
54
- * Controls whether or not a cookie is sent with cross-site requests, providing some protection
55
- * against cross-site request forgery attacks (CSRF).
56
- *
57
- * Strict - Means that the browser sends the cookie only for same-site requests, that is,
58
- * requests originating from the same site that set the cookie. If a request originates from a
59
- * different domain or scheme (even with the same domain), no cookies with the SameSite=Strict
60
- * attribute are sent.
61
- *
62
- * Lax - Means that the cookie is not sent on cross-site requests, such as on requests to load
63
- * images or frames, but is sent when a user is navigating to the origin site from an external
64
- * site (for example, when following a link). This is the default behavior if the SameSite
65
- * attribute is not specified.
66
- *
67
- * None - Means that the browser sends the cookie with both cross-site and same-site requests.
68
- * The Secure attribute must also be set when setting this value.
69
- */
70
- sameSite?: 'Strict' | 'Lax' | 'None' | 'strict' | 'lax' | 'none';
71
- /**
72
- * Indicates that the cookie should be stored using partitioned storage. Note that if this is
73
- * set, the Secure directive must also be set.
74
- *
75
- * @see https://developer.mozilla.org/en-US/docs/Web/Privacy/Privacy_sandbox/Partitioned_cookies
76
- */
77
- partitioned?: boolean;
78
- /**
79
- * Cooke Prefix
80
- *
81
- * - secure: `__Secure-` -> `__Secure-cookie-name`
82
- * - host: `__Host-` -> `__Host-cookie-name`
83
- *
84
- * `secure` must be set to true to use prefixes
85
- */
86
- prefix?: CookiePrefixOptions;
87
- };
88
- export declare const verifySignature: (base64Signature: string, value: string, secret: CryptoKey) => Promise<boolean>;
89
- export declare const signCookieValue: (value: string, secret: string | BufferSource) => Promise<string>;
90
- export declare const serializeCookie: (key: string, value: string, opt?: CookieOptions) => string;
91
- export declare const serializeSignedCookie: (key: string, value: string, secret: string, opt?: CookieOptions) => Promise<string>;
92
- export declare const getCookieKey: (key: string, prefix?: CookiePrefixOptions) => string | undefined;
93
- export declare function tryDecode(str: string): string;
94
- /**
95
- * Parse an HTTP Cookie header string and returning an object of all cookie
96
- * name-value pairs.
97
- *
98
- * Inspired by https://github.com/unjs/cookie-es/blob/main/src/cookie/parse.ts
99
- *
100
- * @param str the string representing a `Cookie` header value
101
- */
102
- export declare function parseCookies(str: string): Map<string, string>;
103
- export declare const getSignedCookie: (key: string, secret: string, headers: Headers, prefix?: CookiePrefixOptions) => Promise<string | false | null>;
104
- export {};
105
- //# sourceMappingURL=serialize-cookie.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"serialize-cookie.d.ts","sourceRoot":"","sources":["../../../../src/better-auth/plugin/helpers/serialize-cookie.ts"],"names":[],"mappings":"AAEA,KAAK,mBAAmB,GAAG,MAAM,GAAG,QAAQ,CAAA;AAE5C,KAAK,aAAa,GAAG;IACnB;;;;;;;;;;OAUG;IACH,MAAM,CAAC,EAAE,MAAM,CAAA;IACf;;;;;;;;OAQG;IACH,OAAO,CAAC,EAAE,IAAI,CAAA;IACd;;;;;OAKG;IACH,QAAQ,CAAC,EAAE,OAAO,CAAA;IAClB;;;;;OAKG;IACH,MAAM,CAAC,EAAE,MAAM,CAAA;IACf;;;;;;;OAOG;IACH,IAAI,CAAC,EAAE,MAAM,CAAA;IACb;;;OAGG;IACH,MAAM,CAAC,EAAE,OAAO,CAAA;IAChB;;;;;;;;;;;;;;;;OAgBG;IACH,QAAQ,CAAC,EAAE,QAAQ,GAAG,KAAK,GAAG,MAAM,GAAG,QAAQ,GAAG,KAAK,GAAG,MAAM,CAAA;IAChE;;;;;OAKG;IACH,WAAW,CAAC,EAAE,OAAO,CAAA;IACrB;;;;;;;OAOG;IACH,MAAM,CAAC,EAAE,mBAAmB,CAAA;CAC7B,CAAA;AAED,eAAO,MAAM,eAAe,GAAU,iBAAiB,MAAM,EAAE,OAAO,MAAM,EAAE,QAAQ,SAAS,KAAG,OAAO,CAAC,OAAO,CAWhH,CAAA;AAyFD,eAAO,MAAM,eAAe,GAAU,OAAO,MAAM,EAAE,QAAQ,MAAM,GAAG,YAAY,oBAMjF,CAAA;AAED,eAAO,MAAM,eAAe,GAAI,KAAK,MAAM,EAAE,OAAO,MAAM,EAAE,MAAM,aAAa,WAG9E,CAAA;AAED,eAAO,MAAM,qBAAqB,GAAU,KAAK,MAAM,EAAE,OAAO,MAAM,EAAE,QAAQ,MAAM,EAAE,MAAM,aAAa,oBAG1G,CAAA;AAED,eAAO,MAAM,YAAY,GAAI,KAAK,MAAM,EAAE,SAAS,mBAAmB,uBAYrE,CAAA;AAED,wBAAgB,SAAS,CAAC,GAAG,EAAE,MAAM,UAMpC;AAED;;;;;;;GAOG;AACH,wBAAgB,YAAY,CAAC,GAAG,EAAE,MAAM,uBAqCvC;AAED,eAAO,MAAM,eAAe,GAAU,KAAK,MAAM,EAAE,QAAQ,MAAM,EAAE,SAAS,OAAO,EAAE,SAAS,mBAAmB,mCAuBhH,CAAA"}
@@ -1,186 +0,0 @@
1
- import { subtle } from "uncrypto";
2
- export const verifySignature = async (base64Signature, value, secret)=>{
3
- try {
4
- const signatureBinStr = atob(base64Signature);
5
- const signature = new Uint8Array(signatureBinStr.length);
6
- for(let i = 0, len = signatureBinStr.length; i < len; i++){
7
- signature[i] = signatureBinStr.charCodeAt(i);
8
- }
9
- return await subtle.verify(algorithm, secret, signature, new TextEncoder().encode(value));
10
- } catch (e) {
11
- return false;
12
- }
13
- };
14
- const _serialize = (key, value, opt = {})=>{
15
- let cookie;
16
- if (opt?.prefix === 'secure') {
17
- cookie = `${`__Secure-${key}`}=${value}`;
18
- } else if (opt?.prefix === 'host') {
19
- cookie = `${`__Host-${key}`}=${value}`;
20
- } else {
21
- cookie = `${key}=${value}`;
22
- }
23
- if (key.startsWith('__Secure-') && !opt.secure) {
24
- opt.secure = true;
25
- }
26
- if (key.startsWith('__Host-')) {
27
- if (!opt.secure) {
28
- opt.secure = true;
29
- }
30
- if (opt.path !== '/') {
31
- opt.path = '/';
32
- }
33
- if (opt.domain) {
34
- opt.domain = undefined;
35
- }
36
- }
37
- if (opt && typeof opt.maxAge === 'number' && opt.maxAge >= 0) {
38
- if (opt.maxAge > 34560000) {
39
- throw new Error('Cookies Max-Age SHOULD NOT be greater than 400 days (34560000 seconds) in duration.');
40
- }
41
- cookie += `; Max-Age=${Math.floor(opt.maxAge)}`;
42
- }
43
- if (opt.domain && opt.prefix !== 'host') {
44
- cookie += `; Domain=${opt.domain}`;
45
- }
46
- if (opt.path) {
47
- cookie += `; Path=${opt.path}`;
48
- }
49
- if (opt.expires) {
50
- if (opt.expires.getTime() - Date.now() > 34560000_000) {
51
- throw new Error('Cookies Expires SHOULD NOT be greater than 400 days (34560000 seconds) in the future.');
52
- }
53
- cookie += `; Expires=${opt.expires.toUTCString()}`;
54
- }
55
- if (opt.httpOnly) {
56
- cookie += '; HttpOnly';
57
- }
58
- if (opt.secure) {
59
- cookie += '; Secure';
60
- }
61
- if (opt.sameSite) {
62
- cookie += `; SameSite=${opt.sameSite.charAt(0).toUpperCase() + opt.sameSite.slice(1)}`;
63
- }
64
- if (opt.partitioned) {
65
- if (!opt.secure) {
66
- opt.secure = true;
67
- }
68
- cookie += '; Partitioned';
69
- }
70
- return cookie;
71
- };
72
- const algorithm = {
73
- name: 'HMAC',
74
- hash: 'SHA-256'
75
- };
76
- const getCryptoKey = async (secret)=>{
77
- const secretBuf = typeof secret === 'string' ? new TextEncoder().encode(secret) : secret;
78
- return await subtle.importKey('raw', secretBuf, algorithm, false, [
79
- 'sign',
80
- 'verify'
81
- ]);
82
- };
83
- const makeSignature = async (value, secret)=>{
84
- const key = await getCryptoKey(secret);
85
- const signature = await subtle.sign(algorithm.name, key, new TextEncoder().encode(value));
86
- // the returned base64 encoded signature will always be 44 characters long and end with one or two equal signs
87
- return btoa(String.fromCharCode(...new Uint8Array(signature)));
88
- };
89
- export const signCookieValue = async (value, secret)=>{
90
- const signature = await makeSignature(value, secret);
91
- value = `${value}.${signature}`;
92
- value = encodeURIComponent(value);
93
- value = decodeURIComponent(value);
94
- return value;
95
- };
96
- export const serializeCookie = (key, value, opt)=>{
97
- value = encodeURIComponent(value);
98
- return _serialize(key, value, opt);
99
- };
100
- export const serializeSignedCookie = async (key, value, secret, opt)=>{
101
- value = await signCookieValue(value, secret);
102
- return _serialize(key, value, opt);
103
- };
104
- export const getCookieKey = (key, prefix)=>{
105
- let finalKey = key;
106
- if (prefix) {
107
- if (prefix === 'secure') {
108
- finalKey = '__Secure-' + key;
109
- } else if (prefix === 'host') {
110
- finalKey = '__Host-' + key;
111
- } else {
112
- return undefined;
113
- }
114
- }
115
- return finalKey;
116
- };
117
- export function tryDecode(str) {
118
- try {
119
- return str.includes('%') ? decodeURIComponent(str) : str;
120
- } catch {
121
- return str;
122
- }
123
- }
124
- /**
125
- * Parse an HTTP Cookie header string and returning an object of all cookie
126
- * name-value pairs.
127
- *
128
- * Inspired by https://github.com/unjs/cookie-es/blob/main/src/cookie/parse.ts
129
- *
130
- * @param str the string representing a `Cookie` header value
131
- */ export function parseCookies(str) {
132
- if (typeof str !== 'string') {
133
- throw new TypeError('argument str must be a string');
134
- }
135
- const cookies = new Map();
136
- let index = 0;
137
- while(index < str.length){
138
- const eqIdx = str.indexOf('=', index);
139
- if (eqIdx === -1) {
140
- break;
141
- }
142
- let endIdx = str.indexOf(';', index);
143
- if (endIdx === -1) {
144
- endIdx = str.length;
145
- } else if (endIdx < eqIdx) {
146
- index = str.lastIndexOf(';', eqIdx - 1) + 1;
147
- continue;
148
- }
149
- const key = str.slice(index, eqIdx).trim();
150
- if (!cookies.has(key)) {
151
- let val = str.slice(eqIdx + 1, endIdx).trim();
152
- if (val.codePointAt(0) === 0x22) {
153
- val = val.slice(1, -1);
154
- }
155
- cookies.set(key, tryDecode(val));
156
- }
157
- index = endIdx + 1;
158
- }
159
- return cookies;
160
- }
161
- export const getSignedCookie = async (key, secret, headers, prefix)=>{
162
- const finalKey = getCookieKey(key, prefix);
163
- if (!finalKey) {
164
- return null;
165
- }
166
- const cookieHeader = headers.get('cookie');
167
- const parsedCookies = cookieHeader ? parseCookies(cookieHeader) : undefined;
168
- const value = parsedCookies?.get(finalKey);
169
- if (!value) {
170
- return null;
171
- }
172
- const signatureStartPos = value.lastIndexOf('.');
173
- if (signatureStartPos < 1) {
174
- return null;
175
- }
176
- const signedValue = value.substring(0, signatureStartPos);
177
- const signature = value.substring(signatureStartPos + 1);
178
- if (signature.length !== 44 || !signature.endsWith('=')) {
179
- return null;
180
- }
181
- const secretKey = await getCryptoKey(secret);
182
- const isVerified = await verifySignature(signature, signedValue, secretKey);
183
- return isVerified ? signedValue : false;
184
- };
185
-
186
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../src/better-auth/plugin/helpers/serialize-cookie.ts"],"sourcesContent":["import { subtle } from 'uncrypto'\n\ntype CookiePrefixOptions = 'host' | 'secure'\n\ntype CookieOptions = {\n  /**\n   * Domain of the cookie\n   *\n   * The Domain attribute specifies which server can receive a cookie. If specified, cookies are\n   * available on the specified server and its subdomains. If the it is not\n   * specified, the cookies are available on the server that sets it but not on\n   * its subdomains.\n   *\n   * @example\n   * `domain: \"example.com\"`\n   */\n  domain?: string\n  /**\n   * A lifetime of a cookie. Permanent cookies are deleted after the date specified in the\n   * Expires attribute:\n   *\n   * Expires has been available for longer than Max-Age, however Max-Age is less error-prone, and\n   * takes precedence when both are set. The rationale behind this is that when you set an\n   * Expires date and time, they're relative to the client the cookie is being set on. If the\n   * server is set to a different time, this could cause errors\n   */\n  expires?: Date\n  /**\n   * Forbids JavaScript from accessing the cookie, for example, through the Document.cookie\n   * property. Note that a cookie that has been created with HttpOnly will still be sent with\n   * JavaScript-initiated requests, for example, when calling XMLHttpRequest.send() or fetch().\n   * This mitigates attacks against cross-site scripting\n   */\n  httpOnly?: boolean\n  /**\n   * Indicates the number of seconds until the cookie expires. A zero or negative number will\n   * expire the cookie immediately. If both Expires and Max-Age are set, Max-Age has precedence.\n   *\n   * @example 604800 - 7 days\n   */\n  maxAge?: number\n  /**\n   * Indicates the path that must exist in the requested URL for the browser to send the Cookie\n   * header.\n   *\n   * @example\n   * \"/docs\"\n   * // -> the request paths /docs, /docs/, /docs/Web/, and /docs/Web/HTTP will all match. the request paths /, /fr/docs will not match.\n   */\n  path?: string\n  /**\n   * Indicates that the cookie is sent to the server only when a request is made with the https:\n   * scheme (except on localhost), and therefore, is more resistant to man-in-the-middle attacks.\n   */\n  secure?: boolean\n  /**\n   * Controls whether or not a cookie is sent with cross-site requests, providing some protection\n   * against cross-site request forgery attacks (CSRF).\n   *\n   * Strict -  Means that the browser sends the cookie only for same-site requests, that is,\n   * requests originating from the same site that set the cookie. If a request originates from a\n   * different domain or scheme (even with the same domain), no cookies with the SameSite=Strict\n   * attribute are sent.\n   *\n   * Lax - Means that the cookie is not sent on cross-site requests, such as on requests to load\n   * images or frames, but is sent when a user is navigating to the origin site from an external\n   * site (for example, when following a link). This is the default behavior if the SameSite\n   * attribute is not specified.\n   *\n   * None - Means that the browser sends the cookie with both cross-site and same-site requests.\n   * The Secure attribute must also be set when setting this value.\n   */\n  sameSite?: 'Strict' | 'Lax' | 'None' | 'strict' | 'lax' | 'none'\n  /**\n   * Indicates that the cookie should be stored using partitioned storage. Note that if this is\n   * set, the Secure directive must also be set.\n   *\n   * @see https://developer.mozilla.org/en-US/docs/Web/Privacy/Privacy_sandbox/Partitioned_cookies\n   */\n  partitioned?: boolean\n  /**\n   * Cooke Prefix\n   *\n   * - secure: `__Secure-` -> `__Secure-cookie-name`\n   * - host: `__Host-` -> `__Host-cookie-name`\n   *\n   * `secure` must be set to true to use prefixes\n   */\n  prefix?: CookiePrefixOptions\n}\n\nexport const verifySignature = async (base64Signature: string, value: string, secret: CryptoKey): Promise<boolean> => {\n  try {\n    const signatureBinStr = atob(base64Signature)\n    const signature = new Uint8Array(signatureBinStr.length)\n    for (let i = 0, len = signatureBinStr.length; i < len; i++) {\n      signature[i] = signatureBinStr.charCodeAt(i)\n    }\n    return await subtle.verify(algorithm, secret, signature, new TextEncoder().encode(value))\n  } catch (e) {\n    return false\n  }\n}\n\nconst _serialize = (key: string, value: string, opt: CookieOptions = {}) => {\n  let cookie: string\n\n  if (opt?.prefix === 'secure') {\n    cookie = `${`__Secure-${key}`}=${value}`\n  } else if (opt?.prefix === 'host') {\n    cookie = `${`__Host-${key}`}=${value}`\n  } else {\n    cookie = `${key}=${value}`\n  }\n\n  if (key.startsWith('__Secure-') && !opt.secure) {\n    opt.secure = true\n  }\n\n  if (key.startsWith('__Host-')) {\n    if (!opt.secure) {\n      opt.secure = true\n    }\n\n    if (opt.path !== '/') {\n      opt.path = '/'\n    }\n\n    if (opt.domain) {\n      opt.domain = undefined\n    }\n  }\n\n  if (opt && typeof opt.maxAge === 'number' && opt.maxAge >= 0) {\n    if (opt.maxAge > 34560000) {\n      throw new Error('Cookies Max-Age SHOULD NOT be greater than 400 days (34560000 seconds) in duration.')\n    }\n    cookie += `; Max-Age=${Math.floor(opt.maxAge)}`\n  }\n\n  if (opt.domain && opt.prefix !== 'host') {\n    cookie += `; Domain=${opt.domain}`\n  }\n\n  if (opt.path) {\n    cookie += `; Path=${opt.path}`\n  }\n\n  if (opt.expires) {\n    if (opt.expires.getTime() - Date.now() > 34560000_000) {\n      throw new Error('Cookies Expires SHOULD NOT be greater than 400 days (34560000 seconds) in the future.')\n    }\n    cookie += `; Expires=${opt.expires.toUTCString()}`\n  }\n\n  if (opt.httpOnly) {\n    cookie += '; HttpOnly'\n  }\n\n  if (opt.secure) {\n    cookie += '; Secure'\n  }\n\n  if (opt.sameSite) {\n    cookie += `; SameSite=${opt.sameSite.charAt(0).toUpperCase() + opt.sameSite.slice(1)}`\n  }\n\n  if (opt.partitioned) {\n    if (!opt.secure) {\n      opt.secure = true\n    }\n    cookie += '; Partitioned'\n  }\n\n  return cookie\n}\n\nconst algorithm = { name: 'HMAC', hash: 'SHA-256' }\n\nconst getCryptoKey = async (secret: string | BufferSource) => {\n  const secretBuf = typeof secret === 'string' ? new TextEncoder().encode(secret) : secret\n  return await subtle.importKey('raw', secretBuf, algorithm, false, ['sign', 'verify'])\n}\n\nconst makeSignature = async (value: string, secret: string | BufferSource): Promise<string> => {\n  const key = await getCryptoKey(secret)\n  const signature = await subtle.sign(algorithm.name, key, new TextEncoder().encode(value))\n  // the returned base64 encoded signature will always be 44 characters long and end with one or two equal signs\n  return btoa(String.fromCharCode(...new Uint8Array(signature)))\n}\n\nexport const signCookieValue = async (value: string, secret: string | BufferSource) => {\n  const signature = await makeSignature(value, secret)\n  value = `${value}.${signature}`\n  value = encodeURIComponent(value)\n  value = decodeURIComponent(value)\n  return value\n}\n\nexport const serializeCookie = (key: string, value: string, opt?: CookieOptions) => {\n  value = encodeURIComponent(value)\n  return _serialize(key, value, opt)\n}\n\nexport const serializeSignedCookie = async (key: string, value: string, secret: string, opt?: CookieOptions) => {\n  value = await signCookieValue(value, secret)\n  return _serialize(key, value, opt)\n}\n\nexport const getCookieKey = (key: string, prefix?: CookiePrefixOptions) => {\n  let finalKey = key\n  if (prefix) {\n    if (prefix === 'secure') {\n      finalKey = '__Secure-' + key\n    } else if (prefix === 'host') {\n      finalKey = '__Host-' + key\n    } else {\n      return undefined\n    }\n  }\n  return finalKey\n}\n\nexport function tryDecode(str: string) {\n  try {\n    return str.includes('%') ? decodeURIComponent(str) : str\n  } catch {\n    return str\n  }\n}\n\n/**\n * Parse an HTTP Cookie header string and returning an object of all cookie\n * name-value pairs.\n *\n * Inspired by https://github.com/unjs/cookie-es/blob/main/src/cookie/parse.ts\n *\n * @param str the string representing a `Cookie` header value\n */\nexport function parseCookies(str: string) {\n  if (typeof str !== 'string') {\n    throw new TypeError('argument str must be a string')\n  }\n\n  const cookies: Map<string, string> = new Map()\n\n  let index = 0\n  while (index < str.length) {\n    const eqIdx = str.indexOf('=', index)\n\n    if (eqIdx === -1) {\n      break\n    }\n\n    let endIdx = str.indexOf(';', index)\n\n    if (endIdx === -1) {\n      endIdx = str.length\n    } else if (endIdx < eqIdx) {\n      index = str.lastIndexOf(';', eqIdx - 1) + 1\n      continue\n    }\n\n    const key = str.slice(index, eqIdx).trim()\n    if (!cookies.has(key)) {\n      let val = str.slice(eqIdx + 1, endIdx).trim()\n      if (val.codePointAt(0) === 0x22) {\n        val = val.slice(1, -1)\n      }\n      cookies.set(key, tryDecode(val))\n    }\n\n    index = endIdx + 1\n  }\n\n  return cookies\n}\n\nexport const getSignedCookie = async (key: string, secret: string, headers: Headers, prefix?: CookiePrefixOptions) => {\n  const finalKey = getCookieKey(key, prefix)\n  if (!finalKey) {\n    return null\n  }\n  const cookieHeader = headers.get('cookie')\n  const parsedCookies = cookieHeader ? parseCookies(cookieHeader) : undefined\n  const value = parsedCookies?.get(finalKey)\n  if (!value) {\n    return null\n  }\n  const signatureStartPos = value.lastIndexOf('.')\n  if (signatureStartPos < 1) {\n    return null\n  }\n  const signedValue = value.substring(0, signatureStartPos)\n  const signature = value.substring(signatureStartPos + 1)\n  if (signature.length !== 44 || !signature.endsWith('=')) {\n    return null\n  }\n  const secretKey = await getCryptoKey(secret)\n  const isVerified = await verifySignature(signature, signedValue, secretKey)\n  return isVerified ? signedValue : false\n}\n"],"names":["subtle","verifySignature","base64Signature","value","secret","signatureBinStr","atob","signature","Uint8Array","length","i","len","charCodeAt","verify","algorithm","TextEncoder","encode","e","_serialize","key","opt","cookie","prefix","startsWith","secure","path","domain","undefined","maxAge","Error","Math","floor","expires","getTime","Date","now","toUTCString","httpOnly","sameSite","charAt","toUpperCase","slice","partitioned","name","hash","getCryptoKey","secretBuf","importKey","makeSignature","sign","btoa","String","fromCharCode","signCookieValue","encodeURIComponent","decodeURIComponent","serializeCookie","serializeSignedCookie","getCookieKey","finalKey","tryDecode","str","includes","parseCookies","TypeError","cookies","Map","index","eqIdx","indexOf","endIdx","lastIndexOf","trim","has","val","codePointAt","set","getSignedCookie","headers","cookieHeader","get","parsedCookies","signatureStartPos","signedValue","substring","endsWith","secretKey","isVerified"],"mappings":"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"}