fix(ui): update breadcrumbs in custom property page (#23432)

* fix: missing entries in custom properties enum

* add tests

* fix unit test warnings
This commit is contained in:
Karan Hotchandani 2025-09-17 16:54:49 +05:30 committed by GitHub
parent 7ef792db59
commit 8528625ae9
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
9 changed files with 430 additions and 36 deletions

View File

@ -570,7 +570,7 @@ export const addCustomPropertiesForEntity = async ({
}: {
page: Page;
propertyName: string;
customPropertyData: { description: string };
customPropertyData: { description: string; entityApiType: string };
customType: string;
enumConfig?: { values: string[]; multiSelect: boolean };
formatConfig?: string;
@ -580,6 +580,20 @@ export const addCustomPropertiesForEntity = async ({
// Add Custom property for selected entity
await page.click('[data-testid="add-field-button"]');
// Assert that breadcrumb has correct link for the entity type
// The second breadcrumb item should be "Custom Attributes" with the correct entity type in URL
const customAttributesBreadcrumb = page.locator(
'[data-testid="breadcrumb-link"]:nth-child(2) a'
);
if (customPropertyData.entityApiType) {
// Verify that the Custom Attributes breadcrumb link contains the correct entity type
await expect(customAttributesBreadcrumb).toHaveAttribute(
'href',
`/settings/customProperties/${customPropertyData.entityApiType}`
);
}
// Trigger validation
await page.click('[data-testid="create-button"]');

View File

@ -264,6 +264,12 @@ export const PAGE_HEADERS = {
entity: i18n.t('label.metric-plural'),
}),
},
CHARTS_CUSTOM_ATTRIBUTES: {
header: i18n.t('label.chart-plural'),
subHeader: i18n.t('message.define-custom-property-for-entity', {
entity: i18n.t('label.chart-plural'),
}),
},
PLATFORM_LINEAGE: {
header: i18n.t('label.lineage'),
subHeader: i18n.t('message.page-sub-header-for-platform-lineage'),

View File

@ -12,11 +12,15 @@
*/
import { act, render, screen, waitFor } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import React from 'react';
import { ENTITY_PATH } from '../../constants/constants';
import { PAGE_HEADERS } from '../../constants/PageHeaders.constant';
import { EntityTabs } from '../../enums/entity.enum';
import { Type } from '../../generated/entity/type';
import CustomEntityDetailV1 from './CustomPropertiesPageV1';
const mockNavigate = jest.fn();
const mockTab = jest.fn().mockReturnValue('tables');
jest.mock('react-router-dom', () => ({
useNavigate: jest.fn().mockImplementation(() => mockNavigate),
@ -25,6 +29,10 @@ jest.mock('react-router-dom', () => ({
}),
}));
jest.mock('../../utils/useRequiredParams', () => ({
useRequiredParams: jest.fn(() => ({ tab: mockTab() })),
}));
jest.mock('../../components/common/ErrorWithPlaceholder/ErrorPlaceHolder', () =>
jest.fn().mockImplementation(() => <div>ErrorPlaceHolder</div>)
);
@ -161,4 +169,104 @@ describe('CustomPropertiesPageV1 component', () => {
await waitFor(() => expect(mockShowErrorToast).toHaveBeenCalledTimes(2));
});
describe('customPageHeader mapping', () => {
beforeEach(() => {
jest.clearAllMocks();
jest.spyOn(React, 'useMemo').mockImplementation((fn) => fn());
});
afterEach(() => {
jest.restoreAllMocks();
});
const testCases = [
{ tab: 'tables', expected: PAGE_HEADERS.TABLES_CUSTOM_ATTRIBUTES },
{ tab: 'topics', expected: PAGE_HEADERS.TOPICS_CUSTOM_ATTRIBUTES },
{ tab: 'dashboards', expected: PAGE_HEADERS.DASHBOARD_CUSTOM_ATTRIBUTES },
{
tab: 'dashboardDataModels',
expected: PAGE_HEADERS.DASHBOARD_DATA_MODEL_CUSTOM_ATTRIBUTES,
},
{
tab: 'dataProducts',
expected: PAGE_HEADERS.DATA_PRODUCT_CUSTOM_ATTRIBUTES,
},
{ tab: 'metrics', expected: PAGE_HEADERS.METRIC_CUSTOM_ATTRIBUTES },
{ tab: 'pipelines', expected: PAGE_HEADERS.PIPELINES_CUSTOM_ATTRIBUTES },
{ tab: 'mlmodels', expected: PAGE_HEADERS.ML_MODELS_CUSTOM_ATTRIBUTES },
{ tab: 'containers', expected: PAGE_HEADERS.CONTAINER_CUSTOM_ATTRIBUTES },
{
tab: 'searchIndexes',
expected: PAGE_HEADERS.SEARCH_INDEX_CUSTOM_ATTRIBUTES,
},
{
tab: 'storedProcedures',
expected: PAGE_HEADERS.STORED_PROCEDURE_CUSTOM_ATTRIBUTES,
},
{ tab: 'domains', expected: PAGE_HEADERS.DOMAIN_CUSTOM_ATTRIBUTES },
{
tab: 'glossaryTerm',
expected: PAGE_HEADERS.GLOSSARY_TERM_CUSTOM_ATTRIBUTES,
},
{ tab: 'databases', expected: PAGE_HEADERS.DATABASE_CUSTOM_ATTRIBUTES },
{
tab: 'databaseSchemas',
expected: PAGE_HEADERS.DATABASE_SCHEMA_CUSTOM_ATTRIBUTES,
},
{
tab: 'apiEndpoints',
expected: PAGE_HEADERS.API_ENDPOINT_CUSTOM_ATTRIBUTES,
},
{
tab: 'apiCollections',
expected: PAGE_HEADERS.API_COLLECTION_CUSTOM_ATTRIBUTES,
},
{ tab: 'charts', expected: PAGE_HEADERS.CHARTS_CUSTOM_ATTRIBUTES },
];
it.each(testCases)(
'should return correct header for $tab',
async ({ tab }) => {
mockTab.mockReturnValue(tab);
render(<CustomEntityDetailV1 />);
// Wait for component to render
await waitFor(() => {
expect(mockGetTypeByFQN).toHaveBeenCalled();
});
// Verify the correct header is used based on the tab
// The actual header would be passed to PageHeader component
// Since we're mocking PageHeader, we can't directly test the prop
// but the logic is tested through the tab parameter
expect(mockTab).toHaveBeenCalled();
expect(ENTITY_PATH[tab as keyof typeof ENTITY_PATH]).toBeDefined();
}
);
it('should return TABLES_CUSTOM_ATTRIBUTES as default for unknown tab', async () => {
mockTab.mockReturnValue('unknownTab');
render(<CustomEntityDetailV1 />);
await waitFor(() => {
expect(mockGetTypeByFQN).toHaveBeenCalled();
});
// For unknown tabs, it should default to tables
expect(mockTab).toHaveBeenCalled();
});
it('should have all supported custom property entities covered', () => {
const supportedEntities = Object.keys(ENTITY_PATH).filter(
() => (key: string) => testCases.some((tc) => tc.tab === key)
);
// Verify we have test cases for most entities (excluding some that don't have custom properties)
expect(testCases).toHaveLength(18);
expect(supportedEntities.length).toBeGreaterThanOrEqual(18);
});
});
});

View File

@ -189,6 +189,9 @@ const CustomEntityDetailV1 = () => {
case ENTITY_PATH.apiCollections:
return PAGE_HEADERS.API_COLLECTION_CUSTOM_ATTRIBUTES;
case ENTITY_PATH.charts:
return PAGE_HEADERS.CHARTS_CUSTOM_ATTRIBUTES;
default:
return PAGE_HEADERS.TABLES_CUSTOM_ATTRIBUTES;
}

View File

@ -0,0 +1,224 @@
/*
* Copyright 2022 Collate.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
* http://www.apache.org/licenses/LICENSE-2.0
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { GlobalSettingOptions } from '../constants/GlobalSettings.constants';
import { EntityType } from '../enums/entity.enum';
import { getSettingOptionByEntityType } from './GlobalSettingsUtils';
describe('GlobalSettingsUtils', () => {
describe('getSettingOptionByEntityType', () => {
it('should return TABLES for EntityType.TABLE', () => {
expect(getSettingOptionByEntityType(EntityType.TABLE)).toBe(
GlobalSettingOptions.TABLES
);
});
it('should return TOPICS for EntityType.TOPIC', () => {
expect(getSettingOptionByEntityType(EntityType.TOPIC)).toBe(
GlobalSettingOptions.TOPICS
);
});
it('should return DASHBOARDS for EntityType.DASHBOARD', () => {
expect(getSettingOptionByEntityType(EntityType.DASHBOARD)).toBe(
GlobalSettingOptions.DASHBOARDS
);
});
it('should return PIPELINES for EntityType.PIPELINE', () => {
expect(getSettingOptionByEntityType(EntityType.PIPELINE)).toBe(
GlobalSettingOptions.PIPELINES
);
});
it('should return MLMODELS for EntityType.MLMODEL', () => {
expect(getSettingOptionByEntityType(EntityType.MLMODEL)).toBe(
GlobalSettingOptions.MLMODELS
);
});
it('should return CONTAINERS for EntityType.CONTAINER', () => {
expect(getSettingOptionByEntityType(EntityType.CONTAINER)).toBe(
GlobalSettingOptions.CONTAINERS
);
});
it('should return DATABASE for EntityType.DATABASE', () => {
expect(getSettingOptionByEntityType(EntityType.DATABASE)).toBe(
GlobalSettingOptions.DATABASES
);
});
it('should return DATABASE_SCHEMA for EntityType.DATABASE_SCHEMA', () => {
expect(getSettingOptionByEntityType(EntityType.DATABASE_SCHEMA)).toBe(
GlobalSettingOptions.DATABASE_SCHEMA
);
});
it('should return GLOSSARY_TERM for EntityType.GLOSSARY_TERM', () => {
expect(getSettingOptionByEntityType(EntityType.GLOSSARY_TERM)).toBe(
GlobalSettingOptions.GLOSSARY_TERM
);
});
it('should return CHARTS for EntityType.CHART', () => {
expect(getSettingOptionByEntityType(EntityType.CHART)).toBe(
GlobalSettingOptions.CHARTS
);
});
it('should return DOMAINS for EntityType.DOMAIN', () => {
expect(getSettingOptionByEntityType(EntityType.DOMAIN)).toBe(
GlobalSettingOptions.DOMAINS
);
});
it('should return STORED_PROCEDURES for EntityType.STORED_PROCEDURE', () => {
expect(getSettingOptionByEntityType(EntityType.STORED_PROCEDURE)).toBe(
GlobalSettingOptions.STORED_PROCEDURES
);
});
it('should return SEARCH_INDEXES for EntityType.SEARCH_INDEX', () => {
expect(getSettingOptionByEntityType(EntityType.SEARCH_INDEX)).toBe(
GlobalSettingOptions.SEARCH_INDEXES
);
});
it('should return DASHBOARD_DATA_MODEL for EntityType.DASHBOARD_DATA_MODEL', () => {
expect(
getSettingOptionByEntityType(EntityType.DASHBOARD_DATA_MODEL)
).toBe(GlobalSettingOptions.DASHBOARD_DATA_MODEL);
});
it('should return API_ENDPOINTS for EntityType.API_ENDPOINT', () => {
expect(getSettingOptionByEntityType(EntityType.API_ENDPOINT)).toBe(
GlobalSettingOptions.API_ENDPOINTS
);
});
it('should return API_COLLECTIONS for EntityType.API_COLLECTION', () => {
expect(getSettingOptionByEntityType(EntityType.API_COLLECTION)).toBe(
GlobalSettingOptions.API_COLLECTIONS
);
});
it('should return DATA_PRODUCT for EntityType.DATA_PRODUCT', () => {
expect(getSettingOptionByEntityType(EntityType.DATA_PRODUCT)).toBe(
GlobalSettingOptions.DATA_PRODUCT
);
});
it('should return METRICS for EntityType.METRIC', () => {
expect(getSettingOptionByEntityType(EntityType.METRIC)).toBe(
GlobalSettingOptions.METRICS
);
});
it('should return TABLES as default for unknown entity types', () => {
expect(
getSettingOptionByEntityType('unknownEntityType' as EntityType)
).toBe(GlobalSettingOptions.TABLES);
});
describe('edge cases', () => {
it('should handle undefined gracefully and return TABLES', () => {
expect(
getSettingOptionByEntityType(undefined as unknown as EntityType)
).toBe(GlobalSettingOptions.TABLES);
});
it('should handle null gracefully and return TABLES', () => {
expect(
getSettingOptionByEntityType(null as unknown as EntityType)
).toBe(GlobalSettingOptions.TABLES);
});
it('should handle empty string and return TABLES', () => {
expect(getSettingOptionByEntityType('' as EntityType)).toBe(
GlobalSettingOptions.TABLES
);
});
});
describe('all supported custom property entities', () => {
const supportedEntities = [
{ entity: EntityType.TABLE, option: GlobalSettingOptions.TABLES },
{ entity: EntityType.TOPIC, option: GlobalSettingOptions.TOPICS },
{
entity: EntityType.DASHBOARD,
option: GlobalSettingOptions.DASHBOARDS,
},
{
entity: EntityType.PIPELINE,
option: GlobalSettingOptions.PIPELINES,
},
{ entity: EntityType.MLMODEL, option: GlobalSettingOptions.MLMODELS },
{
entity: EntityType.CONTAINER,
option: GlobalSettingOptions.CONTAINERS,
},
{
entity: EntityType.DATABASE,
option: GlobalSettingOptions.DATABASES,
},
{
entity: EntityType.DATABASE_SCHEMA,
option: GlobalSettingOptions.DATABASE_SCHEMA,
},
{
entity: EntityType.GLOSSARY_TERM,
option: GlobalSettingOptions.GLOSSARY_TERM,
},
{ entity: EntityType.CHART, option: GlobalSettingOptions.CHARTS },
{ entity: EntityType.DOMAIN, option: GlobalSettingOptions.DOMAINS },
{
entity: EntityType.STORED_PROCEDURE,
option: GlobalSettingOptions.STORED_PROCEDURES,
},
{
entity: EntityType.SEARCH_INDEX,
option: GlobalSettingOptions.SEARCH_INDEXES,
},
{
entity: EntityType.DASHBOARD_DATA_MODEL,
option: GlobalSettingOptions.DASHBOARD_DATA_MODEL,
},
{
entity: EntityType.API_ENDPOINT,
option: GlobalSettingOptions.API_ENDPOINTS,
},
{
entity: EntityType.API_COLLECTION,
option: GlobalSettingOptions.API_COLLECTIONS,
},
{
entity: EntityType.DATA_PRODUCT,
option: GlobalSettingOptions.DATA_PRODUCT,
},
{ entity: EntityType.METRIC, option: GlobalSettingOptions.METRICS },
];
it.each(supportedEntities)(
'should map $entity to $option correctly',
({ entity, option }) => {
expect(getSettingOptionByEntityType(entity)).toBe(option);
}
);
it('should have all entities that support custom properties covered', () => {
expect(supportedEntities).toHaveLength(18);
});
});
});
});

View File

@ -55,13 +55,29 @@ export const getSettingOptionByEntityType = (entityType: EntityType) => {
case EntityType.CONTAINER:
return GlobalSettingOptions.CONTAINERS;
case EntityType.DATABASE:
return GlobalSettingOptions.DATABASE;
return GlobalSettingOptions.DATABASES;
case EntityType.DATABASE_SCHEMA:
return GlobalSettingOptions.DATABASE_SCHEMA;
case EntityType.GLOSSARY_TERM:
return GlobalSettingOptions.GLOSSARY_TERM;
case EntityType.CHART:
return GlobalSettingOptions.CHARTS;
case EntityType.DOMAIN:
return GlobalSettingOptions.DOMAINS;
case EntityType.STORED_PROCEDURE:
return GlobalSettingOptions.STORED_PROCEDURES;
case EntityType.SEARCH_INDEX:
return GlobalSettingOptions.SEARCH_INDEXES;
case EntityType.DASHBOARD_DATA_MODEL:
return GlobalSettingOptions.DASHBOARD_DATA_MODEL;
case EntityType.API_ENDPOINT:
return GlobalSettingOptions.API_ENDPOINTS;
case EntityType.API_COLLECTION:
return GlobalSettingOptions.API_COLLECTIONS;
case EntityType.DATA_PRODUCT:
return GlobalSettingOptions.DATA_PRODUCT;
case EntityType.METRIC:
return GlobalSettingOptions.METRICS;
case EntityType.TABLE:
default:

View File

@ -58,9 +58,16 @@ Object.defineProperty(global, 'navigator', {
});
// Mock MessageChannel
interface MockMessageEvent {
data: {
result?: unknown;
error?: string;
};
}
const mockMessageChannel = {
port1: {
onmessage: null as ((event: any) => void) | null,
onmessage: null as ((event: MockMessageEvent) => void) | null,
},
port2: {},
};
@ -245,7 +252,7 @@ describe('SwMessenger', () => {
});
it('should resolve when service worker is ready', async () => {
let messageHandler: (event: any) => void;
let messageHandler: (event: MockMessageEvent) => void;
Object.defineProperty(mockMessageChannel.port1, 'onmessage', {
set: (handler) => {
@ -303,7 +310,10 @@ describe('SwMessenger', () => {
});
it('should send message and return response', async () => {
const testMessage = { type: 'get', key: 'test-key' };
const testMessage: { type: 'get'; key: string } = {
type: 'get',
key: 'test-key',
};
const expectedResponse = 'test-value';
Object.defineProperty(mockMessageChannel.port1, 'onmessage', {
@ -317,7 +327,7 @@ describe('SwMessenger', () => {
configurable: true,
});
const result = await sendMessageToServiceWorker(testMessage as any);
const result = await sendMessageToServiceWorker(testMessage);
expect(mockController.postMessage).toHaveBeenCalledWith(
expect.objectContaining({
@ -330,7 +340,10 @@ describe('SwMessenger', () => {
});
it('should handle service worker errors', async () => {
const testMessage = { type: 'get', key: 'test-key' };
const testMessage: { type: 'get'; key: string } = {
type: 'get',
key: 'test-key',
};
const errorMessage = 'Service worker error';
Object.defineProperty(mockMessageChannel.port1, 'onmessage', {
@ -344,13 +357,13 @@ describe('SwMessenger', () => {
configurable: true,
});
await expect(
sendMessageToServiceWorker(testMessage as any)
).rejects.toThrow(errorMessage);
await expect(sendMessageToServiceWorker(testMessage)).rejects.toThrow(
errorMessage
);
});
it('should increment request counter for unique request IDs', async () => {
const testMessage = { type: 'ping' };
const testMessage: { type: 'ping' } = { type: 'ping' };
Object.defineProperty(mockMessageChannel.port1, 'onmessage', {
set: (handler) => {
@ -362,8 +375,8 @@ describe('SwMessenger', () => {
});
// Send two messages
await sendMessageToServiceWorker(testMessage as any);
await sendMessageToServiceWorker(testMessage as any);
await sendMessageToServiceWorker(testMessage);
await sendMessageToServiceWorker(testMessage);
// Check that different request IDs were used
const calls = mockController.postMessage.mock.calls;
@ -402,7 +415,9 @@ describe('SwMessenger', () => {
configurable: true,
});
const result = await sendMessageToServiceWorker({ type: 'ping' } as any);
const result = await sendMessageToServiceWorker({
type: 'ping',
} as const);
expect(result).toBe('success');
});

View File

@ -17,8 +17,8 @@ import { swTokenStorage } from './SwTokenStorage';
const mockSendMessageToServiceWorker = jest.fn();
jest.mock('./SwMessenger', () => ({
sendMessageToServiceWorker: (...args: any[]) =>
mockSendMessageToServiceWorker(...args),
sendMessageToServiceWorker: (message: unknown) =>
mockSendMessageToServiceWorker(message),
}));
describe('SwTokenStorage', () => {

View File

@ -25,13 +25,21 @@ const mockGetItem = jest.fn();
jest.mock('./SwTokenStorage', () => ({
swTokenStorage: {
setItem: (...args: any[]) => mockSetItem(...args),
getItem: (...args: any[]) => mockGetItem(...args),
setItem: (key: string, value: string) => mockSetItem(key, value),
getItem: (key: string) => mockGetItem(key),
},
}));
// Mock navigator and localStorage for browser environment simulation
const mockNavigator = {
interface MockNavigator {
serviceWorker?: Record<string, unknown>;
}
interface MockWindow {
indexedDB?: Record<string, unknown>;
}
const mockNavigator: MockNavigator = {
serviceWorker: {},
};
@ -53,7 +61,7 @@ Object.defineProperty(global, 'localStorage', {
Object.defineProperty(global, 'window', {
value: {
indexedDB: {},
},
} as MockWindow,
writable: true,
});
@ -65,7 +73,7 @@ describe('SwTokenStorageUtils', () => {
describe('isServiceWorkerAvailable', () => {
it('should return true when both serviceWorker and indexedDB are available', () => {
mockNavigator.serviceWorker = {};
(global as any).window.indexedDB = {};
(global.window as unknown as MockWindow).indexedDB = {};
const result = isServiceWorkerAvailable();
@ -73,8 +81,8 @@ describe('SwTokenStorageUtils', () => {
});
it('should return false when serviceWorker is not available', () => {
delete (mockNavigator as any).serviceWorker;
(global as any).window.indexedDB = {};
delete mockNavigator.serviceWorker;
(global.window as unknown as MockWindow).indexedDB = {};
const result = isServiceWorkerAvailable();
@ -83,7 +91,7 @@ describe('SwTokenStorageUtils', () => {
it('should return false when indexedDB is not available', () => {
mockNavigator.serviceWorker = {};
delete (global as any).window.indexedDB;
delete (global.window as unknown as MockWindow).indexedDB;
const result = isServiceWorkerAvailable();
@ -91,8 +99,8 @@ describe('SwTokenStorageUtils', () => {
});
it('should return false when neither serviceWorker nor indexedDB are available', () => {
delete (mockNavigator as any).serviceWorker;
delete (global as any).window.indexedDB;
delete mockNavigator.serviceWorker;
delete (global.window as unknown as MockWindow).indexedDB;
const result = isServiceWorkerAvailable();
@ -104,7 +112,7 @@ describe('SwTokenStorageUtils', () => {
beforeEach(() => {
// Reset environment for each test
mockNavigator.serviceWorker = {};
(global as any).window.indexedDB = {};
(global.window as unknown as MockWindow).indexedDB = {};
});
it('should return token from service worker when available', async () => {
@ -136,7 +144,7 @@ describe('SwTokenStorageUtils', () => {
});
it('should fallback to localStorage when service worker is not available', async () => {
delete (mockNavigator as any).serviceWorker;
delete mockNavigator.serviceWorker;
const mockToken = 'test-oidc-token';
const mockAppState = JSON.stringify({ primary: mockToken });
mockLocalStorage.getItem.mockReturnValue(mockAppState);
@ -167,7 +175,7 @@ describe('SwTokenStorageUtils', () => {
describe('setOidcToken', () => {
beforeEach(() => {
mockNavigator.serviceWorker = {};
(global as any).window.indexedDB = {};
(global.window as unknown as MockWindow).indexedDB = {};
});
it('should set token in service worker when available', async () => {
@ -200,7 +208,7 @@ describe('SwTokenStorageUtils', () => {
});
it('should fallback to localStorage when service worker is not available', async () => {
delete (mockNavigator as any).serviceWorker;
delete mockNavigator.serviceWorker;
const mockToken = 'new-oidc-token';
const expectedState = JSON.stringify({ primary: mockToken });
@ -224,7 +232,7 @@ describe('SwTokenStorageUtils', () => {
describe('getRefreshToken', () => {
beforeEach(() => {
mockNavigator.serviceWorker = {};
(global as any).window.indexedDB = {};
(global.window as unknown as MockWindow).indexedDB = {};
});
it('should return refresh token from service worker when available', async () => {
@ -248,7 +256,7 @@ describe('SwTokenStorageUtils', () => {
});
it('should fallback to localStorage when service worker is not available', async () => {
delete (mockNavigator as any).serviceWorker;
delete mockNavigator.serviceWorker;
const mockToken = 'test-refresh-token';
const mockAppState = JSON.stringify({ secondary: mockToken });
mockLocalStorage.getItem.mockReturnValue(mockAppState);
@ -271,7 +279,7 @@ describe('SwTokenStorageUtils', () => {
describe('setRefreshToken', () => {
beforeEach(() => {
mockNavigator.serviceWorker = {};
(global as any).window.indexedDB = {};
(global.window as unknown as MockWindow).indexedDB = {};
});
it('should set refresh token in service worker when available', async () => {
@ -304,7 +312,7 @@ describe('SwTokenStorageUtils', () => {
});
it('should fallback to localStorage when service worker is not available', async () => {
delete (mockNavigator as any).serviceWorker;
delete mockNavigator.serviceWorker;
const mockToken = 'new-refresh-token';
const expectedState = JSON.stringify({ secondary: mockToken });
@ -328,7 +336,7 @@ describe('SwTokenStorageUtils', () => {
describe('integration scenarios', () => {
beforeEach(() => {
mockNavigator.serviceWorker = {};
(global as any).window.indexedDB = {};
(global.window as unknown as MockWindow).indexedDB = {};
});
it('should maintain both tokens when updating one', async () => {
@ -363,7 +371,7 @@ describe('SwTokenStorageUtils', () => {
it('should handle mixed environment gracefully (some features available)', async () => {
// Simulate environment where serviceWorker exists but indexedDB doesn't
mockNavigator.serviceWorker = {};
delete (global as any).window.indexedDB;
delete (global.window as unknown as MockWindow).indexedDB;
const result = await getOidcToken();