|
| 1 | +import { expect, test } from '@playwright/test'; |
| 2 | +import { |
| 3 | + getSpanOp, |
| 4 | + waitForError, |
| 5 | + waitForRequest, |
| 6 | + waitForStreamedSpan, |
| 7 | + waitForStreamedSpans, |
| 8 | +} from '@sentry-internal/test-utils'; |
| 9 | +import { SDK_VERSION } from '@sentry/cloudflare'; |
| 10 | +import { WebSocket } from 'ws'; |
| 11 | + |
| 12 | +test('Index page', async ({ baseURL }) => { |
| 13 | + const result = await fetch(baseURL!); |
| 14 | + expect(result.status).toBe(200); |
| 15 | + await expect(result.text()).resolves.toBe('Hello World!'); |
| 16 | +}); |
| 17 | + |
| 18 | +test('Sends a streamed span for a basic request', async ({ baseURL }) => { |
| 19 | + const spanPromise = waitForStreamedSpan('cloudflare-workers-streaming', span => { |
| 20 | + return getSpanOp(span) === 'http.server' && span.is_segment; |
| 21 | + }); |
| 22 | + |
| 23 | + await fetch(baseURL!); |
| 24 | + |
| 25 | + const span = await spanPromise; |
| 26 | + |
| 27 | + expect(span.trace_id).toMatch(/[a-f0-9]{32}/); |
| 28 | + expect(span.status).toBe('ok'); |
| 29 | +}); |
| 30 | + |
| 31 | +test("worker's withSentry", async ({ baseURL }) => { |
| 32 | + const eventWaiter = waitForError('cloudflare-workers-streaming', event => { |
| 33 | + return event.exception?.values?.[0]?.mechanism?.type === 'auto.http.cloudflare'; |
| 34 | + }); |
| 35 | + const response = await fetch(`${baseURL}/throwException`); |
| 36 | + expect(response.status).toBe(500); |
| 37 | + const event = await eventWaiter; |
| 38 | + expect(event.exception?.values?.[0]?.value).toBe('To be recorded in Sentry.'); |
| 39 | +}); |
| 40 | + |
| 41 | +test('RPC method which throws an exception to be logged to sentry', async ({ baseURL }) => { |
| 42 | + const eventWaiter = waitForError('cloudflare-workers-streaming', event => { |
| 43 | + return event.exception?.values?.[0]?.mechanism?.type === 'auto.faas.cloudflare.durable_object'; |
| 44 | + }); |
| 45 | + const response = await fetch(`${baseURL}/rpc/throwException`); |
| 46 | + expect(response.status).toBe(500); |
| 47 | + const event = await eventWaiter; |
| 48 | + expect(event.exception?.values?.[0]?.value).toBe('Should be recorded in Sentry.'); |
| 49 | +}); |
| 50 | + |
| 51 | +test("Request processed by DurableObject's fetch is recorded", async ({ baseURL }) => { |
| 52 | + const eventWaiter = waitForError('cloudflare-workers-streaming', event => { |
| 53 | + return event.exception?.values?.[0]?.mechanism?.type === 'auto.faas.cloudflare.durable_object'; |
| 54 | + }); |
| 55 | + const response = await fetch(`${baseURL}/pass-to-object/throwException`); |
| 56 | + expect(response.status).toBe(500); |
| 57 | + const event = await eventWaiter; |
| 58 | + expect(event.exception?.values?.[0]?.value).toBe('Should be recorded in Sentry.'); |
| 59 | +}); |
| 60 | + |
| 61 | +test('Websocket.webSocketMessage', async ({ baseURL }) => { |
| 62 | + const eventWaiter = waitForError('cloudflare-workers-streaming', event => { |
| 63 | + return !!event.exception?.values?.[0]; |
| 64 | + }); |
| 65 | + const url = new URL('/pass-to-object/ws', baseURL); |
| 66 | + url.protocol = url.protocol.replace('http', 'ws'); |
| 67 | + const socket = new WebSocket(url.toString()); |
| 68 | + socket.addEventListener('open', () => { |
| 69 | + socket.send('throwException'); |
| 70 | + }); |
| 71 | + const event = await eventWaiter; |
| 72 | + socket.close(); |
| 73 | + expect(event.exception?.values?.[0]?.value).toBe('Should be recorded in Sentry: webSocketMessage'); |
| 74 | + expect(event.exception?.values?.[0]?.mechanism?.type).toBe('auto.faas.cloudflare.durable_object'); |
| 75 | +}); |
| 76 | + |
| 77 | +test('Websocket.webSocketClose', async ({ baseURL }) => { |
| 78 | + const eventWaiter = waitForError('cloudflare-workers-streaming', event => { |
| 79 | + return !!event.exception?.values?.[0]; |
| 80 | + }); |
| 81 | + const url = new URL('/pass-to-object/ws', baseURL); |
| 82 | + url.protocol = url.protocol.replace('http', 'ws'); |
| 83 | + const socket = new WebSocket(url.toString()); |
| 84 | + socket.addEventListener('open', () => { |
| 85 | + socket.send('throwOnExit'); |
| 86 | + socket.close(); |
| 87 | + }); |
| 88 | + const event = await eventWaiter; |
| 89 | + expect(event.exception?.values?.[0]?.value).toBe('Should be recorded in Sentry: webSocketClose'); |
| 90 | + expect(event.exception?.values?.[0]?.mechanism?.type).toBe('auto.faas.cloudflare.durable_object'); |
| 91 | +}); |
| 92 | + |
| 93 | +test('sends user-agent header with SDK name and version in envelope requests', async ({ baseURL }) => { |
| 94 | + const requestPromise = waitForRequest('cloudflare-workers-streaming', () => true); |
| 95 | + |
| 96 | + await fetch(`${baseURL}/throwException`); |
| 97 | + |
| 98 | + const request = await requestPromise; |
| 99 | + |
| 100 | + expect(request.rawProxyRequestHeaders).toMatchObject({ |
| 101 | + 'user-agent': `sentry.javascript.cloudflare/${SDK_VERSION}`, |
| 102 | + }); |
| 103 | +}); |
| 104 | + |
| 105 | +test('Storage operations create spans in Durable Object', async ({ baseURL }) => { |
| 106 | + const spansPromise = waitForStreamedSpans('cloudflare-workers-streaming', spans => { |
| 107 | + return spans.some(span => span.name === 'durable_object_storage_put' && getSpanOp(span) === 'db'); |
| 108 | + }); |
| 109 | + |
| 110 | + const response = await fetch(`${baseURL}/pass-to-object/storage/put`); |
| 111 | + expect(response.status).toBe(200); |
| 112 | + |
| 113 | + const spans = await spansPromise; |
| 114 | + const putSpan = spans.find(span => span.name === 'durable_object_storage_put' && getSpanOp(span) === 'db'); |
| 115 | + |
| 116 | + expect(putSpan).toBeDefined(); |
| 117 | + expect(putSpan?.attributes?.['db.system.name']?.value).toBe('cloudflare.durable_object.storage'); |
| 118 | + expect(putSpan?.attributes?.['db.operation.name']?.value).toBe('put'); |
| 119 | +}); |
| 120 | + |
| 121 | +test.describe('Alarm instrumentation', () => { |
| 122 | + test.describe.configure({ mode: 'serial' }); |
| 123 | + |
| 124 | + test('captures error from alarm handler', async ({ baseURL }) => { |
| 125 | + const errorWaiter = waitForError('cloudflare-workers-streaming', event => { |
| 126 | + return event.exception?.values?.[0]?.value === 'Alarm error captured by Sentry'; |
| 127 | + }); |
| 128 | + |
| 129 | + const response = await fetch(`${baseURL}/pass-to-object/setAlarm?action=throw`); |
| 130 | + expect(response.status).toBe(200); |
| 131 | + |
| 132 | + const event = await errorWaiter; |
| 133 | + expect(event.exception?.values?.[0]?.mechanism?.type).toBe('auto.faas.cloudflare.durable_object'); |
| 134 | + }); |
| 135 | + |
| 136 | + test('creates a streamed span for alarm with new trace linked to setAlarm', async ({ baseURL }) => { |
| 137 | + const setAlarmSpanPromise = waitForStreamedSpan('cloudflare-workers-streaming', span => { |
| 138 | + return span.name === 'durable_object_storage_setAlarm' && span.is_segment === false; |
| 139 | + }); |
| 140 | + |
| 141 | + const alarmSpanPromise = waitForStreamedSpan('cloudflare-workers-streaming', span => { |
| 142 | + return span.name === 'alarm' && getSpanOp(span) === 'function' && span.is_segment; |
| 143 | + }); |
| 144 | + |
| 145 | + const response = await fetch(`${baseURL}/pass-to-object/setAlarm`); |
| 146 | + expect(response.status).toBe(200); |
| 147 | + |
| 148 | + const setAlarmSpan = await setAlarmSpanPromise; |
| 149 | + const alarmSpan = await alarmSpanPromise; |
| 150 | + |
| 151 | + expect(getSpanOp(alarmSpan)).toBe('function'); |
| 152 | + expect(alarmSpan.attributes?.['sentry.origin']?.value).toBe('auto.faas.cloudflare.durable_object'); |
| 153 | + |
| 154 | + // Alarm starts a new trace (different trace ID from the request that called setAlarm) |
| 155 | + expect(alarmSpan.trace_id).not.toBe(setAlarmSpan.trace_id); |
| 156 | + |
| 157 | + // Alarm links to the trace that called setAlarm via sentry.previous_trace attribute |
| 158 | + const previousTrace = alarmSpan.attributes?.['sentry.previous_trace']?.value; |
| 159 | + expect(previousTrace).toBeDefined(); |
| 160 | + expect(previousTrace).toContain(setAlarmSpan.trace_id); |
| 161 | + }); |
| 162 | +}); |
0 commit comments