mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-04-21 23:09:39 +00:00
Compare commits
2 Commits
pr5-list-v
...
cloud/fix-
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6a5fcc5357 | ||
|
|
52876da4a9 |
@@ -1,3 +1,5 @@
|
|||||||
|
import { createSingletonPromise } from '@vueuse/core'
|
||||||
|
|
||||||
import { api } from '@/scripts/api'
|
import { api } from '@/scripts/api'
|
||||||
import { isCloud } from '@/platform/distribution/types'
|
import { isCloud } from '@/platform/distribution/types'
|
||||||
import { useFirebaseAuthStore } from '@/stores/firebaseAuthStore'
|
import { useFirebaseAuthStore } from '@/stores/firebaseAuthStore'
|
||||||
@@ -12,18 +14,86 @@ export const useSessionCookie = () => {
|
|||||||
* Called after login and on token refresh.
|
* Called after login and on token refresh.
|
||||||
*/
|
*/
|
||||||
const createSession = async (): Promise<void> => {
|
const createSession = async (): Promise<void> => {
|
||||||
|
if (!isCloud || logoutInProgress) return
|
||||||
|
|
||||||
|
const promise = createSessionSingleton()
|
||||||
|
inFlightCreateSession = promise
|
||||||
|
|
||||||
|
try {
|
||||||
|
await promise
|
||||||
|
} finally {
|
||||||
|
if (inFlightCreateSession === promise) {
|
||||||
|
inFlightCreateSession = null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Deletes the session cookie.
|
||||||
|
* Called on logout.
|
||||||
|
*/
|
||||||
|
const deleteSession = async (): Promise<void> => {
|
||||||
if (!isCloud) return
|
if (!isCloud) return
|
||||||
|
|
||||||
const authStore = useFirebaseAuthStore()
|
logoutInProgress = true
|
||||||
const authHeader = await authStore.getAuthHeader()
|
|
||||||
|
|
||||||
if (!authHeader) {
|
try {
|
||||||
throw new Error('No auth header available for session creation')
|
if (inFlightCreateSession) {
|
||||||
|
currentCreateController?.abort()
|
||||||
|
try {
|
||||||
|
await inFlightCreateSession
|
||||||
|
} catch (error: unknown) {
|
||||||
|
if (!isAbortError(error)) {
|
||||||
|
throw error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch(api.apiURL('/auth/session'), {
|
||||||
|
method: 'DELETE',
|
||||||
|
credentials: 'include'
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
const errorData = await response.json().catch(() => ({}))
|
||||||
|
throw new Error(
|
||||||
|
`Failed to delete session: ${
|
||||||
|
errorData.message || response.statusText
|
||||||
|
}`
|
||||||
|
)
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
logoutInProgress = false
|
||||||
|
await createSessionSingleton.reset()
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
createSession,
|
||||||
|
deleteSession
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let inFlightCreateSession: Promise<void> | null = null
|
||||||
|
let currentCreateController: AbortController | null = null
|
||||||
|
let logoutInProgress = false
|
||||||
|
|
||||||
|
const createSessionSingleton = createSingletonPromise(async () => {
|
||||||
|
const authStore = useFirebaseAuthStore()
|
||||||
|
const authHeader = await authStore.getAuthHeader()
|
||||||
|
|
||||||
|
if (!authHeader) {
|
||||||
|
throw new Error('No auth header available for session creation')
|
||||||
|
}
|
||||||
|
|
||||||
|
const controller = new AbortController()
|
||||||
|
currentCreateController = controller
|
||||||
|
|
||||||
|
try {
|
||||||
const response = await fetch(api.apiURL('/auth/session'), {
|
const response = await fetch(api.apiURL('/auth/session'), {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
credentials: 'include',
|
credentials: 'include',
|
||||||
|
signal: controller.signal,
|
||||||
headers: {
|
headers: {
|
||||||
...authHeader,
|
...authHeader,
|
||||||
'Content-Type': 'application/json'
|
'Content-Type': 'application/json'
|
||||||
@@ -36,30 +106,19 @@ export const useSessionCookie = () => {
|
|||||||
`Failed to create session: ${errorData.message || response.statusText}`
|
`Failed to create session: ${errorData.message || response.statusText}`
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
} catch (error: unknown) {
|
||||||
|
if (!isAbortError(error)) {
|
||||||
/**
|
throw error
|
||||||
* Deletes the session cookie.
|
}
|
||||||
* Called on logout.
|
} finally {
|
||||||
*/
|
if (currentCreateController === controller) {
|
||||||
const deleteSession = async (): Promise<void> => {
|
currentCreateController = null
|
||||||
if (!isCloud) return
|
|
||||||
|
|
||||||
const response = await fetch(api.apiURL('/auth/session'), {
|
|
||||||
method: 'DELETE',
|
|
||||||
credentials: 'include'
|
|
||||||
})
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
const errorData = await response.json().catch(() => ({}))
|
|
||||||
throw new Error(
|
|
||||||
`Failed to delete session: ${errorData.message || response.statusText}`
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
})
|
||||||
|
|
||||||
return {
|
const isAbortError = (error: unknown): boolean => {
|
||||||
createSession,
|
if (!error || typeof error !== 'object') return false
|
||||||
deleteSession
|
const name = 'name' in error ? (error as { name?: string }).name : undefined
|
||||||
}
|
return name === 'AbortError'
|
||||||
}
|
}
|
||||||
|
|||||||
139
tests-ui/platform/auth/session/useSessionCookie.test.ts
Normal file
139
tests-ui/platform/auth/session/useSessionCookie.test.ts
Normal file
@@ -0,0 +1,139 @@
|
|||||||
|
import { describe, expect, it, vi } from 'vitest'
|
||||||
|
|
||||||
|
const makeSuccessResponse = () =>
|
||||||
|
new Response('{}', {
|
||||||
|
status: 200,
|
||||||
|
statusText: 'OK',
|
||||||
|
headers: { 'Content-Type': 'application/json' }
|
||||||
|
})
|
||||||
|
|
||||||
|
type Deferred<T> = {
|
||||||
|
promise: Promise<T>
|
||||||
|
resolve: (value: T) => void
|
||||||
|
reject: (reason: unknown) => void
|
||||||
|
}
|
||||||
|
|
||||||
|
const createDeferred = <T>(): Deferred<T> => {
|
||||||
|
let resolve: (value: T) => void
|
||||||
|
let reject: (reason: unknown) => void
|
||||||
|
|
||||||
|
const promise = new Promise<T>((res, rej) => {
|
||||||
|
resolve = res
|
||||||
|
reject = rej
|
||||||
|
})
|
||||||
|
|
||||||
|
// @ts-expect-error initialized via closure assignments above
|
||||||
|
return { promise, resolve, reject }
|
||||||
|
}
|
||||||
|
|
||||||
|
const mockModules = async () => {
|
||||||
|
vi.resetModules()
|
||||||
|
|
||||||
|
const getAuthHeader = vi.fn(async () => ({ Authorization: 'Bearer token' }))
|
||||||
|
|
||||||
|
vi.doMock('@/scripts/api', () => ({
|
||||||
|
api: {
|
||||||
|
apiURL: vi.fn((path: string) => `/api${path}`)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
|
||||||
|
vi.doMock('@/platform/distribution/types', () => ({
|
||||||
|
isCloud: true
|
||||||
|
}))
|
||||||
|
|
||||||
|
vi.doMock('@/stores/firebaseAuthStore', () => ({
|
||||||
|
useFirebaseAuthStore: vi.fn(() => ({
|
||||||
|
getAuthHeader
|
||||||
|
}))
|
||||||
|
}))
|
||||||
|
|
||||||
|
const module = await import('@/platform/auth/session/useSessionCookie')
|
||||||
|
return { getAuthHeader, useSessionCookie: module.useSessionCookie }
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('useSessionCookie', () => {
|
||||||
|
it('deduplicates in-flight session creation', async () => {
|
||||||
|
const { useSessionCookie, getAuthHeader } = await mockModules()
|
||||||
|
|
||||||
|
const postDeferred = createDeferred<Response>()
|
||||||
|
|
||||||
|
const fetchSpy = vi
|
||||||
|
.spyOn(globalThis, 'fetch')
|
||||||
|
.mockImplementation(() => postDeferred.promise)
|
||||||
|
|
||||||
|
const { createSession } = useSessionCookie()
|
||||||
|
|
||||||
|
const firstCall = createSession()
|
||||||
|
const secondCall = createSession()
|
||||||
|
|
||||||
|
await Promise.resolve()
|
||||||
|
|
||||||
|
expect(fetchSpy).toHaveBeenCalledTimes(1)
|
||||||
|
expect(getAuthHeader).toHaveBeenCalledTimes(1)
|
||||||
|
|
||||||
|
postDeferred.resolve(makeSuccessResponse())
|
||||||
|
await expect(firstCall).resolves.toBeUndefined()
|
||||||
|
await expect(secondCall).resolves.toBeUndefined()
|
||||||
|
|
||||||
|
fetchSpy.mockRestore()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('aborts pending create on logout and skips new ones while logout is in progress', async () => {
|
||||||
|
const { useSessionCookie, getAuthHeader } = await mockModules()
|
||||||
|
|
||||||
|
const firstPostDeferred = createDeferred<Response>()
|
||||||
|
const deleteDeferred = createDeferred<Response>()
|
||||||
|
|
||||||
|
let capturedSignal: AbortSignal | undefined
|
||||||
|
|
||||||
|
const fetchSpy = vi.spyOn(globalThis, 'fetch')
|
||||||
|
fetchSpy
|
||||||
|
.mockImplementationOnce((_, init?: RequestInit) => {
|
||||||
|
capturedSignal = init?.signal as AbortSignal | undefined
|
||||||
|
return firstPostDeferred.promise
|
||||||
|
})
|
||||||
|
.mockImplementationOnce((_, init?: RequestInit) => {
|
||||||
|
expect(init?.method).toBe('DELETE')
|
||||||
|
return deleteDeferred.promise
|
||||||
|
})
|
||||||
|
.mockImplementation((_, init?: RequestInit) => {
|
||||||
|
if (init?.method === 'POST') {
|
||||||
|
return Promise.resolve(makeSuccessResponse())
|
||||||
|
}
|
||||||
|
return Promise.resolve(makeSuccessResponse())
|
||||||
|
})
|
||||||
|
|
||||||
|
const { createSession, deleteSession } = useSessionCookie()
|
||||||
|
|
||||||
|
const createPromise = createSession()
|
||||||
|
|
||||||
|
await Promise.resolve()
|
||||||
|
|
||||||
|
const logoutPromise = deleteSession()
|
||||||
|
|
||||||
|
await Promise.resolve()
|
||||||
|
|
||||||
|
expect(fetchSpy).toHaveBeenCalledTimes(1)
|
||||||
|
expect(capturedSignal?.aborted).toBe(true)
|
||||||
|
|
||||||
|
const abortError = new Error('aborted')
|
||||||
|
abortError.name = 'AbortError'
|
||||||
|
firstPostDeferred.reject(abortError)
|
||||||
|
await expect(createPromise).resolves.toBeUndefined()
|
||||||
|
|
||||||
|
await Promise.resolve()
|
||||||
|
expect(fetchSpy).toHaveBeenCalledTimes(2)
|
||||||
|
expect(getAuthHeader).toHaveBeenCalledTimes(1)
|
||||||
|
|
||||||
|
await expect(createSession()).resolves.toBeUndefined()
|
||||||
|
expect(fetchSpy).toHaveBeenCalledTimes(2)
|
||||||
|
|
||||||
|
deleteDeferred.resolve(makeSuccessResponse())
|
||||||
|
await expect(logoutPromise).resolves.toBeUndefined()
|
||||||
|
|
||||||
|
await expect(createSession()).resolves.toBeUndefined()
|
||||||
|
expect(fetchSpy).toHaveBeenCalledTimes(3)
|
||||||
|
|
||||||
|
fetchSpy.mockRestore()
|
||||||
|
})
|
||||||
|
})
|
||||||
Reference in New Issue
Block a user