# Conflicts: # docs/technical/README.md # docs/technical/RUST_API_SERVER_ROUTE_INDEX_2026-04-22.md # docs/technical/SPACETIMEDB_TABLE_CATALOG.md # scripts/generate-spacetime-bindings.mjs # server-rs/crates/api-server/src/app.rs # server-rs/crates/api-server/src/assets.rs # server-rs/crates/api-server/src/big_fish.rs # server-rs/crates/api-server/src/custom_world_ai.rs # server-rs/crates/api-server/src/llm.rs # server-rs/crates/api-server/src/main.rs # server-rs/crates/api-server/src/puzzle.rs # server-rs/crates/api-server/src/runtime_profile.rs # server-rs/crates/api-server/src/runtime_story/compat/ai.rs # server-rs/crates/api-server/src/runtime_story/compat/npc_actions.rs # server-rs/crates/api-server/src/runtime_story/compat/presentation.rs # server-rs/crates/api-server/src/runtime_story/compat/tests.rs # server-rs/crates/api-server/src/state.rs # server-rs/crates/module-auth/src/lib.rs # server-rs/crates/module-big-fish/src/lib.rs # server-rs/crates/module-custom-world/src/lib.rs # server-rs/crates/module-puzzle/src/lib.rs # server-rs/crates/module-runtime/src/lib.rs # server-rs/crates/spacetime-client/src/big_fish.rs # server-rs/crates/spacetime-client/src/lib.rs # server-rs/crates/spacetime-client/src/mapper.rs # server-rs/crates/spacetime-client/src/module_bindings/admin_disable_profile_redeem_code_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/admin_upsert_profile_redeem_code_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/advance_puzzle_next_level_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/append_ai_text_chunk_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/apply_chapter_progression_ledger_entry_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/attach_ai_result_reference_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/authorize_database_migration_operator_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/begin_story_session_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/big_fish_runtime_run_type.rs # server-rs/crates/spacetime-client/src/module_bindings/bind_asset_object_to_entity_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/cancel_ai_task_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/clear_platform_browse_history_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/compile_big_fish_draft_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/compile_custom_world_published_profile_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/compile_puzzle_agent_draft_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/complete_ai_stage_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/complete_ai_task_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/confirm_asset_object_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/consume_profile_wallet_points_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/continue_story_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/create_ai_task_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/create_battle_state_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/create_big_fish_session_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/create_custom_world_agent_session_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/create_profile_recharge_order_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/create_puzzle_agent_session_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/delete_big_fish_work_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/delete_custom_world_agent_session_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/delete_custom_world_profile_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/delete_puzzle_work_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/delete_runtime_snapshot_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/drag_puzzle_piece_or_group_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/execute_custom_world_agent_action_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/export_auth_store_snapshot_from_tables_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/export_database_migration_to_file_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/fail_ai_task_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/finalize_big_fish_agent_message_turn_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/finalize_custom_world_agent_message_turn_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/finalize_puzzle_agent_message_turn_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/generate_big_fish_asset_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_auth_store_snapshot_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_battle_state_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_big_fish_session_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_chapter_progression_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_custom_world_agent_card_detail_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_custom_world_agent_operation_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_custom_world_agent_session_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_custom_world_gallery_detail_by_code_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_custom_world_gallery_detail_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_custom_world_library_detail_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_player_progression_or_default_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_profile_dashboard_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_profile_play_stats_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_profile_recharge_center_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_profile_referral_invite_center_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_puzzle_agent_session_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_puzzle_gallery_detail_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_puzzle_run_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_puzzle_work_detail_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_runtime_inventory_state_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_runtime_setting_or_default_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_runtime_snapshot_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/get_story_session_state_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/grant_player_progression_experience_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/import_auth_store_snapshot_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/import_database_migration_from_file_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/import_database_migration_incremental_from_file_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/list_asset_history_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/list_big_fish_works_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/list_custom_world_gallery_entries_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/list_custom_world_profiles_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/list_custom_world_works_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/list_platform_browse_history_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/list_profile_save_archives_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/list_profile_wallet_ledger_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/list_puzzle_gallery_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/list_puzzle_works_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/mod.rs # server-rs/crates/spacetime-client/src/module_bindings/publish_big_fish_game_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/publish_custom_world_profile_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/publish_custom_world_world_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/publish_puzzle_work_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/record_big_fish_play_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/redeem_profile_referral_invite_code_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/redeem_profile_reward_code_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/refund_profile_wallet_points_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/resolve_combat_action_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/resolve_npc_battle_interaction_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/resolve_npc_interaction_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/resolve_npc_social_action_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/resolve_treasure_interaction_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/resume_profile_save_archive_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/revoke_database_migration_operator_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/save_puzzle_generated_images_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/select_puzzle_cover_image_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/start_puzzle_run_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/submit_big_fish_message_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/submit_custom_world_agent_message_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/submit_puzzle_agent_message_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/submit_puzzle_leaderboard_entry_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/swap_puzzle_pieces_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/unpublish_custom_world_profile_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/update_puzzle_work_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/upsert_auth_store_snapshot_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/upsert_chapter_progression_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/upsert_custom_world_agent_operation_progress_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/upsert_custom_world_profile_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/upsert_npc_state_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/upsert_platform_browse_history_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/upsert_runtime_setting_and_return_procedure.rs # server-rs/crates/spacetime-client/src/module_bindings/upsert_runtime_snapshot_and_return_procedure.rs # server-rs/crates/spacetime-module/src/auth/procedures.rs # server-rs/crates/spacetime-module/src/custom_world/mod.rs # server-rs/crates/spacetime-module/src/lib.rs # server-rs/crates/spacetime-module/src/migration.rs # server-rs/crates/spacetime-module/src/puzzle.rs # server-rs/crates/spacetime-module/src/runtime/profile.rs # src/components/platform-entry/PlatformEntryFlowShellImpl.tsx # src/components/rpg-entry/RpgEntryFlowShell.agent.interaction.test.tsx # src/services/aiService.ts # src/services/puzzle-runtime/puzzleRuntimeClient.ts
499 lines
13 KiB
TypeScript
499 lines
13 KiB
TypeScript
import { beforeEach, describe, expect, it, vi } from 'vitest';
|
|
|
|
import {
|
|
AUTH_STATE_EVENT,
|
|
ApiClientError,
|
|
clearStoredAccessToken,
|
|
fetchWithApiAuth,
|
|
getStoredAccessToken,
|
|
isTimeoutError,
|
|
requestJson,
|
|
setStoredAccessToken,
|
|
} from './apiClient';
|
|
|
|
function createLocalStorageMock() {
|
|
const store = new Map<string, string>();
|
|
|
|
return {
|
|
getItem(key: string) {
|
|
return store.has(key) ? store.get(key)! : null;
|
|
},
|
|
setItem(key: string, value: string) {
|
|
store.set(key, String(value));
|
|
},
|
|
removeItem(key: string) {
|
|
store.delete(key);
|
|
},
|
|
clear() {
|
|
store.clear();
|
|
},
|
|
};
|
|
}
|
|
|
|
function createResponseMock(params: {
|
|
status: number;
|
|
body?: string;
|
|
headers?: Record<string, string>;
|
|
}) {
|
|
const headers = new Map(
|
|
Object.entries(params.headers ?? {}).map(([key, value]) => [
|
|
key.toLowerCase(),
|
|
value,
|
|
]),
|
|
);
|
|
|
|
return {
|
|
status: params.status,
|
|
ok: params.status >= 200 && params.status < 300,
|
|
headers: {
|
|
get(name: string) {
|
|
return headers.get(name.toLowerCase()) ?? null;
|
|
},
|
|
},
|
|
text: vi.fn(async () => params.body ?? ''),
|
|
};
|
|
}
|
|
|
|
describe('apiClient', () => {
|
|
const fetchMock = vi.fn();
|
|
const dispatchEventMock = vi.fn();
|
|
|
|
beforeEach(() => {
|
|
vi.stubGlobal('fetch', fetchMock);
|
|
vi.stubGlobal('window', {
|
|
dispatchEvent: dispatchEventMock,
|
|
localStorage: createLocalStorageMock(),
|
|
});
|
|
fetchMock.mockReset();
|
|
dispatchEventMock.mockReset();
|
|
clearStoredAccessToken({ emit: false });
|
|
});
|
|
|
|
it('refreshes bearer token once and retries the original request', async () => {
|
|
setStoredAccessToken('expired-token', { emit: false });
|
|
fetchMock
|
|
.mockResolvedValueOnce(createResponseMock({ status: 401 }))
|
|
.mockResolvedValueOnce(
|
|
createResponseMock({
|
|
status: 200,
|
|
body: JSON.stringify({
|
|
ok: true,
|
|
data: {
|
|
ok: true,
|
|
token: 'fresh-token',
|
|
},
|
|
error: null,
|
|
meta: {
|
|
apiVersion: '2026-04-08',
|
|
},
|
|
}),
|
|
}),
|
|
)
|
|
.mockResolvedValueOnce(
|
|
createResponseMock({
|
|
status: 200,
|
|
body: JSON.stringify({
|
|
ok: true,
|
|
data: {
|
|
value: 7,
|
|
},
|
|
error: null,
|
|
meta: {
|
|
apiVersion: '2026-04-08',
|
|
},
|
|
}),
|
|
}),
|
|
);
|
|
|
|
const result = await requestJson<{ value: number }>(
|
|
'/api/runtime/protected',
|
|
{ method: 'GET' },
|
|
'读取受保护数据失败',
|
|
);
|
|
|
|
expect(result).toEqual({ value: 7 });
|
|
expect(fetchMock).toHaveBeenCalledTimes(3);
|
|
expect(fetchMock).toHaveBeenNthCalledWith(
|
|
1,
|
|
'/api/runtime/protected',
|
|
expect.objectContaining({
|
|
credentials: 'same-origin',
|
|
headers: expect.objectContaining({
|
|
Authorization: 'Bearer expired-token',
|
|
'x-genarrative-response-envelope': 'v1',
|
|
}),
|
|
}),
|
|
);
|
|
expect(fetchMock).toHaveBeenNthCalledWith(
|
|
2,
|
|
'/api/auth/refresh',
|
|
expect.objectContaining({
|
|
method: 'POST',
|
|
credentials: 'same-origin',
|
|
}),
|
|
);
|
|
expect(fetchMock).toHaveBeenNthCalledWith(
|
|
3,
|
|
'/api/runtime/protected',
|
|
expect.objectContaining({
|
|
credentials: 'same-origin',
|
|
headers: expect.objectContaining({
|
|
Authorization: 'Bearer fresh-token',
|
|
}),
|
|
}),
|
|
);
|
|
expect(dispatchEventMock).not.toHaveBeenCalled();
|
|
expect(getStoredAccessToken()).toBe('fresh-token');
|
|
});
|
|
|
|
it('hydrates a missing local bearer token before the first protected request', async () => {
|
|
fetchMock
|
|
.mockResolvedValueOnce(
|
|
createResponseMock({
|
|
status: 200,
|
|
body: JSON.stringify({
|
|
ok: true,
|
|
data: {
|
|
ok: true,
|
|
token: 'fresh-token',
|
|
},
|
|
error: null,
|
|
meta: {
|
|
apiVersion: '2026-04-08',
|
|
},
|
|
}),
|
|
}),
|
|
)
|
|
.mockResolvedValueOnce(
|
|
createResponseMock({
|
|
status: 200,
|
|
body: JSON.stringify({
|
|
ok: true,
|
|
data: {
|
|
value: 9,
|
|
},
|
|
error: null,
|
|
meta: {
|
|
apiVersion: '2026-04-08',
|
|
},
|
|
}),
|
|
}),
|
|
);
|
|
|
|
const result = await requestJson<{ value: number }>(
|
|
'/api/runtime/protected',
|
|
{ method: 'GET' },
|
|
'读取受保护数据失败',
|
|
);
|
|
|
|
expect(result).toEqual({ value: 9 });
|
|
expect(fetchMock).toHaveBeenCalledTimes(2);
|
|
expect(fetchMock).toHaveBeenNthCalledWith(
|
|
1,
|
|
'/api/auth/refresh',
|
|
expect.objectContaining({
|
|
method: 'POST',
|
|
credentials: 'same-origin',
|
|
}),
|
|
);
|
|
expect(fetchMock).toHaveBeenNthCalledWith(
|
|
2,
|
|
'/api/runtime/protected',
|
|
expect.objectContaining({
|
|
headers: expect.objectContaining({
|
|
Authorization: 'Bearer fresh-token',
|
|
}),
|
|
}),
|
|
);
|
|
expect(getStoredAccessToken()).toBe('fresh-token');
|
|
});
|
|
|
|
it('does not emit auth change events when 401 probe requests opt into silent mode', async () => {
|
|
fetchMock.mockResolvedValueOnce(createResponseMock({ status: 401 }));
|
|
|
|
const response = await fetchWithApiAuth(
|
|
'/api/auth/me',
|
|
{
|
|
method: 'GET',
|
|
},
|
|
{
|
|
notifyAuthStateChange: false,
|
|
skipRefresh: true,
|
|
},
|
|
);
|
|
|
|
expect(response.status).toBe(401);
|
|
expect(fetchMock).toHaveBeenCalledTimes(1);
|
|
expect(dispatchEventMock).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('emits auth change events when refresh fails on protected requests', async () => {
|
|
setStoredAccessToken('expired-token', { emit: false });
|
|
fetchMock
|
|
.mockResolvedValueOnce(createResponseMock({ status: 401 }))
|
|
.mockResolvedValueOnce(createResponseMock({ status: 401 }));
|
|
|
|
const response = await fetchWithApiAuth('/api/runtime/protected', {
|
|
method: 'GET',
|
|
});
|
|
|
|
expect(response.status).toBe(401);
|
|
expect(fetchMock).toHaveBeenCalledTimes(2);
|
|
expect(dispatchEventMock).toHaveBeenCalledTimes(1);
|
|
expect(getStoredAccessToken()).toBe('');
|
|
});
|
|
|
|
it('keeps the refreshed token when the retried protected request is still unauthorized', async () => {
|
|
setStoredAccessToken('expired-token', { emit: false });
|
|
fetchMock
|
|
.mockResolvedValueOnce(createResponseMock({ status: 401 }))
|
|
.mockResolvedValueOnce(
|
|
createResponseMock({
|
|
status: 200,
|
|
body: JSON.stringify({
|
|
ok: true,
|
|
data: {
|
|
ok: true,
|
|
token: 'fresh-token',
|
|
},
|
|
error: null,
|
|
meta: {
|
|
apiVersion: '2026-04-08',
|
|
},
|
|
}),
|
|
}),
|
|
)
|
|
.mockResolvedValueOnce(createResponseMock({ status: 401 }));
|
|
|
|
const response = await fetchWithApiAuth('/api/runtime/puzzle/works', {
|
|
method: 'GET',
|
|
});
|
|
|
|
expect(response.status).toBe(401);
|
|
expect(fetchMock).toHaveBeenCalledTimes(3);
|
|
expect(getStoredAccessToken()).toBe('fresh-token');
|
|
expect(dispatchEventMock).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('rejects refresh responses that do not return a renewed bearer token', async () => {
|
|
setStoredAccessToken('expired-token', { emit: false });
|
|
fetchMock
|
|
.mockResolvedValueOnce(createResponseMock({ status: 401 }))
|
|
.mockResolvedValueOnce(
|
|
createResponseMock({
|
|
status: 200,
|
|
body: JSON.stringify({
|
|
ok: true,
|
|
data: {
|
|
ok: true,
|
|
},
|
|
error: null,
|
|
meta: {
|
|
apiVersion: '2026-04-08',
|
|
},
|
|
}),
|
|
}),
|
|
);
|
|
|
|
await expect(
|
|
requestJson<{ value: number }>(
|
|
'/api/runtime/protected',
|
|
{ method: 'GET' },
|
|
'读取受保护数据失败',
|
|
),
|
|
).rejects.toMatchObject({
|
|
status: 401,
|
|
message: '读取受保护数据失败',
|
|
});
|
|
expect(fetchMock).toHaveBeenCalledTimes(2);
|
|
expect(getStoredAccessToken()).toBe('');
|
|
expect(dispatchEventMock).toHaveBeenCalledTimes(1);
|
|
});
|
|
|
|
it('keeps the current access token when a public request explicitly skips auth', async () => {
|
|
setStoredAccessToken('still-valid-token');
|
|
vi.mocked(window.dispatchEvent).mockClear();
|
|
fetchMock.mockResolvedValueOnce(createResponseMock({ status: 401 }));
|
|
|
|
const response = await fetchWithApiAuth(
|
|
'/api/runtime/custom-world-gallery',
|
|
{ method: 'GET' },
|
|
{
|
|
skipAuth: true,
|
|
skipRefresh: true,
|
|
},
|
|
);
|
|
|
|
expect(response.status).toBe(401);
|
|
expect(fetchMock).toHaveBeenCalledTimes(1);
|
|
expect(fetchMock).toHaveBeenCalledWith(
|
|
'/api/runtime/custom-world-gallery',
|
|
expect.objectContaining({
|
|
headers: expect.not.objectContaining({
|
|
Authorization: expect.any(String),
|
|
}),
|
|
}),
|
|
);
|
|
expect(getStoredAccessToken()).toBe('still-valid-token');
|
|
expect(window.dispatchEvent).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('retries transient get requests before unwrapping the response envelope', async () => {
|
|
fetchMock
|
|
.mockRejectedValueOnce(new TypeError('network unavailable'))
|
|
.mockResolvedValueOnce(
|
|
createResponseMock({
|
|
status: 200,
|
|
body: JSON.stringify({
|
|
ok: true,
|
|
data: {
|
|
value: 42,
|
|
},
|
|
error: null,
|
|
meta: {
|
|
apiVersion: '2026-04-08',
|
|
},
|
|
}),
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
}),
|
|
);
|
|
|
|
const result = await requestJson<{ value: number }>(
|
|
'/api/runtime/settings',
|
|
{ method: 'GET' },
|
|
'读取设置失败',
|
|
);
|
|
|
|
expect(fetchMock).toHaveBeenCalledTimes(2);
|
|
expect(result).toEqual({ value: 42 });
|
|
});
|
|
|
|
it('aborts requests when timeoutMs is reached', async () => {
|
|
setStoredAccessToken('timeout-token', { emit: false });
|
|
fetchMock.mockImplementation(
|
|
async (_input: string, init?: RequestInit) =>
|
|
new Promise((_resolve, reject) => {
|
|
init?.signal?.addEventListener(
|
|
'abort',
|
|
() => {
|
|
reject(init.signal?.reason);
|
|
},
|
|
{ once: true },
|
|
);
|
|
}),
|
|
);
|
|
|
|
let capturedError: unknown;
|
|
try {
|
|
await requestJson(
|
|
'/api/runtime/protected',
|
|
{ method: 'POST' },
|
|
'创建会话失败',
|
|
{
|
|
timeoutMs: 20,
|
|
skipRefresh: true,
|
|
},
|
|
);
|
|
} catch (error) {
|
|
capturedError = error;
|
|
}
|
|
|
|
expect(fetchMock).toHaveBeenCalledTimes(1);
|
|
expect(isTimeoutError(capturedError)).toBe(true);
|
|
expect(capturedError).toBeInstanceOf(Error);
|
|
});
|
|
|
|
it('surfaces response metadata through ApiClientError', async () => {
|
|
setStoredAccessToken('metadata-token', { emit: false });
|
|
fetchMock.mockResolvedValueOnce(
|
|
createResponseMock({
|
|
status: 503,
|
|
body: JSON.stringify({
|
|
ok: false,
|
|
data: null,
|
|
error: {
|
|
code: 'UPSTREAM_ERROR',
|
|
message: '上游暂不可用',
|
|
details: {
|
|
scope: 'runtime',
|
|
},
|
|
},
|
|
meta: {
|
|
apiVersion: '2026-04-08',
|
|
requestId: 'req-body',
|
|
},
|
|
}),
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'x-request-id': 'req-header',
|
|
'x-route-version': 'runtime.v2',
|
|
},
|
|
}),
|
|
);
|
|
|
|
let capturedError: unknown;
|
|
try {
|
|
await requestJson(
|
|
'/api/runtime/protected-error',
|
|
{ method: 'POST' },
|
|
'请求失败',
|
|
);
|
|
} catch (error) {
|
|
capturedError = error;
|
|
}
|
|
|
|
expect(capturedError).toBeInstanceOf(ApiClientError);
|
|
expect(capturedError).toMatchObject({
|
|
status: 503,
|
|
code: 'UPSTREAM_ERROR',
|
|
details: {
|
|
scope: 'runtime',
|
|
},
|
|
meta: {
|
|
requestId: 'req-body',
|
|
routeVersion: 'runtime.v2',
|
|
},
|
|
});
|
|
});
|
|
|
|
it('uses api error details.message as ApiClientError message', async () => {
|
|
setStoredAccessToken('details-message-token', { emit: false });
|
|
fetchMock.mockResolvedValueOnce(
|
|
createResponseMock({
|
|
status: 400,
|
|
body: JSON.stringify({
|
|
ok: false,
|
|
data: null,
|
|
error: {
|
|
code: 'BAD_REQUEST',
|
|
message: '请求参数不合法',
|
|
details: {
|
|
provider: 'dashscope',
|
|
message: '拼图图片生成失败:请求参数不合法',
|
|
},
|
|
},
|
|
meta: {},
|
|
}),
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
}),
|
|
);
|
|
|
|
await expect(
|
|
requestJson('/api/runtime/puzzle/agent/sessions/test/actions', {
|
|
method: 'POST',
|
|
}, '执行拼图操作失败。'),
|
|
).rejects.toMatchObject({
|
|
message: '拼图图片生成失败:请求参数不合法',
|
|
status: 400,
|
|
code: 'BAD_REQUEST',
|
|
details: {
|
|
provider: 'dashscope',
|
|
},
|
|
});
|
|
});
|
|
});
|