Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add Supabase Integration #15719

Open
wants to merge 36 commits into
base: develop
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
c173a9f
feat(core): Add Supabase Integration
onurtemizkan Mar 18, 2025
efee98d
Add missing package exports
onurtemizkan Mar 18, 2025
27913a2
Remove debug logging
onurtemizkan Mar 18, 2025
51f42a7
Bump next 14 version
onurtemizkan Mar 18, 2025
48f197f
Hard-code default development variables
onurtemizkan Mar 18, 2025
5812ac0
Add playwright to dev-dependencies.
onurtemizkan Mar 18, 2025
7a1435b
Move `supabase` into its own internal package.
onurtemizkan Mar 18, 2025
c5b70ef
Add new package reference to Remix integration tests
onurtemizkan Mar 18, 2025
cf95f17
Update deps
onurtemizkan Mar 19, 2025
3cbcafb
Separate anon and service clients.
onurtemizkan Mar 19, 2025
096ee99
Make `supabase` a public package
onurtemizkan Mar 19, 2025
69dcb3c
Add `supabase` to verdaccio config
onurtemizkan Mar 19, 2025
9d3f67b
Remove unused resolutions and dependencies
onurtemizkan Mar 20, 2025
473c9f3
Move `supabaseIntegration` to `@sentry/core`
onurtemizkan Mar 24, 2025
8ad65ab
Update import paths
onurtemizkan Mar 25, 2025
c9500ec
Fix tests
onurtemizkan Mar 25, 2025
a8f8411
Fix tests
onurtemizkan Mar 25, 2025
207044d
Fix formatting
onurtemizkan Mar 25, 2025
73d2cfd
Dedupe dependencies.
onurtemizkan Mar 25, 2025
b9b0c37
Skip tests on non-tracing bundles
onurtemizkan Mar 26, 2025
4ba5196
Deduplicate deps.
onurtemizkan Mar 26, 2025
fddc7b1
Remove test-debug mode
onurtemizkan Mar 26, 2025
cfd40bb
Try reducing bundle size
onurtemizkan Mar 26, 2025
a9b8501
Remove `supabaseIntegration` from non-Tracing bundles
onurtemizkan Mar 26, 2025
9d6c259
Bring filter-mappings back.
onurtemizkan Mar 26, 2025
7e7665a
Export supabase from all tracing bundles
onurtemizkan Mar 26, 2025
2a6dddc
Add vendor license
onurtemizkan Mar 26, 2025
69e4874
Clean up
onurtemizkan Mar 27, 2025
0a2d829
Add `auth` support
onurtemizkan Mar 27, 2025
e724a99
Add `auth` error capturing
onurtemizkan Mar 27, 2025
d387514
Remove `signOut` from `admin` operations
onurtemizkan Mar 27, 2025
b630367
Clean up
onurtemizkan Mar 31, 2025
28f41f2
Address review comments
onurtemizkan Apr 3, 2025
9755690
Update packages/core/src/integrations/supabase.ts
onurtemizkan Apr 3, 2025
4b46035
Remove README.md
onurtemizkan Apr 3, 2025
80d558a
Merge branch 'develop' into onur/supabase-integration
onurtemizkan Apr 3, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions dev-packages/browser-integration-tests/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@
"@playwright/test": "~1.50.0",
"@sentry-internal/rrweb": "2.34.0",
"@sentry/browser": "9.11.0",
"@supabase/supabase-js": "2.49.3",
"axios": "1.8.2",
"babel-loader": "^8.2.2",
"fflate": "0.8.2",
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import * as Sentry from '@sentry/browser';

import { createClient } from '@supabase/supabase-js';
window.Sentry = Sentry;

const supabase = createClient('https://test.supabase.co', 'test-key');

Sentry.init({
dsn: 'https://[email protected]/1337',
integrations: [Sentry.browserTracingIntegration(), Sentry.supabaseIntegration(supabase)],
tracesSampleRate: 1.0,
});

// Simulate authentication operations
async function performAuthenticationOperations() {
await supabase.auth.signInWithPassword({
email: '[email protected]',
password: 'test-password',
});

await supabase.auth.signOut();
}

performAuthenticationOperations();
Original file line number Diff line number Diff line change
@@ -0,0 +1,140 @@
import type { Page } from '@playwright/test';
import { expect } from '@playwright/test';
import type { Event } from '@sentry/core';

import { sentryTest } from '../../../../utils/fixtures';
import {
getFirstSentryEnvelopeRequest,
getMultipleSentryEnvelopeRequests,
shouldSkipTracingTest,
} from '../../../../utils/helpers';

async function mockSupabaseAuthRoutesSuccess(page: Page) {
await page.route('**/auth/v1/token?grant_type=password**', route => {
return route.fulfill({
status: 200,
body: JSON.stringify({
access_token: 'test-access-token',
refresh_token: 'test-refresh-token',
token_type: 'bearer',
expires_in: 3600,
}),
headers: {
'Content-Type': 'application/json',
},
});
});

await page.route('**/auth/v1/logout**', route => {
return route.fulfill({
status: 200,
body: JSON.stringify({
message: 'Logged out',
}),
headers: {
'Content-Type': 'application/json',
},
});
});
}

async function mockSupabaseAuthRoutesFailure(page: Page) {
await page.route('**/auth/v1/token?grant_type=password**', route => {
return route.fulfill({
status: 400,
body: JSON.stringify({
error_description: 'Invalid email or password',
error: 'invalid_grant',
}),
headers: {
'Content-Type': 'application/json',
},
});
});

await page.route('**/auth/v1/logout**', route => {
return route.fulfill({
status: 400,
body: JSON.stringify({
error_description: 'Invalid refresh token',
error: 'invalid_grant',
}),
headers: {
'Content-Type': 'application/json',
},
});
});
}

sentryTest('should capture Supabase authentication spans', async ({ getLocalTestUrl, page }) => {
if (shouldSkipTracingTest()) {
return;
}

await mockSupabaseAuthRoutesSuccess(page);

const url = await getLocalTestUrl({ testDir: __dirname });

const eventData = await getFirstSentryEnvelopeRequest<Event>(page, url);
const supabaseSpans = eventData.spans?.filter(({ op }) => op?.startsWith('db.supabase.auth'));

expect(supabaseSpans).toHaveLength(2);
expect(supabaseSpans![0]).toMatchObject({
description: 'signInWithPassword',
parent_span_id: eventData.contexts?.trace?.span_id,
span_id: expect.any(String),
start_timestamp: expect.any(Number),
timestamp: expect.any(Number),
trace_id: eventData.contexts?.trace?.trace_id,
status: 'ok',
data: expect.objectContaining({
'sentry.op': 'db.supabase.auth.signInWithPassword',
'sentry.origin': 'auto.db.supabase',
}),
});

expect(supabaseSpans![1]).toMatchObject({
description: 'signOut',
parent_span_id: eventData.contexts?.trace?.span_id,
span_id: expect.any(String),
start_timestamp: expect.any(Number),
timestamp: expect.any(Number),
trace_id: eventData.contexts?.trace?.trace_id,
status: 'ok',
data: expect.objectContaining({
'sentry.op': 'db.supabase.auth.signOut',
'sentry.origin': 'auto.db.supabase',
}),
});
});

sentryTest('should capture Supabase authentication errors', async ({ getLocalTestUrl, page }) => {
if (shouldSkipTracingTest()) {
return;
}

await mockSupabaseAuthRoutesFailure(page);

const url = await getLocalTestUrl({ testDir: __dirname });

const [errorEvent, transactionEvent] = await getMultipleSentryEnvelopeRequests<Event>(page, 2, { url });

const supabaseSpans = transactionEvent.spans?.filter(({ op }) => op?.startsWith('db.supabase.auth'));

expect(errorEvent.exception?.values?.[0].value).toBe('Invalid email or password');

expect(supabaseSpans).toHaveLength(2);
expect(supabaseSpans![0]).toMatchObject({
description: 'signInWithPassword',
parent_span_id: transactionEvent.contexts?.trace?.span_id,
span_id: expect.any(String),
start_timestamp: expect.any(Number),
timestamp: expect.any(Number),
trace_id: transactionEvent.contexts?.trace?.trace_id,
status: 'unknown_error',
data: expect.objectContaining({
'sentry.op': 'db.supabase.auth.signInWithPassword',
'sentry.origin': 'auto.db.supabase',
}),
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
import * as Sentry from '@sentry/browser';

import { createClient } from '@supabase/supabase-js';
window.Sentry = Sentry;

const supabase = createClient(
'https://test.supabase.co',
'test-key'
);

Sentry.init({
dsn: 'https://[email protected]/1337',
integrations: [
Sentry.browserTracingIntegration(),
Sentry.supabaseIntegration(supabase)
],
tracesSampleRate: 1.0,
});

// Simulate database operations
async function performDatabaseOperations() {
try {
await supabase
.from('todos')
.insert([{ title: 'Test Todo' }]);

await supabase
.from('todos')
.select('*');

// Trigger an error to capture the breadcrumbs
throw new Error('Test Error');
} catch (error) {
Sentry.captureException(error);
}
}

performDatabaseOperations();
Original file line number Diff line number Diff line change
@@ -0,0 +1,82 @@
import type { Page } from '@playwright/test';
import { expect } from '@playwright/test';
import type { Event } from '@sentry/core';

import { sentryTest } from '../../../../utils/fixtures';
import { getFirstSentryEnvelopeRequest, getMultipleSentryEnvelopeRequests, shouldSkipTracingTest } from '../../../../utils/helpers';

async function mockSupabaseRoute(page: Page) {
await page.route('**/rest/v1/todos**', route => {
return route.fulfill({
status: 200,
body: JSON.stringify({
userNames: ['John', 'Jane'],
}),
headers: {
'Content-Type': 'application/json',
},
});
});
}

sentryTest('should capture Supabase database operation breadcrumbs', async ({ getLocalTestUrl, page }) => {
if (shouldSkipTracingTest()) {
return;
}

await mockSupabaseRoute(page);

const url = await getLocalTestUrl({ testDir: __dirname });

const eventData = await getFirstSentryEnvelopeRequest<Event>(page, url);

expect(eventData.breadcrumbs).toBeDefined();
expect(eventData.breadcrumbs).toContainEqual({
timestamp: expect.any(Number),
type: 'supabase',
category: 'db.insert',
message: 'from(todos)',
data: expect.any(Object),
});
});

sentryTest('should capture multiple Supabase operations in sequence', async ({ getLocalTestUrl, page }) => {
if (shouldSkipTracingTest()) {
return;
}

await mockSupabaseRoute(page);

const url = await getLocalTestUrl({ testDir: __dirname });

const events = await getMultipleSentryEnvelopeRequests<Event>(page, 2, { url });

expect(events).toHaveLength(2);

events.forEach(event => {
expect(
event.breadcrumbs?.some(breadcrumb => breadcrumb.type === 'supabase' && breadcrumb?.category?.startsWith('db.')),
).toBe(true);
});
});

sentryTest('should include correct data payload in Supabase breadcrumbs', async ({ getLocalTestUrl, page }) => {
if (shouldSkipTracingTest()) {
return;
}

await mockSupabaseRoute(page);

const url = await getLocalTestUrl({ testDir: __dirname });

const eventData = await getFirstSentryEnvelopeRequest<Event>(page, url);

const supabaseBreadcrumb = eventData.breadcrumbs?.find(b => b.type === 'supabase');

expect(supabaseBreadcrumb).toBeDefined();
expect(supabaseBreadcrumb?.data).toMatchObject({
query: expect.arrayContaining([
'filter(columns, )'
]),
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.

# dependencies
/node_modules
/.pnp
.pnp.js

# testing
/coverage

# next.js
/.next/
/out/

# production
/build

# misc
.DS_Store
*.pem

# debug
npm-debug.log*
yarn-debug.log*
yarn-error.log*
.pnpm-debug.log*

# local env files
.env*.local

# vercel
.vercel

# typescript
*.tsbuildinfo
next-env.d.ts

# Sentry Config File
.env.sentry-build-plugin
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
@sentry:registry=http://127.0.0.1:4873
@sentry-internal:registry=http://127.0.0.1:4873
Loading
Loading