NotificationProviderCardStockAlerts.test.tsx 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. /**
  2. * Tests for stock alert toggles added to NotificationProviderCard.
  3. *
  4. * Coverage:
  5. * - Reorder Alert and Stock Break Alert badges render in the summary strip when enabled.
  6. * - Badges are absent when both flags are false.
  7. * - Inventory Alerts section renders in the expanded settings panel.
  8. * - Toggling a stock alert fires an update mutation with the correct field.
  9. */
  10. import { describe, it, expect, afterEach, vi } from 'vitest';
  11. import { screen, waitFor, within } from '@testing-library/react';
  12. import userEvent from '@testing-library/user-event';
  13. import { http, HttpResponse } from 'msw';
  14. import { render } from '../utils';
  15. import { server } from '../mocks/server';
  16. import { NotificationProviderCard } from '../../components/NotificationProviderCard';
  17. import type { NotificationProvider } from '../../api/client';
  18. afterEach(() => {
  19. server.resetHandlers();
  20. vi.restoreAllMocks();
  21. });
  22. function buildProvider(overrides: Partial<NotificationProvider> = {}): NotificationProvider {
  23. return {
  24. id: 1,
  25. name: 'Test Provider',
  26. provider_type: 'ntfy',
  27. enabled: true,
  28. config: { server: 'https://ntfy.sh', topic: 'bambuddy' },
  29. on_print_start: false,
  30. on_print_complete: false,
  31. on_print_failed: false,
  32. on_print_stopped: false,
  33. on_print_progress: false,
  34. on_print_missing_spool_assignment: false,
  35. on_printer_offline: false,
  36. on_printer_error: false,
  37. on_filament_low: false,
  38. on_maintenance_due: false,
  39. on_ams_humidity_high: false,
  40. on_ams_temperature_high: false,
  41. on_ams_ht_humidity_high: false,
  42. on_ams_ht_temperature_high: false,
  43. on_plate_not_empty: false,
  44. on_bed_cooled: false,
  45. on_first_layer_complete: false,
  46. on_queue_job_added: false,
  47. on_queue_job_assigned: false,
  48. on_queue_job_started: false,
  49. on_queue_job_waiting: false,
  50. on_queue_job_skipped: false,
  51. on_queue_job_failed: false,
  52. on_queue_completed: false,
  53. on_stock_reorder_alert: false,
  54. on_stock_break_alert: false,
  55. quiet_hours_enabled: false,
  56. quiet_hours_start: null,
  57. quiet_hours_end: null,
  58. daily_digest_enabled: false,
  59. daily_digest_time: null,
  60. printer_id: null,
  61. last_success: null,
  62. last_error: null,
  63. last_error_at: null,
  64. created_at: '2026-04-25T00:00:00Z',
  65. updated_at: '2026-04-25T00:00:00Z',
  66. ...overrides,
  67. };
  68. }
  69. describe('NotificationProviderCard — stock alert badges', () => {
  70. it('shows Reorder Alert badge when on_stock_reorder_alert is true', async () => {
  71. render(<NotificationProviderCard provider={buildProvider({ on_stock_reorder_alert: true })} onEdit={vi.fn()} />);
  72. expect(await screen.findByText('Reorder Alert')).toBeInTheDocument();
  73. });
  74. it('shows Stock Break Alert badge when on_stock_break_alert is true', async () => {
  75. render(<NotificationProviderCard provider={buildProvider({ on_stock_break_alert: true })} onEdit={vi.fn()} />);
  76. expect(await screen.findByText('Stock Break Alert')).toBeInTheDocument();
  77. });
  78. it('shows both badges when both flags are true', async () => {
  79. render(
  80. <NotificationProviderCard
  81. provider={buildProvider({ on_stock_reorder_alert: true, on_stock_break_alert: true })}
  82. onEdit={vi.fn()}
  83. />,
  84. );
  85. expect(await screen.findByText('Reorder Alert')).toBeInTheDocument();
  86. expect(screen.getByText('Stock Break Alert')).toBeInTheDocument();
  87. });
  88. it('shows no stock alert badges when both flags are false', async () => {
  89. render(<NotificationProviderCard provider={buildProvider()} onEdit={vi.fn()} />);
  90. // Wait for the card to mount
  91. await screen.findByText('Test Provider');
  92. expect(screen.queryByText('Reorder Alert')).not.toBeInTheDocument();
  93. expect(screen.queryByText('Stock Break Alert')).not.toBeInTheDocument();
  94. });
  95. });
  96. describe('NotificationProviderCard — Inventory Alerts expanded section', () => {
  97. it('renders the Inventory Alerts section header when settings are expanded', async () => {
  98. const user = userEvent.setup();
  99. render(<NotificationProviderCard provider={buildProvider()} onEdit={vi.fn()} />);
  100. const settingsBtn = await screen.findByText(/event settings/i);
  101. await user.click(settingsBtn);
  102. expect(await screen.findByText('Inventory Alerts')).toBeInTheDocument();
  103. });
  104. it('renders both stock alert toggles in the expanded section', async () => {
  105. const user = userEvent.setup();
  106. render(<NotificationProviderCard provider={buildProvider()} onEdit={vi.fn()} />);
  107. await user.click(await screen.findByText(/event settings/i));
  108. const section = (await screen.findByText('Inventory Alerts')).closest('div')!;
  109. expect(within(section).getByText('Reorder Alert')).toBeInTheDocument();
  110. expect(within(section).getByText('Stock Break Alert')).toBeInTheDocument();
  111. });
  112. it('stock alert toggles reflect the provider state', async () => {
  113. const user = userEvent.setup();
  114. render(
  115. <NotificationProviderCard
  116. provider={buildProvider({ on_stock_reorder_alert: true, on_stock_break_alert: false })}
  117. onEdit={vi.fn()}
  118. />,
  119. );
  120. await user.click(await screen.findByText(/event settings/i));
  121. const section = (await screen.findByText('Inventory Alerts')).closest('div')!;
  122. const switches = within(section).getAllByRole('switch');
  123. // First switch = reorder alert (true), second = break alert (false)
  124. expect(switches[0]).toHaveAttribute('aria-checked', 'true');
  125. expect(switches[1]).toHaveAttribute('aria-checked', 'false');
  126. });
  127. it('toggling Reorder Alert sends correct PATCH payload', async () => {
  128. let captured: unknown = null;
  129. server.use(
  130. http.patch('*/api/v1/notifications/1', async ({ request }) => {
  131. captured = await request.json();
  132. return HttpResponse.json(buildProvider({ on_stock_reorder_alert: true }));
  133. }),
  134. );
  135. const user = userEvent.setup();
  136. render(<NotificationProviderCard provider={buildProvider()} onEdit={vi.fn()} />);
  137. await user.click(await screen.findByText(/event settings/i));
  138. const section = (await screen.findByText('Inventory Alerts')).closest('div')!;
  139. const [reorderSwitch] = within(section).getAllByRole('switch');
  140. await user.click(reorderSwitch);
  141. await waitFor(() => expect(captured).not.toBeNull());
  142. expect(captured).toMatchObject({ on_stock_reorder_alert: true });
  143. });
  144. it('toggling Stock Break Alert sends correct PATCH payload', async () => {
  145. let captured: unknown = null;
  146. server.use(
  147. http.patch('*/api/v1/notifications/1', async ({ request }) => {
  148. captured = await request.json();
  149. return HttpResponse.json(buildProvider({ on_stock_break_alert: true }));
  150. }),
  151. );
  152. const user = userEvent.setup();
  153. render(<NotificationProviderCard provider={buildProvider()} onEdit={vi.fn()} />);
  154. await user.click(await screen.findByText(/event settings/i));
  155. const section = (await screen.findByText('Inventory Alerts')).closest('div')!;
  156. const switches = within(section).getAllByRole('switch');
  157. await user.click(switches[1]);
  158. await waitFor(() => expect(captured).not.toBeNull());
  159. expect(captured).toMatchObject({ on_stock_break_alert: true });
  160. });
  161. });