mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-01-26 10:59:53 +00:00
## Summary Improves type safety in test files by replacing unsafe type patterns with proper TypeScript idioms. ## Changes - Define typed `TestWindow` interface extending `Window` for Playwright tests with custom properties - Use `Partial<HTMLElement>` with single type assertion for DOM element mocks - Remove redundant type imports - Fix `console.log` → `console.warn` in test fixture ## Files Changed 16 test files across browser_tests, packages, and src/components ## Test Plan - ✅ `pnpm typecheck` passes - ✅ No new `any` types introduced - ✅ All pre-commit hooks pass ┆Issue is synchronized with this [Notion page](https://www.notion.so/PR-8253-refactor-improve-TypeScript-patterns-in-test-files-Group-1-8-2f16d73d365081548f9ece7bcf0525ee) by [Unito](https://www.unito.io) --------- Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com>
384 lines
13 KiB
TypeScript
384 lines
13 KiB
TypeScript
import { expect } from '@playwright/test'
|
|
|
|
import { comfyPageFixture as test } from '../fixtures/ComfyPage'
|
|
|
|
test.beforeEach(async ({ comfyPage }) => {
|
|
await comfyPage.setSetting('Comfy.UseNewMenu', 'Disabled')
|
|
})
|
|
|
|
test.describe('Feature Flags', () => {
|
|
test('Client and server exchange feature flags on connection', async ({
|
|
comfyPage
|
|
}) => {
|
|
// Navigate to a new page to capture the initial WebSocket connection
|
|
const newPage = await comfyPage.page.context().newPage()
|
|
|
|
// Set up monitoring before navigation
|
|
await newPage.addInitScript(() => {
|
|
// This runs before any page scripts
|
|
window.__capturedMessages = {
|
|
clientFeatureFlags: null,
|
|
serverFeatureFlags: null
|
|
}
|
|
|
|
// Capture outgoing client messages
|
|
const originalSend = WebSocket.prototype.send
|
|
WebSocket.prototype.send = function (data) {
|
|
try {
|
|
const parsed = JSON.parse(data)
|
|
if (parsed.type === 'feature_flags') {
|
|
window.__capturedMessages!.clientFeatureFlags = parsed
|
|
}
|
|
} catch (e) {
|
|
// Not JSON, ignore
|
|
}
|
|
return originalSend.call(this, data)
|
|
}
|
|
|
|
// Monitor for server feature flags
|
|
const checkInterval = setInterval(() => {
|
|
if (
|
|
window['app']?.api?.serverFeatureFlags &&
|
|
Object.keys(window['app'].api.serverFeatureFlags).length > 0
|
|
) {
|
|
window.__capturedMessages!.serverFeatureFlags =
|
|
window['app'].api.serverFeatureFlags
|
|
clearInterval(checkInterval)
|
|
}
|
|
}, 100)
|
|
|
|
// Clear after 10 seconds
|
|
setTimeout(() => clearInterval(checkInterval), 10000)
|
|
})
|
|
|
|
// Navigate to the app
|
|
await newPage.goto(comfyPage.url)
|
|
|
|
// Wait for both client and server feature flags
|
|
await newPage.waitForFunction(
|
|
() =>
|
|
window.__capturedMessages!.clientFeatureFlags !== null &&
|
|
window.__capturedMessages!.serverFeatureFlags !== null,
|
|
{ timeout: 10000 }
|
|
)
|
|
|
|
// Get the captured messages
|
|
const messages = await newPage.evaluate(() => window.__capturedMessages)
|
|
|
|
// Verify client sent feature flags
|
|
expect(messages!.clientFeatureFlags).toBeTruthy()
|
|
expect(messages!.clientFeatureFlags).toHaveProperty('type', 'feature_flags')
|
|
expect(messages!.clientFeatureFlags).toHaveProperty('data')
|
|
expect(messages!.clientFeatureFlags!.data).toHaveProperty(
|
|
'supports_preview_metadata'
|
|
)
|
|
expect(
|
|
typeof messages!.clientFeatureFlags!.data.supports_preview_metadata
|
|
).toBe('boolean')
|
|
|
|
// Verify server sent feature flags back
|
|
expect(messages!.serverFeatureFlags).toBeTruthy()
|
|
expect(messages!.serverFeatureFlags).toHaveProperty(
|
|
'supports_preview_metadata'
|
|
)
|
|
expect(typeof messages!.serverFeatureFlags!.supports_preview_metadata).toBe(
|
|
'boolean'
|
|
)
|
|
expect(messages!.serverFeatureFlags).toHaveProperty('max_upload_size')
|
|
expect(typeof messages!.serverFeatureFlags!.max_upload_size).toBe('number')
|
|
expect(Object.keys(messages!.serverFeatureFlags!).length).toBeGreaterThan(0)
|
|
|
|
await newPage.close()
|
|
})
|
|
|
|
test('Server feature flags are received and accessible', async ({
|
|
comfyPage
|
|
}) => {
|
|
// Get the actual server feature flags from the backend
|
|
const serverFlags = await comfyPage.page.evaluate(() => {
|
|
return window['app']!.api.serverFeatureFlags
|
|
})
|
|
|
|
// Verify we received real feature flags from the backend
|
|
expect(serverFlags).toBeTruthy()
|
|
expect(Object.keys(serverFlags).length).toBeGreaterThan(0)
|
|
|
|
// The backend should send feature flags
|
|
expect(serverFlags).toHaveProperty('supports_preview_metadata')
|
|
expect(typeof serverFlags.supports_preview_metadata).toBe('boolean')
|
|
expect(serverFlags).toHaveProperty('max_upload_size')
|
|
expect(typeof serverFlags.max_upload_size).toBe('number')
|
|
})
|
|
|
|
test('serverSupportsFeature method works with real backend flags', async ({
|
|
comfyPage
|
|
}) => {
|
|
// Test serverSupportsFeature with real backend flags
|
|
const supportsPreviewMetadata = await comfyPage.page.evaluate(() => {
|
|
return window['app']!.api.serverSupportsFeature(
|
|
'supports_preview_metadata'
|
|
)
|
|
})
|
|
// The method should return a boolean based on the backend's value
|
|
expect(typeof supportsPreviewMetadata).toBe('boolean')
|
|
|
|
// Test non-existent feature - should always return false
|
|
const supportsNonExistent = await comfyPage.page.evaluate(() => {
|
|
return window['app']!.api.serverSupportsFeature(
|
|
'non_existent_feature_xyz'
|
|
)
|
|
})
|
|
expect(supportsNonExistent).toBe(false)
|
|
|
|
// Test that the method only returns true for boolean true values
|
|
const testResults = await comfyPage.page.evaluate(() => {
|
|
// Temporarily modify serverFeatureFlags to test behavior
|
|
const original = window['app']!.api.serverFeatureFlags
|
|
window['app']!.api.serverFeatureFlags = {
|
|
bool_true: true,
|
|
bool_false: false,
|
|
string_value: 'yes',
|
|
number_value: 1,
|
|
null_value: null
|
|
}
|
|
|
|
const results = {
|
|
bool_true: window['app']!.api.serverSupportsFeature('bool_true'),
|
|
bool_false: window['app']!.api.serverSupportsFeature('bool_false'),
|
|
string_value: window['app']!.api.serverSupportsFeature('string_value'),
|
|
number_value: window['app']!.api.serverSupportsFeature('number_value'),
|
|
null_value: window['app']!.api.serverSupportsFeature('null_value')
|
|
}
|
|
|
|
// Restore original
|
|
window['app']!.api.serverFeatureFlags = original
|
|
return results
|
|
})
|
|
|
|
// serverSupportsFeature should only return true for boolean true values
|
|
expect(testResults.bool_true).toBe(true)
|
|
expect(testResults.bool_false).toBe(false)
|
|
expect(testResults.string_value).toBe(false)
|
|
expect(testResults.number_value).toBe(false)
|
|
expect(testResults.null_value).toBe(false)
|
|
})
|
|
|
|
test('getServerFeature method works with real backend data', async ({
|
|
comfyPage
|
|
}) => {
|
|
// Test getServerFeature method
|
|
const previewMetadataValue = await comfyPage.page.evaluate(() => {
|
|
return window['app']!.api.getServerFeature('supports_preview_metadata')
|
|
})
|
|
expect(typeof previewMetadataValue).toBe('boolean')
|
|
|
|
// Test getting max_upload_size
|
|
const maxUploadSize = await comfyPage.page.evaluate(() => {
|
|
return window['app']!.api.getServerFeature('max_upload_size')
|
|
})
|
|
expect(typeof maxUploadSize).toBe('number')
|
|
expect(maxUploadSize).toBeGreaterThan(0)
|
|
|
|
// Test getServerFeature with default value for non-existent feature
|
|
const defaultValue = await comfyPage.page.evaluate(() => {
|
|
return window['app']!.api.getServerFeature(
|
|
'non_existent_feature_xyz',
|
|
'default'
|
|
)
|
|
})
|
|
expect(defaultValue).toBe('default')
|
|
})
|
|
|
|
test('getServerFeatures returns all backend feature flags', async ({
|
|
comfyPage
|
|
}) => {
|
|
// Test getServerFeatures returns all flags
|
|
const allFeatures = await comfyPage.page.evaluate(() => {
|
|
return window['app']!.api.getServerFeatures()
|
|
})
|
|
|
|
expect(allFeatures).toBeTruthy()
|
|
expect(allFeatures).toHaveProperty('supports_preview_metadata')
|
|
expect(typeof allFeatures.supports_preview_metadata).toBe('boolean')
|
|
expect(allFeatures).toHaveProperty('max_upload_size')
|
|
expect(Object.keys(allFeatures).length).toBeGreaterThan(0)
|
|
})
|
|
|
|
test('Client feature flags are immutable', async ({ comfyPage }) => {
|
|
// Test that getClientFeatureFlags returns a copy
|
|
const immutabilityTest = await comfyPage.page.evaluate(() => {
|
|
const flags1 = window['app']!.api.getClientFeatureFlags()
|
|
const flags2 = window['app']!.api.getClientFeatureFlags()
|
|
|
|
// Modify the first object
|
|
flags1.test_modification = true
|
|
|
|
// Get flags again to check if original was modified
|
|
const flags3 = window['app']!.api.getClientFeatureFlags()
|
|
|
|
return {
|
|
areEqual: flags1 === flags2,
|
|
hasModification: flags3.test_modification !== undefined,
|
|
hasSupportsPreview: flags1.supports_preview_metadata !== undefined,
|
|
supportsPreviewValue: flags1.supports_preview_metadata
|
|
}
|
|
})
|
|
|
|
// Verify they are different objects (not the same reference)
|
|
expect(immutabilityTest.areEqual).toBe(false)
|
|
|
|
// Verify modification didn't affect the original
|
|
expect(immutabilityTest.hasModification).toBe(false)
|
|
|
|
// Verify the flags contain expected properties
|
|
expect(immutabilityTest.hasSupportsPreview).toBe(true)
|
|
expect(typeof immutabilityTest.supportsPreviewValue).toBe('boolean') // From clientFeatureFlags.json
|
|
})
|
|
|
|
test('Server features are immutable when accessed via getServerFeatures', async ({
|
|
comfyPage
|
|
}) => {
|
|
const immutabilityTest = await comfyPage.page.evaluate(() => {
|
|
// Get a copy of server features
|
|
const features1 = window['app']!.api.getServerFeatures()
|
|
|
|
// Try to modify it
|
|
features1.supports_preview_metadata = false
|
|
features1.new_feature = 'added'
|
|
|
|
// Get another copy
|
|
const features2 = window['app']!.api.getServerFeatures()
|
|
|
|
return {
|
|
modifiedValue: features1.supports_preview_metadata,
|
|
originalValue: features2.supports_preview_metadata,
|
|
hasNewFeature: features2.new_feature !== undefined,
|
|
hasSupportsPreview: features2.supports_preview_metadata !== undefined
|
|
}
|
|
})
|
|
|
|
// The modification should only affect the copy
|
|
expect(immutabilityTest.modifiedValue).toBe(false)
|
|
expect(typeof immutabilityTest.originalValue).toBe('boolean') // Backend sends boolean for supports_preview_metadata
|
|
expect(immutabilityTest.hasNewFeature).toBe(false)
|
|
expect(immutabilityTest.hasSupportsPreview).toBe(true)
|
|
})
|
|
|
|
test('Feature flags are negotiated early in connection lifecycle', async ({
|
|
comfyPage
|
|
}) => {
|
|
// This test verifies that feature flags are available early in the app lifecycle
|
|
// which is important for protocol negotiation
|
|
|
|
// Create a new page to ensure clean state
|
|
const newPage = await comfyPage.page.context().newPage()
|
|
|
|
// Set up monitoring before navigation
|
|
await newPage.addInitScript(() => {
|
|
// Track when various app components are ready
|
|
|
|
window.__appReadiness = {
|
|
featureFlagsReceived: false,
|
|
apiInitialized: false,
|
|
appInitialized: false
|
|
}
|
|
|
|
// Monitor when feature flags arrive by checking periodically
|
|
const checkFeatureFlags = setInterval(() => {
|
|
if (
|
|
window['app']?.api?.serverFeatureFlags?.supports_preview_metadata !==
|
|
undefined
|
|
) {
|
|
window.__appReadiness = {
|
|
...window.__appReadiness,
|
|
featureFlagsReceived: true
|
|
}
|
|
clearInterval(checkFeatureFlags)
|
|
}
|
|
}, 10)
|
|
|
|
// Monitor API initialization
|
|
const checkApi = setInterval(() => {
|
|
if (window['app']?.api) {
|
|
window.__appReadiness = {
|
|
...window.__appReadiness,
|
|
apiInitialized: true
|
|
}
|
|
clearInterval(checkApi)
|
|
}
|
|
}, 10)
|
|
|
|
// Monitor app initialization
|
|
const checkApp = setInterval(() => {
|
|
if (window['app']?.graph) {
|
|
window.__appReadiness = {
|
|
...window.__appReadiness,
|
|
appInitialized: true
|
|
}
|
|
clearInterval(checkApp)
|
|
}
|
|
}, 10)
|
|
|
|
// Clean up after 10 seconds
|
|
setTimeout(() => {
|
|
clearInterval(checkFeatureFlags)
|
|
clearInterval(checkApi)
|
|
clearInterval(checkApp)
|
|
}, 10000)
|
|
})
|
|
|
|
// Navigate to the app
|
|
await newPage.goto(comfyPage.url)
|
|
|
|
// Wait for feature flags to be received
|
|
await newPage.waitForFunction(
|
|
() =>
|
|
window['app']?.api?.serverFeatureFlags?.supports_preview_metadata !==
|
|
undefined,
|
|
{
|
|
timeout: 10000
|
|
}
|
|
)
|
|
|
|
// Get readiness state
|
|
const readiness = await newPage.evaluate(() => {
|
|
return {
|
|
...window.__appReadiness,
|
|
currentFlags: window['app']!.api.serverFeatureFlags
|
|
}
|
|
})
|
|
|
|
// Verify feature flags are available
|
|
expect(readiness.currentFlags).toHaveProperty('supports_preview_metadata')
|
|
expect(typeof readiness.currentFlags.supports_preview_metadata).toBe(
|
|
'boolean'
|
|
)
|
|
expect(readiness.currentFlags).toHaveProperty('max_upload_size')
|
|
|
|
// Verify feature flags were received (we detected them via polling)
|
|
expect(readiness.featureFlagsReceived).toBe(true)
|
|
|
|
// Verify API was initialized (feature flags require API)
|
|
expect(readiness.apiInitialized).toBe(true)
|
|
|
|
await newPage.close()
|
|
})
|
|
|
|
test('Backend /features endpoint returns feature flags', async ({
|
|
comfyPage
|
|
}) => {
|
|
// Test the HTTP endpoint directly
|
|
const response = await comfyPage.page.request.get(
|
|
`${comfyPage.url}/api/features`
|
|
)
|
|
expect(response.ok()).toBe(true)
|
|
|
|
const features = await response.json()
|
|
expect(features).toBeTruthy()
|
|
expect(features).toHaveProperty('supports_preview_metadata')
|
|
expect(typeof features.supports_preview_metadata).toBe('boolean')
|
|
expect(features).toHaveProperty('max_upload_size')
|
|
expect(Object.keys(features).length).toBeGreaterThan(0)
|
|
})
|
|
})
|