Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[PR changes #392] feat: move pointer and sessionDataType to the SessionConfig #592

Merged
merged 39 commits into from
May 16, 2024
Merged
Show file tree
Hide file tree
Changes from 27 commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
e44b7c7
feat: add generic to "jsonPointerGet"
Danielwinkelmann May 14, 2023
adfb6b1
feat: implement the sessionDataJsonPointer
Danielwinkelmann May 14, 2023
ad8231d
feat: add to docs
Danielwinkelmann May 14, 2023
83087ce
Merge branch 'main' into feat/json-pointer-extractor
Danielwinkelmann May 15, 2023
63cd53e
Merge branch 'main' into feat/json-pointer-extractor
Danielwinkelmann May 15, 2023
f7730c6
Merge branch 'main' into feat/json-pointer-extractor
zoey-kaiser Oct 11, 2023
20d18a6
feat: move pointer and sessionDataType to the SessionConfig
Nov 30, 2023
d4761b4
fix: remove default baseUrl
Nov 30, 2023
65dbc63
Merge branch 'refs/heads/main' into feat/json-pointer-extractor
Nov 30, 2023
01ca023
fix merge conflicts from main
Nov 30, 2023
9fe12da
fix: merge conflict with SessionConfig
Nov 30, 2023
f034257
reset to main branch state
Nov 30, 2023
e788160
fix: lint
Nov 30, 2023
4aa8a12
feat: make session refresh params optional
Dec 5, 2023
3ad2296
feat: rebase `jsonPointerGet` function on @danielwinkelmann's origina…
Dec 5, 2023
df753ae
refacto: set the return type when extracting json pointers
Dec 5, 2023
134d70f
fix: should use the pointer to retrieve the user session not datatype…
Dec 5, 2023
53b6edd
revert: import as type
Dec 5, 2023
1ca6e69
Merge branch 'main' into feat/json-pointer-extractor
valh1996 Dec 5, 2023
2044dce
fix lint
Dec 5, 2023
4613e42
fix: type issue - forgotten changes from daniel
Dec 5, 2023
84b2128
Merge branch 'main' into feat/json-pointer-extractor
zoey-kaiser Dec 11, 2023
ca2af93
move the session data type back to the providers (local and refresh)
Jan 8, 2024
15e0b7e
Merge branch 'main' into feat/json-pointer-extractor
valh1996 Jan 8, 2024
4387181
forget to remove baseurl while testing
Jan 8, 2024
6fade21
fix: use config directly
Jan 8, 2024
d596f39
docs: add session data type to the refresh docs
Jan 9, 2024
24d165c
Merge branch 'main' of github.com:sidebase/nuxt-auth into feat/json-p…
Feb 22, 2024
f7fde1f
Merge branch 'main' into feat/json-pointer-extractor
phoenix-ru Feb 22, 2024
cd55182
Merge branch 'main' into feat/json-pointer-extractor
zoey-kaiser Feb 23, 2024
eacf639
Merge branch 'main' into feat/json-pointer-extractor
phoenix-ru Mar 14, 2024
fff8137
Merge branch 'main' into feat/json-pointer-extractor
zoey-kaiser Mar 14, 2024
9af5920
fix(playground): reset to default pointer '/'
valh1996 Mar 14, 2024
7f2ad14
Merge branch 'main' into feat/json-pointer-extractor
phoenix-ru May 16, 2024
e43a549
Update useAuth.ts
phoenix-ru May 16, 2024
8aa3e2b
Update useAuth.ts
phoenix-ru May 16, 2024
9f8a71c
Update helpers.ts
phoenix-ru May 16, 2024
4381445
Update helpers.ts
phoenix-ru May 16, 2024
697b5da
fix: usage of `/` in json pointer get
phoenix-ru May 16, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
66 changes: 50 additions & 16 deletions docs/content/2.configuration/2.nuxt-config.md
Original file line number Diff line number Diff line change
Expand Up @@ -219,14 +219,31 @@ type ProviderLocal = {
*/
sameSiteAttribute?: boolean | 'lax' | 'strict' | 'none' | undefined,
},
/**
* Define an interface for the session data object that `nuxt-auth` expects to receive from the `getSession` endpoint.
*
* @default { id: 'string | number' }
* @example { id: 'string', name: 'string', email: 'string' }
* @advanced_array_example { id: 'string', email: 'string', name: 'string', role: 'admin | guest | account', subscriptions: "{ id: number, status: 'ACTIVE' | 'INACTIVE' }[]" }
/*
* Settings for the session-data that `nuxt-auth` receives from the `getSession` endpoint.
*/
sessionDataType?: SessionDataObject,
session?: {
/**
* Define an interface for the session data object that `nuxt-auth` expects to receive from the `getSession` endpoint.
*
* @default { id: 'string | number' }
* @example { id: 'string', name: 'string', email: 'string' }
* @advanced_array_example { id: 'string', email: 'string', name: 'string', role: 'admin | guest | account', subscriptions: "{ id: number, status: 'ACTIVE' | 'INACTIVE' }[]" }
*/
dataType?: SessionDataObject;
/**
* How to extract the session-data from the session response.
*
* E.g., setting this to `/data/user` and returning an object like `{ data: { user: { id:number, name: string } }, status: 'ok' }` from the `getSession` endpoint will
* storing the 'User' object typed as the type created via the 'dataType' prop.
*
* This follows the JSON Pointer standard, see it's RFC6901 here: https://www.rfc-editor.org/rfc/rfc6901
*
* @default / Access the root of the session response object
* @example /data/user Access the `data/user` property of the session response object
*/
dataResponsePointer?: string;
}
}

```
Expand Down Expand Up @@ -364,7 +381,7 @@ type ProviderRefresh = {
* @default /refreshToken Access the `refreshToken` property of the sign-in response object
* @example / Access the root of the sign-in response object, useful when your endpoint returns a plain, non-object string as the refreshToken
*/
signInResponseRefreshTokenPointer?: string
signInResponseRefreshTokenPointer?: string,
/**
* Maximum age to store the authentication token for. After the expiry time the token is automatically deleted on the application side, i.e., in the users' browser.
*
Expand All @@ -375,14 +392,31 @@ type ProviderRefresh = {
*/
maxAgeInSeconds?: number,
},
/**
* Define an interface for the session data object that `nuxt-auth` expects to receive from the `getSession` endpoint.
*
* @default { id: 'string | number' }
* @example { id: 'string', name: 'string', email: 'string' }
* @advanced_array_example { id: 'string', email: 'string', name: 'string', role: 'admin | guest | account', subscriptions: "{ id: number, status: 'ACTIVE' | 'INACTIVE' }[]" }
/*
* Settings for the session-data that `nuxt-auth` receives from the `getSession` endpoint.
*/
sessionDataType?: SessionDataObject,
session?: {
/**
* Define an interface for the session data object that `nuxt-auth` expects to receive from the `getSession` endpoint.
*
* @default { id: 'string | number' }
* @example { id: 'string', name: 'string', email: 'string' }
* @advanced_array_example { id: 'string', email: 'string', name: 'string', role: 'admin | guest | account', subscriptions: "{ id: number, status: 'ACTIVE' | 'INACTIVE' }[]" }
*/
dataType?: SessionDataObject;
/**
* How to extract the session-data from the session response.
*
* E.g., setting this to `/data/user` and returning an object like `{ data: { user: { id:number, name: string } }, status: 'ok' }` from the `getSession` endpoint will
* storing the 'User' object typed as the type created via the 'dataType' prop.
*
* This follows the JSON Pointer standard, see it's RFC6901 here: https://www.rfc-editor.org/rfc/rfc6901
*
* @default / Access the root of the session response object
* @example /data/user Access the `data/user` property of the session response object
*/
dataResponsePointer?: string;
}
}
```
```ts [SessionConfig]
Expand All @@ -401,7 +435,7 @@ type SessionConfig = {
* @default false
*
*/
enableRefreshPeriodically: number | boolean
enableRefreshPeriodically: number | boolean,
/**
* Whether to refresh the session every time the browser window is refocused.
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -171,7 +171,7 @@ inferface SessionData {
id: string | number
}

// Option B: You configured `auth.provider.sessionDataType` to something like ` { id: 'string', email: 'string', name: 'string', role: 'admin | guest | account' }`
// Option B: You configured `auth.provider.session.dataType` to something like ` { id: 'string', email: 'string', name: 'string', role: 'admin | guest | account' }`
inferface SessionData {
id: string
email: string
Expand All @@ -181,7 +181,7 @@ inferface SessionData {
```
::

### About `auth.provider.sessionDataType`
### About `auth.provider.session.dataType`

This is a configuration option available to dynamically type the `SessionData` that the `local` provider will return when accessing `data.value`. Read more about this in the [nuxt.config.ts configuration documentation](/nuxt-auth/v0.6/configuration/nuxt-config) of the `local` provider.

Expand Down
6 changes: 4 additions & 2 deletions playground-local/nuxt.config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,12 +15,14 @@ export default defineNuxtConfig({
token: {
signInResponseTokenPointer: '/token/accessToken'
},
sessionDataType: { id: 'string', email: 'string', name: 'string', role: 'admin | guest | account', subscriptions: "{ id: number, status: 'ACTIVE' | 'INACTIVE' }[]" }
session: {
dataType: { id: 'string', email: 'string', name: 'string', role: 'admin | guest | account', subscriptions: "{ id: number, status: 'ACTIVE' | 'INACTIVE' }[]" },
dataResponsePointer: '/data/user'
valh1996 marked this conversation as resolved.
Show resolved Hide resolved
}
},
session: {
// Whether to refresh the session every time the browser window is refocused.
enableRefreshOnWindowFocus: true,

// Whether to refresh the session every `X` milliseconds. Set this to `false` to turn it off. The session will only be refreshed if a session already exists.
enableRefreshPeriodically: 5000
},
Expand Down
12 changes: 9 additions & 3 deletions src/module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,10 @@ const defaultsByBackend: {
maxAgeInSeconds: 30 * 60,
sameSiteAttribute: 'lax'
},
sessionDataType: { id: 'string | number' }
session: {
dataType: { id: 'string | number' },
dataResponsePointer: '/'
}
},

refresh: {
Expand All @@ -82,7 +85,10 @@ const defaultsByBackend: {
signInResponseRefreshTokenPointer: '/refreshToken',
maxAgeInSeconds: 60 * 60 * 24 * 7 // 7 days
},
sessionDataType: { id: 'string | number' }
session: {
dataType: { id: 'string | number' },
dataResponsePointer: '/'
}
},

authjs: {
Expand Down Expand Up @@ -198,7 +204,7 @@ export default defineNuxtModule<ModuleOptions>({
options.provider.type === 'local'
? genInterface(
'SessionData',
(options.provider as any).sessionDataType
(options.provider.session as any).dataType
)
: '',
'}'
Expand Down
12 changes: 9 additions & 3 deletions src/runtime/composables/local/useAuth.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ const signIn: SignInFunc<Credentials, any> = async (credentials, signInOptions,
params: signInParams ?? {}
})

const extractedToken = jsonPointerGet(response, config.token.signInResponseTokenPointer)
const extractedToken = jsonPointerGet<string>(response, config.token.signInResponseTokenPointer)
if (typeof extractedToken !== 'string') {
console.error(`Auth: string token expected, received instead: ${JSON.stringify(extractedToken)}. Tried to find token at ${config.token.signInResponseTokenPointer} in ${JSON.stringify(response)}`)
return
Expand Down Expand Up @@ -84,8 +84,14 @@ const getSession: GetSessionFunc<SessionData | null | void> = async (getSessionO

loading.value = true
try {
data.value = await _fetch<SessionData>(nuxt, path, { method, headers })
} catch {
const result = await _fetch<any>(nuxt, path, { method, headers })
const { dataResponsePointer: sessionDataResponsePointer } = config.session
data.value = jsonPointerGet<SessionData>(result, sessionDataResponsePointer)
} catch (err) {
if (!data.value && err instanceof Error) {
console.error(`Session: unable to extract session, ${err.message}`)
}

// Clear all data: Request failed so we must not be authenticated
data.value = null
rawToken.value = null
Expand Down
8 changes: 4 additions & 4 deletions src/runtime/composables/refresh/useAuth.ts
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ const signIn: ReturnType<typeof useLocalAuth>['signIn'] = async (
params: signInParams ?? {}
})

const extractedToken = jsonPointerGet(
const extractedToken = jsonPointerGet<string>(
response,
config.token.signInResponseTokenPointer
)
Expand All @@ -47,7 +47,7 @@ const signIn: ReturnType<typeof useLocalAuth>['signIn'] = async (
return
}

const extractedRefreshToken = jsonPointerGet(
const extractedRefreshToken = jsonPointerGet<string>(
response,
config.refreshToken.signInResponseRefreshTokenPointer
)
Expand Down Expand Up @@ -96,7 +96,7 @@ const refresh = async () => {
}
})

const extractedToken = jsonPointerGet(
const extractedToken = jsonPointerGet<string>(
response,
config.token.signInResponseTokenPointer
)
Expand All @@ -112,7 +112,7 @@ const refresh = async () => {
}

if (!config.refreshOnlyToken) {
const extractedRefreshToken = jsonPointerGet(
const extractedRefreshToken = jsonPointerGet<string>(
response,
config.refreshToken.signInResponseRefreshTokenPointer
)
Expand Down
18 changes: 12 additions & 6 deletions src/runtime/helpers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -49,10 +49,15 @@ export const useTypedBackendConfig = <T extends SupportedAuthProviders>(
* @param obj
* @param pointer
*/
export const jsonPointerGet = (
obj: Record<string, any>,
export const jsonPointerGet = <TResult, T extends object = object>(
obj: T,
pointer: string
): string | Record<string, any> => {
): TResult => {
let result = obj as unknown as TResult
if (pointer === '/') {
return result
}

const unescape = (str: string) => str.replace(/~1/g, '/').replace(/~0/g, '~')
const parse = (pointer: string) => {
if (pointer === '') {
Expand All @@ -68,10 +73,11 @@ export const jsonPointerGet = (

for (let i = 0; i < refTokens.length; ++i) {
const tok = refTokens[i]
if (!(typeof obj === 'object' && tok in obj)) {
if (!(typeof result === 'object' && result && tok in result)) {
throw new Error('Invalid reference token: ' + tok)
}
obj = obj[tok]
result = (result as any)[tok]
}
return obj

return result
}
4 changes: 2 additions & 2 deletions src/runtime/server/plugins/refresh-token.server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ export default defineNuxtPlugin({
headers
})

const extractedToken = jsonPointerGet(
const extractedToken = jsonPointerGet<string>(
response,
config.provider.token.signInResponseTokenPointer
)
Expand All @@ -44,7 +44,7 @@ export default defineNuxtPlugin({

// check if refereshTokenOnly
if (!configToken.refreshOnlyToken) {
const extractedRefreshToken = jsonPointerGet(
const extractedRefreshToken = jsonPointerGet<string>(
response,
config.provider.refreshToken.signInResponseRefreshTokenPointer
)
Expand Down
29 changes: 23 additions & 6 deletions src/runtime/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -162,13 +162,30 @@ type ProviderLocal = {
sameSiteAttribute?: boolean | 'lax' | 'strict' | 'none' | undefined;
};
/**
* Define an interface for the session data object that `nuxt-auth` expects to receive from the `getSession` endpoint.
*
* @default { id: 'string | number' }
* @example { id: 'string', name: 'string', email: 'string' }
* @advanced_array_example { id: 'string', email: 'string', name: 'string', role: 'admin | guest | account', subscriptions: "{ id: number, status: 'ACTIVE' | 'INACTIVE' }[]" }
* Settings for the session-data that `nuxt-auth` receives from the `getSession` endpoint.
*/
sessionDataType?: SessionDataObject;
session?: {
/*
* Define an interface for the session data object that `nuxt-auth` expects to receive from the `getSession` endpoint.
*
* @default { id: 'string | number' }
* @example { id: 'string', name: 'string', email: 'string' }
* @advanced_array_example { id: 'string', email: 'string', name: 'string', role: 'admin | guest | account', subscriptions: "{ id: number, status: 'ACTIVE' | 'INACTIVE' }[]" }
*/
dataType?: SessionDataObject;
/**
* How to extract the session-data from the session response.
*
* E.g., setting this to `/data/user` and returning an object like `{ data: { user: { id:number, name: string } }, status: 'ok' }` from the `getSession` endpoint will
* storing the 'User' object typed as the type created via the 'dataType' prop.
*
* This follows the JSON Pointer standard, see it's RFC6901 here: https://www.rfc-editor.org/rfc/rfc6901
*
* @default / Access the root of the session response object
* @example /data/user Access the `data/user` property of the session response object
*/
dataResponsePointer?: string;
};
};

/**
Expand Down