|
| 1 | +import { describe, it, expect, vi, beforeEach } from 'vitest' |
| 2 | +import { ChangeDataCapturePlugin } from './index' |
| 3 | +import { StarbaseDBConfiguration } from '../../src/handler' |
| 4 | +import { DataSource } from '../../src/types' |
| 5 | +import type { DurableObjectStub } from '@cloudflare/workers-types' |
| 6 | + |
| 7 | +const parser = new (require('node-sql-parser').Parser)() |
| 8 | + |
| 9 | +let cdcPlugin: ChangeDataCapturePlugin |
| 10 | +let mockDurableObjectStub: DurableObjectStub<any> |
| 11 | +let mockConfig: StarbaseDBConfiguration |
| 12 | + |
| 13 | +beforeEach(() => { |
| 14 | + vi.clearAllMocks() |
| 15 | + mockDurableObjectStub = { |
| 16 | + fetch: vi.fn().mockResolvedValue(new Response('OK', { status: 200 })), |
| 17 | + } as unknown as DurableObjectStub |
| 18 | + |
| 19 | + mockConfig = { |
| 20 | + role: 'admin', |
| 21 | + } as any |
| 22 | + |
| 23 | + cdcPlugin = new ChangeDataCapturePlugin({ |
| 24 | + stub: mockDurableObjectStub, |
| 25 | + broadcastAllEvents: false, |
| 26 | + events: [ |
| 27 | + { action: 'INSERT', schema: 'public', table: 'users' }, |
| 28 | + { action: 'DELETE', schema: 'public', table: 'orders' }, |
| 29 | + ], |
| 30 | + }) |
| 31 | +}) |
| 32 | + |
| 33 | +beforeEach(() => { |
| 34 | + vi.clearAllMocks() |
| 35 | + mockDurableObjectStub = { |
| 36 | + fetch: vi.fn(), |
| 37 | + } as any |
| 38 | + |
| 39 | + mockConfig = { |
| 40 | + role: 'admin', |
| 41 | + } as any |
| 42 | + |
| 43 | + cdcPlugin = new ChangeDataCapturePlugin({ |
| 44 | + stub: mockDurableObjectStub, |
| 45 | + broadcastAllEvents: false, |
| 46 | + events: [ |
| 47 | + { action: 'INSERT', schema: 'public', table: 'users' }, |
| 48 | + { action: 'DELETE', schema: 'public', table: 'orders' }, |
| 49 | + ], |
| 50 | + }) |
| 51 | +}) |
| 52 | + |
| 53 | +describe('ChangeDataCapturePlugin - Initialization', () => { |
| 54 | + it('should initialize correctly with given options', () => { |
| 55 | + expect(cdcPlugin.prefix).toBe('/cdc') |
| 56 | + expect(cdcPlugin.broadcastAllEvents).toBe(false) |
| 57 | + expect(cdcPlugin.listeningEvents).toHaveLength(2) |
| 58 | + }) |
| 59 | + |
| 60 | + it('should allow all events when broadcastAllEvents is true', () => { |
| 61 | + const plugin = new ChangeDataCapturePlugin({ |
| 62 | + stub: mockDurableObjectStub, |
| 63 | + broadcastAllEvents: true, |
| 64 | + }) |
| 65 | + |
| 66 | + expect(plugin.broadcastAllEvents).toBe(true) |
| 67 | + expect(plugin.listeningEvents).toBeUndefined() |
| 68 | + }) |
| 69 | +}) |
| 70 | + |
| 71 | +describe('ChangeDataCapturePlugin - isEventMatch', () => { |
| 72 | + it('should return true for matching event', () => { |
| 73 | + expect(cdcPlugin.isEventMatch('INSERT', 'public', 'users')).toBe(true) |
| 74 | + expect(cdcPlugin.isEventMatch('DELETE', 'public', 'orders')).toBe(true) |
| 75 | + }) |
| 76 | + |
| 77 | + it('should return false for non-matching event', () => { |
| 78 | + expect(cdcPlugin.isEventMatch('UPDATE', 'public', 'users')).toBe(false) |
| 79 | + expect(cdcPlugin.isEventMatch('INSERT', 'public', 'products')).toBe( |
| 80 | + false |
| 81 | + ) |
| 82 | + }) |
| 83 | + |
| 84 | + it('should return true for any event if broadcastAllEvents is enabled', () => { |
| 85 | + cdcPlugin.broadcastAllEvents = true |
| 86 | + expect(cdcPlugin.isEventMatch('INSERT', 'any', 'table')).toBe(true) |
| 87 | + }) |
| 88 | +}) |
| 89 | + |
| 90 | +describe('ChangeDataCapturePlugin - extractValuesFromQuery', () => { |
| 91 | + it('should extract values from INSERT queries', () => { |
| 92 | + const ast = parser.astify( |
| 93 | + `INSERT INTO users (id, name) VALUES (1, 'Alice')` |
| 94 | + ) |
| 95 | + const extracted = cdcPlugin.extractValuesFromQuery(ast, []) |
| 96 | + expect(extracted).toEqual({ id: 1, name: 'Alice' }) |
| 97 | + }) |
| 98 | + |
| 99 | + it('should extract values from UPDATE queries', () => { |
| 100 | + const ast = parser.astify(`UPDATE users SET name = 'Bob' WHERE id = 2`) |
| 101 | + const extracted = cdcPlugin.extractValuesFromQuery(ast, []) |
| 102 | + expect(extracted).toEqual({ name: 'Bob', id: 2 }) |
| 103 | + }) |
| 104 | + |
| 105 | + it('should extract values from DELETE queries', () => { |
| 106 | + const ast = parser.astify(`DELETE FROM users WHERE id = 3`) |
| 107 | + const extracted = cdcPlugin.extractValuesFromQuery(ast, []) |
| 108 | + expect(extracted).toEqual({ id: 3 }) |
| 109 | + }) |
| 110 | + |
| 111 | + it('should use result data when available', () => { |
| 112 | + const result = { id: 4, name: 'Charlie' } |
| 113 | + const extracted = cdcPlugin.extractValuesFromQuery({}, result) |
| 114 | + expect(extracted).toEqual(result) |
| 115 | + }) |
| 116 | +}) |
| 117 | + |
| 118 | +describe('ChangeDataCapturePlugin - queryEventDetected', () => { |
| 119 | + it('should not trigger CDC event for unmatched actions', () => { |
| 120 | + const mockCallback = vi.fn() |
| 121 | + cdcPlugin.onEvent(mockCallback) |
| 122 | + |
| 123 | + const ast = parser.astify(`UPDATE users SET name = 'Emma' WHERE id = 6`) |
| 124 | + cdcPlugin.queryEventDetected('UPDATE', ast, []) |
| 125 | + |
| 126 | + expect(mockCallback).not.toHaveBeenCalled() |
| 127 | + }) |
| 128 | +}) |
| 129 | + |
| 130 | +describe('ChangeDataCapturePlugin - onEvent', () => { |
| 131 | + it('should register event callbacks', () => { |
| 132 | + const mockCallback = vi.fn() |
| 133 | + cdcPlugin.onEvent(mockCallback) |
| 134 | + |
| 135 | + const registeredCallbacks = cdcPlugin['eventCallbacks'] |
| 136 | + |
| 137 | + expect(registeredCallbacks).toHaveLength(1) |
| 138 | + expect(registeredCallbacks[0]).toBeInstanceOf(Function) |
| 139 | + }) |
| 140 | + |
| 141 | + it('should call registered callbacks when event occurs', () => { |
| 142 | + const mockCallback = vi.fn() |
| 143 | + cdcPlugin.onEvent(mockCallback) |
| 144 | + |
| 145 | + const eventPayload = { |
| 146 | + action: 'INSERT', |
| 147 | + schema: 'public', |
| 148 | + table: 'users', |
| 149 | + data: { id: 8, name: 'Frank' }, |
| 150 | + } |
| 151 | + |
| 152 | + cdcPlugin['eventCallbacks'].forEach((cb) => cb(eventPayload)) |
| 153 | + |
| 154 | + expect(mockCallback).toHaveBeenCalledWith(eventPayload) |
| 155 | + }) |
| 156 | +}) |
0 commit comments