Make coverage filing sync explicit
This commit is contained in:
@@ -35,6 +35,14 @@ type FormState = {
|
|||||||
tags: string;
|
tags: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
type PostCreateNotice = {
|
||||||
|
ticker: string;
|
||||||
|
category: string | null;
|
||||||
|
tags: string[];
|
||||||
|
syncState: 'idle' | 'pending' | 'queued';
|
||||||
|
error: string | null;
|
||||||
|
};
|
||||||
|
|
||||||
const STATUS_OPTIONS: Array<{ value: CoverageStatus; label: string }> = [
|
const STATUS_OPTIONS: Array<{ value: CoverageStatus; label: string }> = [
|
||||||
{ value: 'backlog', label: 'Backlog' },
|
{ value: 'backlog', label: 'Backlog' },
|
||||||
{ value: 'active', label: 'Active' },
|
{ value: 'active', label: 'Active' },
|
||||||
@@ -117,6 +125,7 @@ export default function WatchlistPage() {
|
|||||||
const [error, setError] = useState<string | null>(null);
|
const [error, setError] = useState<string | null>(null);
|
||||||
const [editingItemId, setEditingItemId] = useState<number | null>(null);
|
const [editingItemId, setEditingItemId] = useState<number | null>(null);
|
||||||
const [form, setForm] = useState<FormState>(EMPTY_FORM);
|
const [form, setForm] = useState<FormState>(EMPTY_FORM);
|
||||||
|
const [postCreateNotice, setPostCreateNotice] = useState<PostCreateNotice | null>(null);
|
||||||
|
|
||||||
const loadCoverage = useCallback(async () => {
|
const loadCoverage = useCallback(async () => {
|
||||||
const options = watchlistQueryOptions();
|
const options = watchlistQueryOptions();
|
||||||
@@ -170,6 +179,7 @@ export default function WatchlistPage() {
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const beginEdit = useCallback((item: WatchlistItem) => {
|
const beginEdit = useCallback((item: WatchlistItem) => {
|
||||||
|
setPostCreateNotice(null);
|
||||||
setEditingItemId(item.id);
|
setEditingItemId(item.id);
|
||||||
setForm({
|
setForm({
|
||||||
ticker: item.ticker,
|
ticker: item.ticker,
|
||||||
@@ -191,12 +201,34 @@ export default function WatchlistPage() {
|
|||||||
}
|
}
|
||||||
}, [queryClient]);
|
}, [queryClient]);
|
||||||
|
|
||||||
|
const queueCoverageSync = useCallback(async (input: {
|
||||||
|
ticker: string;
|
||||||
|
category?: string | null;
|
||||||
|
tags?: string[];
|
||||||
|
}) => {
|
||||||
|
const ticker = input.ticker.trim().toUpperCase();
|
||||||
|
|
||||||
|
await queueFilingSync({
|
||||||
|
ticker,
|
||||||
|
limit: 20,
|
||||||
|
category: input.category ?? undefined,
|
||||||
|
tags: input.tags && input.tags.length > 0 ? input.tags : undefined
|
||||||
|
});
|
||||||
|
|
||||||
|
void queryClient.invalidateQueries({ queryKey: queryKeys.recentTasks(20) });
|
||||||
|
void queryClient.invalidateQueries({ queryKey: ['filings', ticker] });
|
||||||
|
void queryClient.invalidateQueries({ queryKey: queryKeys.companyAnalysis(ticker) });
|
||||||
|
}, [queryClient]);
|
||||||
|
|
||||||
const saveCoverage = async (event: React.FormEvent<HTMLFormElement>) => {
|
const saveCoverage = async (event: React.FormEvent<HTMLFormElement>) => {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
setSaving(true);
|
setSaving(true);
|
||||||
setError(null);
|
setError(null);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
const isCreate = editingItemId === null;
|
||||||
|
const ticker = form.ticker.trim().toUpperCase();
|
||||||
|
const wasExistingItem = items.some((item) => item.ticker === ticker);
|
||||||
const payload = {
|
const payload = {
|
||||||
companyName: form.companyName.trim(),
|
companyName: form.companyName.trim(),
|
||||||
sector: form.sector.trim() || undefined,
|
sector: form.sector.trim() || undefined,
|
||||||
@@ -206,16 +238,29 @@ export default function WatchlistPage() {
|
|||||||
tags: parseTagsInput(form.tags)
|
tags: parseTagsInput(form.tags)
|
||||||
};
|
};
|
||||||
|
|
||||||
if (editingItemId === null) {
|
if (isCreate) {
|
||||||
await upsertWatchlistItem({
|
const response = await upsertWatchlistItem({
|
||||||
ticker: form.ticker.trim().toUpperCase(),
|
ticker,
|
||||||
...payload
|
...payload
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (!wasExistingItem) {
|
||||||
|
setPostCreateNotice({
|
||||||
|
ticker: response.item.ticker,
|
||||||
|
category: response.item.category,
|
||||||
|
tags: response.item.tags,
|
||||||
|
syncState: 'idle',
|
||||||
|
error: null
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
setPostCreateNotice(null);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
await updateWatchlistItem(editingItemId, payload);
|
await updateWatchlistItem(editingItemId, payload);
|
||||||
|
setPostCreateNotice(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
invalidateCoverageQueries(form.ticker);
|
invalidateCoverageQueries(ticker);
|
||||||
await loadCoverage();
|
await loadCoverage();
|
||||||
resetForm();
|
resetForm();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
@@ -248,19 +293,51 @@ export default function WatchlistPage() {
|
|||||||
|
|
||||||
const queueSync = async (item: WatchlistItem) => {
|
const queueSync = async (item: WatchlistItem) => {
|
||||||
try {
|
try {
|
||||||
await queueFilingSync({
|
setError(null);
|
||||||
|
await queueCoverageSync({
|
||||||
ticker: item.ticker,
|
ticker: item.ticker,
|
||||||
limit: 20,
|
category: item.category,
|
||||||
category: item.category ?? undefined,
|
tags: item.tags
|
||||||
tags: item.tags.length > 0 ? item.tags : undefined
|
|
||||||
});
|
});
|
||||||
void queryClient.invalidateQueries({ queryKey: queryKeys.recentTasks(20) });
|
|
||||||
void queryClient.invalidateQueries({ queryKey: queryKeys.filings(item.ticker, 120) });
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
setError(err instanceof Error ? err.message : `Failed to queue filing sync for ${item.ticker}`);
|
setError(err instanceof Error ? err.message : `Failed to queue filing sync for ${item.ticker}`);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const queuePostCreateSync = async () => {
|
||||||
|
if (!postCreateNotice || postCreateNotice.syncState === 'queued') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setPostCreateNotice((current) => current
|
||||||
|
? {
|
||||||
|
...current,
|
||||||
|
syncState: 'pending',
|
||||||
|
error: null
|
||||||
|
}
|
||||||
|
: null);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await queueCoverageSync(postCreateNotice);
|
||||||
|
setPostCreateNotice((current) => current
|
||||||
|
? {
|
||||||
|
...current,
|
||||||
|
syncState: 'queued',
|
||||||
|
error: null
|
||||||
|
}
|
||||||
|
: null);
|
||||||
|
} catch (err) {
|
||||||
|
const message = err instanceof Error ? err.message : `Failed to queue filing sync for ${postCreateNotice.ticker}`;
|
||||||
|
setPostCreateNotice((current) => current
|
||||||
|
? {
|
||||||
|
...current,
|
||||||
|
syncState: 'idle',
|
||||||
|
error: message
|
||||||
|
}
|
||||||
|
: null);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
if (isPending || !isAuthenticated) {
|
if (isPending || !isAuthenticated) {
|
||||||
return <div className="flex min-h-screen items-center justify-center text-sm text-[color:var(--terminal-muted)]">Loading coverage terminal...</div>;
|
return <div className="flex min-h-screen items-center justify-center text-sm text-[color:var(--terminal-muted)]">Loading coverage terminal...</div>;
|
||||||
}
|
}
|
||||||
@@ -407,7 +484,7 @@ export default function WatchlistPage() {
|
|||||||
|
|
||||||
<div className="mt-3 grid grid-cols-2 gap-2">
|
<div className="mt-3 grid grid-cols-2 gap-2">
|
||||||
<Button variant="secondary" className="px-2 py-1 text-xs" onClick={() => void queueSync(item)}>
|
<Button variant="secondary" className="px-2 py-1 text-xs" onClick={() => void queueSync(item)}>
|
||||||
Sync
|
Sync filings
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
variant="ghost"
|
variant="ghost"
|
||||||
@@ -437,6 +514,9 @@ export default function WatchlistPage() {
|
|||||||
await deleteWatchlistItem(item.id);
|
await deleteWatchlistItem(item.id);
|
||||||
invalidateCoverageQueries(item.ticker);
|
invalidateCoverageQueries(item.ticker);
|
||||||
await loadCoverage();
|
await loadCoverage();
|
||||||
|
if (postCreateNotice?.ticker === item.ticker) {
|
||||||
|
setPostCreateNotice(null);
|
||||||
|
}
|
||||||
if (editingItemId === item.id) {
|
if (editingItemId === item.id) {
|
||||||
resetForm();
|
resetForm();
|
||||||
}
|
}
|
||||||
@@ -567,7 +647,7 @@ export default function WatchlistPage() {
|
|||||||
Filings
|
Filings
|
||||||
</Link>
|
</Link>
|
||||||
<Button variant="secondary" className="px-2 py-1 text-xs" onClick={() => void queueSync(item)}>
|
<Button variant="secondary" className="px-2 py-1 text-xs" onClick={() => void queueSync(item)}>
|
||||||
Sync
|
Sync filings
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
variant="ghost"
|
variant="ghost"
|
||||||
@@ -597,6 +677,9 @@ export default function WatchlistPage() {
|
|||||||
await deleteWatchlistItem(item.id);
|
await deleteWatchlistItem(item.id);
|
||||||
invalidateCoverageQueries(item.ticker);
|
invalidateCoverageQueries(item.ticker);
|
||||||
await loadCoverage();
|
await loadCoverage();
|
||||||
|
if (postCreateNotice?.ticker === item.ticker) {
|
||||||
|
setPostCreateNotice(null);
|
||||||
|
}
|
||||||
if (editingItemId === item.id) {
|
if (editingItemId === item.id) {
|
||||||
resetForm();
|
resetForm();
|
||||||
}
|
}
|
||||||
@@ -625,6 +708,50 @@ export default function WatchlistPage() {
|
|||||||
variant="surface"
|
variant="surface"
|
||||||
>
|
>
|
||||||
<form onSubmit={saveCoverage} className="space-y-3">
|
<form onSubmit={saveCoverage} className="space-y-3">
|
||||||
|
{editingItemId === null ? (
|
||||||
|
<p className="rounded-lg border border-[color:var(--line-weak)] bg-[color:var(--panel-soft)] px-3 py-2 text-sm text-[color:var(--terminal-muted)]">
|
||||||
|
Saving coverage adds the company to your board only. Filing sync starts when you choose Sync filings.
|
||||||
|
</p>
|
||||||
|
) : null}
|
||||||
|
{postCreateNotice ? (
|
||||||
|
<div
|
||||||
|
data-testid="watchlist-post-create-notice"
|
||||||
|
className="rounded-lg border border-[color:var(--line-weak)] bg-[color:var(--panel-soft)] px-3 py-3 text-sm"
|
||||||
|
>
|
||||||
|
<p className="text-[color:var(--terminal-bright)]">
|
||||||
|
{postCreateNotice.syncState === 'queued'
|
||||||
|
? `${postCreateNotice.ticker} added to coverage. Filing sync is queued.`
|
||||||
|
: `${postCreateNotice.ticker} added to coverage. Filing sync has not started yet.`}
|
||||||
|
</p>
|
||||||
|
{postCreateNotice.error ? (
|
||||||
|
<p className="mt-2 text-sm text-[#ffb5b5]">{postCreateNotice.error}</p>
|
||||||
|
) : null}
|
||||||
|
<div className="mt-3 flex flex-wrap gap-2">
|
||||||
|
<Button
|
||||||
|
type="button"
|
||||||
|
className="w-full sm:w-auto"
|
||||||
|
disabled={postCreateNotice.syncState === 'pending' || postCreateNotice.syncState === 'queued'}
|
||||||
|
onClick={() => {
|
||||||
|
void queuePostCreateSync();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{postCreateNotice.syncState === 'pending'
|
||||||
|
? 'Queueing...'
|
||||||
|
: postCreateNotice.syncState === 'queued'
|
||||||
|
? 'Sync queued'
|
||||||
|
: 'Sync filings'}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
type="button"
|
||||||
|
variant="ghost"
|
||||||
|
className="w-full sm:w-auto"
|
||||||
|
onClick={() => setPostCreateNotice(null)}
|
||||||
|
>
|
||||||
|
Dismiss
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : null}
|
||||||
<div>
|
<div>
|
||||||
<label className="mb-1 block text-xs uppercase tracking-[0.2em] text-[color:var(--terminal-muted)]">Ticker</label>
|
<label className="mb-1 block text-xs uppercase tracking-[0.2em] text-[color:var(--terminal-muted)]">Ticker</label>
|
||||||
<Input
|
<Input
|
||||||
|
|||||||
86
e2e/watchlist.spec.ts
Normal file
86
e2e/watchlist.spec.ts
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
import { expect, test, type Page } from '@playwright/test';
|
||||||
|
|
||||||
|
const PASSWORD = 'Sup3rSecure!123';
|
||||||
|
|
||||||
|
test.describe.configure({ mode: 'serial' });
|
||||||
|
|
||||||
|
function uniqueEmail(prefix: string) {
|
||||||
|
return `${prefix}-${Date.now()}-${Math.random().toString(36).slice(2, 8)}@example.com`;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function gotoAuthPage(page: Page, path: string) {
|
||||||
|
await page.goto(path, { waitUntil: 'domcontentloaded' });
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
}
|
||||||
|
|
||||||
|
async function signUp(page: Page, email: string) {
|
||||||
|
await gotoAuthPage(page, '/auth/signup');
|
||||||
|
await page.locator('input[autocomplete="name"]').fill('Playwright Watchlist User');
|
||||||
|
await page.locator('input[autocomplete="email"]').fill(email);
|
||||||
|
await page.locator('input[autocomplete="new-password"]').first().fill(PASSWORD);
|
||||||
|
await page.locator('input[autocomplete="new-password"]').nth(1).fill(PASSWORD);
|
||||||
|
await page.getByRole('button', { name: 'Create account' }).click();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function expectStableDashboard(page: Page) {
|
||||||
|
await expect(page.getByRole('heading', { name: 'Command Center' })).toBeVisible({ timeout: 30_000 });
|
||||||
|
await expect(page).toHaveURL(/\/$/, { timeout: 30_000 });
|
||||||
|
}
|
||||||
|
|
||||||
|
async function countSyncTasks(page: Page, ticker: string) {
|
||||||
|
return await page.evaluate(async (requestedTicker) => {
|
||||||
|
const response = await fetch('/api/tasks?limit=20', {
|
||||||
|
credentials: 'include',
|
||||||
|
cache: 'no-store'
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`Unable to load tasks: ${response.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const payload = await response.json() as {
|
||||||
|
tasks?: Array<{
|
||||||
|
task_type?: string;
|
||||||
|
payload?: {
|
||||||
|
ticker?: string;
|
||||||
|
};
|
||||||
|
}>;
|
||||||
|
};
|
||||||
|
|
||||||
|
return (payload.tasks ?? []).filter((task) => (
|
||||||
|
task.task_type === 'sync_filings'
|
||||||
|
&& task.payload?.ticker === requestedTicker
|
||||||
|
)).length;
|
||||||
|
}, ticker);
|
||||||
|
}
|
||||||
|
|
||||||
|
test('coverage save stays metadata-only until sync filings is clicked', async ({ page }) => {
|
||||||
|
await signUp(page, uniqueEmail('playwright-watchlist-sync'));
|
||||||
|
await expectStableDashboard(page);
|
||||||
|
|
||||||
|
await page.goto('/watchlist', { waitUntil: 'domcontentloaded' });
|
||||||
|
await expect(page.getByRole('heading', { name: 'Coverage', exact: true })).toBeVisible({ timeout: 30_000 });
|
||||||
|
|
||||||
|
await page.getByLabel('Coverage ticker').fill('NVDA');
|
||||||
|
await page.getByLabel('Coverage company name').fill('NVIDIA Corporation');
|
||||||
|
await page.getByLabel('Coverage sector').fill('Technology');
|
||||||
|
await page.getByLabel('Coverage category').fill('core');
|
||||||
|
await page.getByLabel('Coverage tags').fill('semis, ai');
|
||||||
|
await page.getByRole('button', { name: 'Save coverage' }).click();
|
||||||
|
|
||||||
|
const coverageRow = page.locator('tr').filter({ hasText: 'NVDA' });
|
||||||
|
await expect(coverageRow).toContainText('NVIDIA Corporation');
|
||||||
|
|
||||||
|
const notice = page.getByTestId('watchlist-post-create-notice');
|
||||||
|
await expect(notice).toContainText('NVDA added to coverage. Filing sync has not started yet.');
|
||||||
|
await expect(notice.getByRole('button', { name: 'Sync filings' })).toBeVisible();
|
||||||
|
await expect(coverageRow.getByRole('button', { name: 'Sync filings' })).toBeVisible();
|
||||||
|
|
||||||
|
await page.waitForTimeout(1_000);
|
||||||
|
expect(await countSyncTasks(page, 'NVDA')).toBe(0);
|
||||||
|
|
||||||
|
await notice.getByRole('button', { name: 'Sync filings' }).click();
|
||||||
|
|
||||||
|
await expect(notice).toContainText('NVDA added to coverage. Filing sync is queued.');
|
||||||
|
await expect.poll(async () => await countSyncTasks(page, 'NVDA')).toBe(1);
|
||||||
|
});
|
||||||
@@ -168,7 +168,7 @@ export async function upsertWatchlistItem(input: {
|
|||||||
lastReviewedAt?: string;
|
lastReviewedAt?: string;
|
||||||
}) {
|
}) {
|
||||||
const result = await client.api.watchlist.post(input);
|
const result = await client.api.watchlist.post(input);
|
||||||
return await unwrapData<{ item: WatchlistItem }>(result, 'Unable to save watchlist item');
|
return await unwrapData<{ item: WatchlistItem; autoFilingSyncQueued: boolean }>(result, 'Unable to save watchlist item');
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function updateWatchlistItem(id: number, input: {
|
export async function updateWatchlistItem(id: number, input: {
|
||||||
|
|||||||
@@ -500,7 +500,7 @@ export const app = new Elysia({ prefix: '/api' })
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const { item, created } = await upsertWatchlistItemRecord({
|
const { item } = await upsertWatchlistItemRecord({
|
||||||
userId: session.user.id,
|
userId: session.user.id,
|
||||||
ticker,
|
ticker,
|
||||||
companyName,
|
companyName,
|
||||||
@@ -512,14 +512,10 @@ export const app = new Elysia({ prefix: '/api' })
|
|||||||
lastReviewedAt
|
lastReviewedAt
|
||||||
});
|
});
|
||||||
|
|
||||||
const autoFilingSyncQueued = created
|
return Response.json({
|
||||||
? await queueAutoFilingSync(session.user.id, ticker, {
|
item,
|
||||||
category: item.category,
|
autoFilingSyncQueued: false
|
||||||
tags: item.tags
|
});
|
||||||
})
|
|
||||||
: false;
|
|
||||||
|
|
||||||
return Response.json({ item, autoFilingSyncQueued });
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
return jsonError(asErrorMessage(error, 'Failed to create watchlist item'));
|
return jsonError(asErrorMessage(error, 'Failed to create watchlist item'));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -393,7 +393,7 @@ if (process.env.RUN_TASK_WORKFLOW_E2E === '1') {
|
|||||||
expect(tasks.every((task) => typeof task.workflow_run_id === 'string' && task.workflow_run_id.length > 0)).toBe(true);
|
expect(tasks.every((task) => typeof task.workflow_run_id === 'string' && task.workflow_run_id.length > 0)).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('persists watchlist category and tags and forwards them to auto sync task payload', async () => {
|
it('persists watchlist category and tags without auto queueing a filing sync task', async () => {
|
||||||
const created = await jsonRequest('POST', '/api/watchlist', {
|
const created = await jsonRequest('POST', '/api/watchlist', {
|
||||||
ticker: 'shop',
|
ticker: 'shop',
|
||||||
companyName: 'Shopify Inc.',
|
companyName: 'Shopify Inc.',
|
||||||
@@ -415,12 +415,12 @@ if (process.env.RUN_TASK_WORKFLOW_E2E === '1') {
|
|||||||
expect(createdBody.item.ticker).toBe('SHOP');
|
expect(createdBody.item.ticker).toBe('SHOP');
|
||||||
expect(createdBody.item.category).toBe('core');
|
expect(createdBody.item.category).toBe('core');
|
||||||
expect(createdBody.item.tags).toEqual(['growth', 'ecommerce']);
|
expect(createdBody.item.tags).toEqual(['growth', 'ecommerce']);
|
||||||
expect(createdBody.autoFilingSyncQueued).toBe(true);
|
expect(createdBody.autoFilingSyncQueued).toBe(false);
|
||||||
|
|
||||||
const tasksResponse = await jsonRequest('GET', '/api/tasks?limit=5');
|
const tasksResponse = await jsonRequest('GET', '/api/tasks?limit=5');
|
||||||
expect(tasksResponse.response.status).toBe(200);
|
expect(tasksResponse.response.status).toBe(200);
|
||||||
|
|
||||||
const task = (tasksResponse.json as {
|
const syncTasks = (tasksResponse.json as {
|
||||||
tasks: Array<{
|
tasks: Array<{
|
||||||
task_type: string;
|
task_type: string;
|
||||||
payload: {
|
payload: {
|
||||||
@@ -430,13 +430,86 @@ if (process.env.RUN_TASK_WORKFLOW_E2E === '1') {
|
|||||||
limit?: number;
|
limit?: number;
|
||||||
};
|
};
|
||||||
}>;
|
}>;
|
||||||
}).tasks.find((entry) => entry.task_type === 'sync_filings');
|
}).tasks.filter((entry) => entry.task_type === 'sync_filings');
|
||||||
|
|
||||||
expect(task).toBeTruthy();
|
expect(syncTasks).toHaveLength(0);
|
||||||
expect(task?.payload.ticker).toBe('SHOP');
|
});
|
||||||
expect(task?.payload.limit).toBe(20);
|
|
||||||
expect(task?.payload.category).toBe('core');
|
it('does not queue a filing sync task when coverage metadata is edited', async () => {
|
||||||
expect(task?.payload.tags).toEqual(['growth', 'ecommerce']);
|
const created = await jsonRequest('POST', '/api/watchlist', {
|
||||||
|
ticker: 'amd',
|
||||||
|
companyName: 'Advanced Micro Devices, Inc.',
|
||||||
|
sector: 'Technology',
|
||||||
|
category: 'watch',
|
||||||
|
tags: ['semis']
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(created.response.status).toBe(200);
|
||||||
|
const item = (created.json as {
|
||||||
|
item: { id: number };
|
||||||
|
}).item;
|
||||||
|
|
||||||
|
const updated = await jsonRequest('PATCH', `/api/watchlist/${item.id}`, {
|
||||||
|
category: 'core',
|
||||||
|
tags: ['semis', 'ai']
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(updated.response.status).toBe(200);
|
||||||
|
|
||||||
|
const tasksResponse = await jsonRequest('GET', '/api/tasks?limit=5');
|
||||||
|
expect(tasksResponse.response.status).toBe(200);
|
||||||
|
|
||||||
|
const syncTasks = (tasksResponse.json as {
|
||||||
|
tasks: Array<{ task_type: string }>;
|
||||||
|
}).tasks.filter((entry) => entry.task_type === 'sync_filings');
|
||||||
|
|
||||||
|
expect(syncTasks).toHaveLength(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('forwards watchlist metadata when filing sync is started explicitly', async () => {
|
||||||
|
const created = await jsonRequest('POST', '/api/watchlist', {
|
||||||
|
ticker: 'shop',
|
||||||
|
companyName: 'Shopify Inc.',
|
||||||
|
sector: 'Technology',
|
||||||
|
category: 'core',
|
||||||
|
tags: ['growth', 'ecommerce', 'growth', ' ']
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(created.response.status).toBe(200);
|
||||||
|
const createdBody = created.json as {
|
||||||
|
item: {
|
||||||
|
ticker: string;
|
||||||
|
category: string | null;
|
||||||
|
tags: string[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const sync = await jsonRequest('POST', '/api/filings/sync', {
|
||||||
|
ticker: createdBody.item.ticker,
|
||||||
|
limit: 20,
|
||||||
|
category: createdBody.item.category,
|
||||||
|
tags: createdBody.item.tags
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(sync.response.status).toBe(200);
|
||||||
|
|
||||||
|
const task = (sync.json as {
|
||||||
|
task: {
|
||||||
|
task_type: string;
|
||||||
|
payload: {
|
||||||
|
ticker: string;
|
||||||
|
limit: number;
|
||||||
|
category?: string;
|
||||||
|
tags?: string[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}).task;
|
||||||
|
|
||||||
|
expect(task.task_type).toBe('sync_filings');
|
||||||
|
expect(task.payload.ticker).toBe('SHOP');
|
||||||
|
expect(task.payload.limit).toBe(20);
|
||||||
|
expect(task.payload.category).toBe('core');
|
||||||
|
expect(task.payload.tags).toEqual(['growth', 'ecommerce']);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('accepts category and comma-separated tags on manual filings sync payload', async () => {
|
it('accepts category and comma-separated tags on manual filings sync payload', async () => {
|
||||||
|
|||||||
Reference in New Issue
Block a user