mirror of
https://github.com/ruvnet/RuVector.git
synced 2026-05-30 12:13:34 +00:00
Major additions: - Complete Next.js studio application with 1600+ components - Docker support (Dockerfile.combined, docker-compose.yml) - GCP deployment documentation and benchmarks - SQL benchmark scripts for performance testing - Sentry integration for monitoring - Comprehensive test suite and mocks Studio features: - Dashboard and admin interfaces - Data visualization components - Authentication and user management - API integration with RuVector backend - Static data and public assets 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
372 lines
10 KiB
TypeScript
372 lines
10 KiB
TypeScript
import { Clock5, Layers, Timer, Vault, Webhook } from 'lucide-react'
|
|
import dynamic from 'next/dynamic'
|
|
import Image from 'next/image'
|
|
import { ComponentType, ReactNode } from 'react'
|
|
|
|
import { GenericSkeletonLoader } from 'components/ui/ShimmeringLoader'
|
|
import { BASE_PATH, DOCS_URL } from 'lib/constants'
|
|
import { cn } from 'ui'
|
|
import { UpgradeDatabaseAlert } from '../Queues/UpgradeDatabaseAlert'
|
|
import { WRAPPERS } from '../Wrappers/Wrappers.constants'
|
|
import { WrapperMeta } from '../Wrappers/Wrappers.types'
|
|
|
|
export type Navigation = {
|
|
route: string
|
|
label: string
|
|
hasChild?: boolean
|
|
childIcon?: React.ReactNode
|
|
children?: Navigation[]
|
|
}
|
|
|
|
const Loading = () => (
|
|
<div className="p-10">
|
|
<GenericSkeletonLoader />
|
|
</div>
|
|
)
|
|
|
|
export type IntegrationDefinition = {
|
|
id: string
|
|
name: string
|
|
status?: 'alpha' | 'beta'
|
|
icon: (props?: { className?: string; style?: Record<string, any> }) => ReactNode
|
|
description: string
|
|
docsUrl: string
|
|
author: {
|
|
name: string
|
|
websiteUrl: string
|
|
}
|
|
requiredExtensions: string[]
|
|
/** Optional component to render if the integration requires extensions that are not available on the current database image */
|
|
missingExtensionsAlert?: ReactNode
|
|
navigation?: Navigation[]
|
|
navigate: (
|
|
id: string,
|
|
pageId: string | undefined,
|
|
childId: string | undefined
|
|
) => ComponentType<{}> | null
|
|
} & ({ type: 'wrapper'; meta: WrapperMeta } | { type: 'postgres_extension' } | { type: 'custom' })
|
|
|
|
const authorSupabase = {
|
|
name: 'Supabase',
|
|
websiteUrl: 'https://supabase.com',
|
|
}
|
|
|
|
const supabaseIntegrations: IntegrationDefinition[] = [
|
|
{
|
|
id: 'queues',
|
|
type: 'postgres_extension' as const,
|
|
requiredExtensions: ['pgmq'],
|
|
missingExtensionsAlert: <UpgradeDatabaseAlert minimumVersion="15.6.1.143" />,
|
|
name: `Queues`,
|
|
icon: ({ className, ...props } = {}) => (
|
|
<Layers className={cn('inset-0 p-2 text-black w-full h-full', className)} {...props} />
|
|
),
|
|
description: 'Lightweight message queue in Postgres',
|
|
docsUrl: 'https://github.com/tembo-io/pgmq',
|
|
author: {
|
|
name: 'pgmq',
|
|
websiteUrl: 'https://github.com/tembo-io/pgmq',
|
|
},
|
|
navigation: [
|
|
{
|
|
route: 'overview',
|
|
label: 'Overview',
|
|
},
|
|
{
|
|
route: 'queues',
|
|
label: 'Queues',
|
|
hasChild: true,
|
|
childIcon: (
|
|
<Layers size={12} strokeWidth={1.5} className={cn('text-foreground w-full h-full')} />
|
|
),
|
|
},
|
|
{
|
|
route: 'settings',
|
|
label: 'Settings',
|
|
},
|
|
],
|
|
navigate: (id: string, pageId: string = 'overview', childId: string | undefined) => {
|
|
if (childId) {
|
|
return dynamic(() => import('../Queues/QueuePage').then((mod) => mod.QueuePage), {
|
|
loading: Loading,
|
|
})
|
|
}
|
|
switch (pageId) {
|
|
case 'overview':
|
|
return dynamic(
|
|
() =>
|
|
import('components/interfaces/Integrations/Queues/OverviewTab').then(
|
|
(mod) => mod.QueuesOverviewTab
|
|
),
|
|
{ loading: Loading }
|
|
)
|
|
case 'queues':
|
|
return dynamic(() => import('../Queues/QueuesTab').then((mod) => mod.QueuesTab), {
|
|
loading: Loading,
|
|
})
|
|
case 'settings':
|
|
return dynamic(
|
|
() => import('../Queues/QueuesSettings').then((mod) => mod.QueuesSettings),
|
|
{ loading: Loading }
|
|
)
|
|
}
|
|
return null
|
|
},
|
|
},
|
|
{
|
|
id: 'cron',
|
|
type: 'postgres_extension' as const,
|
|
requiredExtensions: ['pg_cron'],
|
|
name: `Cron`,
|
|
icon: ({ className, ...props } = {}) => (
|
|
<Clock5 className={cn('inset-0 p-2 text-black w-full h-full', className)} {...props} />
|
|
),
|
|
description: 'Schedule recurring Jobs in Postgres',
|
|
docsUrl: 'https://github.com/citusdata/pg_cron',
|
|
author: {
|
|
name: 'Citus Data',
|
|
websiteUrl: 'https://github.com/citusdata/pg_cron',
|
|
},
|
|
navigation: [
|
|
{
|
|
route: 'overview',
|
|
label: 'Overview',
|
|
},
|
|
{
|
|
route: 'jobs',
|
|
label: 'Jobs',
|
|
hasChild: true,
|
|
childIcon: (
|
|
<Timer size={12} strokeWidth={1.5} className={cn('text-foreground w-full h-full')} />
|
|
),
|
|
},
|
|
],
|
|
navigate: (id: string, pageId: string = 'overview', childId: string | undefined) => {
|
|
if (childId) {
|
|
return dynamic(() => import('../CronJobs/CronJobPage').then((mod) => mod.CronJobPage), {
|
|
loading: Loading,
|
|
})
|
|
}
|
|
switch (pageId) {
|
|
case 'overview':
|
|
return dynamic(
|
|
() =>
|
|
import('components/interfaces/Integrations/Integration/IntegrationOverviewTab').then(
|
|
(mod) => mod.IntegrationOverviewTab
|
|
),
|
|
{
|
|
loading: Loading,
|
|
}
|
|
)
|
|
case 'jobs':
|
|
return dynamic(() => import('../CronJobs/CronJobsTab').then((mod) => mod.CronjobsTab), {
|
|
loading: Loading,
|
|
})
|
|
}
|
|
return null
|
|
},
|
|
},
|
|
{
|
|
id: 'vault',
|
|
type: 'postgres_extension' as const,
|
|
requiredExtensions: ['supabase_vault'],
|
|
missingExtensionsAlert: <UpgradeDatabaseAlert />,
|
|
name: `Vault`,
|
|
status: 'beta',
|
|
icon: ({ className, ...props } = {}) => (
|
|
<Vault className={cn('inset-0 p-2 text-black w-full h-full', className)} {...props} />
|
|
),
|
|
description: 'Application level encryption for your project',
|
|
docsUrl: DOCS_URL,
|
|
author: authorSupabase,
|
|
navigation: [
|
|
{
|
|
route: 'overview',
|
|
label: 'Overview',
|
|
},
|
|
{
|
|
route: 'secrets',
|
|
label: 'Secrets',
|
|
},
|
|
],
|
|
navigate: (id: string, pageId: string = 'overview', childId: string | undefined) => {
|
|
switch (pageId) {
|
|
case 'overview':
|
|
return dynamic(
|
|
() =>
|
|
import('components/interfaces/Integrations/Integration/IntegrationOverviewTab').then(
|
|
(mod) => mod.IntegrationOverviewTab
|
|
),
|
|
{
|
|
loading: Loading,
|
|
}
|
|
)
|
|
case 'secrets':
|
|
return dynamic(
|
|
() => import('../Vault/Secrets/SecretsManagement').then((mod) => mod.SecretsManagement),
|
|
{
|
|
loading: Loading,
|
|
}
|
|
)
|
|
}
|
|
return null
|
|
},
|
|
},
|
|
{
|
|
id: 'webhooks',
|
|
type: 'postgres_extension' as const,
|
|
name: `Database Webhooks`,
|
|
icon: ({ className, ...props } = {}) => (
|
|
<Webhook className={cn('inset-0 p-2 text-black w-full h-full', className)} {...props} />
|
|
),
|
|
description:
|
|
'Send real-time data from your database to another system when a table event occurs',
|
|
docsUrl: DOCS_URL,
|
|
author: authorSupabase,
|
|
requiredExtensions: [],
|
|
navigation: [
|
|
{
|
|
route: 'overview',
|
|
label: 'Overview',
|
|
},
|
|
{
|
|
route: 'webhooks',
|
|
label: 'Webhooks',
|
|
},
|
|
],
|
|
navigate: (id: string, pageId: string = 'overview', childId: string | undefined) => {
|
|
switch (pageId) {
|
|
case 'overview':
|
|
return dynamic(
|
|
() =>
|
|
import('components/interfaces/Integrations/Webhooks/OverviewTab').then(
|
|
(mod) => mod.WebhooksOverviewTab
|
|
),
|
|
{
|
|
loading: Loading,
|
|
}
|
|
)
|
|
case 'webhooks':
|
|
return dynamic(
|
|
() =>
|
|
import('components/interfaces/Integrations/Webhooks/ListTab').then(
|
|
(mod) => mod.WebhooksListTab
|
|
),
|
|
{
|
|
loading: Loading,
|
|
}
|
|
)
|
|
}
|
|
return null
|
|
},
|
|
},
|
|
{
|
|
id: 'graphiql',
|
|
type: 'postgres_extension' as const,
|
|
requiredExtensions: ['pg_graphql'],
|
|
name: `GraphQL`,
|
|
icon: ({ className, ...props } = {}) => (
|
|
<Image
|
|
fill
|
|
src={`${BASE_PATH}/img/graphql.svg`}
|
|
alt="GraphiQL"
|
|
className={cn('p-2', className)}
|
|
{...props}
|
|
/>
|
|
),
|
|
description: 'Run GraphQL queries through our interactive in-browser IDE',
|
|
docsUrl: DOCS_URL,
|
|
author: authorSupabase,
|
|
navigation: [
|
|
{
|
|
route: 'overview',
|
|
label: 'Overview',
|
|
},
|
|
{
|
|
route: 'graphiql',
|
|
label: 'GraphiQL',
|
|
},
|
|
],
|
|
navigate: (id: string, pageId: string = 'overview', childId: string | undefined) => {
|
|
switch (pageId) {
|
|
case 'overview':
|
|
return dynamic(
|
|
() =>
|
|
import('components/interfaces/Integrations/Integration/IntegrationOverviewTab').then(
|
|
(mod) => mod.IntegrationOverviewTab
|
|
),
|
|
{
|
|
loading: Loading,
|
|
}
|
|
)
|
|
case 'graphiql':
|
|
return dynamic(
|
|
() =>
|
|
import('components/interfaces/Integrations/GraphQL/GraphiQLTab').then(
|
|
(mod) => mod.GraphiQLTab
|
|
),
|
|
{
|
|
loading: Loading,
|
|
}
|
|
)
|
|
}
|
|
return null
|
|
},
|
|
},
|
|
] as const
|
|
|
|
const wrapperIntegrations: IntegrationDefinition[] = WRAPPERS.map((w) => {
|
|
return {
|
|
id: w.name,
|
|
type: 'wrapper' as const,
|
|
name: `${w.label} Wrapper`,
|
|
icon: ({ className, ...props } = {}) => (
|
|
<Image fill src={w.icon} alt={w.name} className={cn('p-2', className)} {...props} />
|
|
),
|
|
requiredExtensions: ['wrappers', 'supabase_vault'],
|
|
description: w.description,
|
|
docsUrl: w.docsUrl,
|
|
meta: w,
|
|
author: authorSupabase,
|
|
navigation: [
|
|
{
|
|
route: 'overview',
|
|
label: 'Overview',
|
|
},
|
|
{
|
|
route: 'wrappers',
|
|
label: 'Wrappers',
|
|
},
|
|
],
|
|
navigate: (id: string, pageId: string = 'overview', childId: string | undefined) => {
|
|
switch (pageId) {
|
|
case 'overview':
|
|
return dynamic(
|
|
() =>
|
|
import('components/interfaces/Integrations/Wrappers/OverviewTab').then(
|
|
(mod) => mod.WrapperOverviewTab
|
|
),
|
|
{
|
|
loading: Loading,
|
|
}
|
|
)
|
|
case 'wrappers':
|
|
return dynamic(
|
|
() =>
|
|
import('components/interfaces/Integrations/Wrappers/WrappersTab').then(
|
|
(mod) => mod.WrappersTab
|
|
),
|
|
{
|
|
loading: Loading,
|
|
}
|
|
)
|
|
}
|
|
return null
|
|
},
|
|
}
|
|
})
|
|
|
|
export const INTEGRATIONS: IntegrationDefinition[] = [
|
|
...wrapperIntegrations,
|
|
...supabaseIntegrations,
|
|
]
|