From 66f2adcb505270742c7f7967d57581d333f82783 Mon Sep 17 00:00:00 2001 From: Sam J Combs Date: Fri, 29 Nov 2024 12:28:53 -0500 Subject: [PATCH 1/2] feat: support for anonymous Apollo key usage --- backend/src/routes/apolloApiKey.ts | 35 ++++++------- backend/src/server.ts | 15 ++++-- frontend/package-lock.json | 4 +- frontend/src/components/ui/home.tsx | 17 ++++++- frontend/src/components/ui/settings-badge.tsx | 51 +++++++++++++++++++ frontend/src/components/ui/settings.tsx | 3 ++ frontend/tailwind.config.js | 6 ++- 7 files changed, 103 insertions(+), 28 deletions(-) create mode 100644 frontend/src/components/ui/settings-badge.tsx diff --git a/backend/src/routes/apolloApiKey.ts b/backend/src/routes/apolloApiKey.ts index e76a6b6..bf7b080 100644 --- a/backend/src/routes/apolloApiKey.ts +++ b/backend/src/routes/apolloApiKey.ts @@ -1,19 +1,22 @@ -import express, {Response} from 'express'; +import express, {Response, Request} from 'express'; import {DI} from '../server'; import {logger} from '../utilities/logger'; import {ApolloApiKey} from '../models/apolloApiKey'; import {SessionRequest} from 'supertokens-node/framework/express'; +import config from '../config/config'; const router = express.Router(); -router.get('/apollo-api-key', async (req: SessionRequest, res: Response) => { - try { - const userId = req.session?.getUserId(); - if (!userId) { - return res.status(401).json({error: 'Unauthorized'}); - } +const handleRequest = async (req: Request | SessionRequest, defaultUserId = 'anonymous') => { + const userId = (req as SessionRequest).session?.getUserId() || defaultUserId; + return userId; +}; +router.get('/apollo-api-key', async (req: Request | SessionRequest, res: Response) => { + try { + const userId = await handleRequest(req); const apiKey = await DI.apolloApiKeys.findOne({userId}); + if (apiKey) { const decryptedKey = apiKey.getDecryptedKey(); const obfuscatedKey = `${decryptedKey.slice(0, 4)}****${decryptedKey.slice(-4)}`; @@ -27,14 +30,11 @@ router.get('/apollo-api-key', async (req: SessionRequest, res: Response) => { } }); -router.post('/apollo-api-key', async (req: SessionRequest, res: Response) => { +router.post('/apollo-api-key', async (req: Request | SessionRequest, res: Response) => { try { - const userId = req.session?.getUserId(); - if (!userId) { - return res.status(401).json({error: 'Unauthorized'}); - } - + const userId = await handleRequest(req); const {key} = req.body; + if (!key) { return res.status(400).json({error: 'API key is required'}); } @@ -59,14 +59,11 @@ router.post('/apollo-api-key', async (req: SessionRequest, res: Response) => { } }); -router.delete('/apollo-api-key', async (req: SessionRequest, res: Response) => { +router.delete('/apollo-api-key', async (req: Request | SessionRequest, res: Response) => { try { - const userId = req.session?.getUserId(); - if (!userId) { - return res.status(401).json({error: 'Unauthorized'}); - } - + const userId = await handleRequest(req); const apiKey = await DI.apolloApiKeys.findOne({userId}); + if (!apiKey) { return res.status(404).json({error: 'API key not found'}); } diff --git a/backend/src/server.ts b/backend/src/server.ts index 871081d..d036c34 100644 --- a/backend/src/server.ts +++ b/backend/src/server.ts @@ -1,6 +1,11 @@ import './loadEnv'; import 'reflect-metadata'; -import {EntityManager, EntityRepository, MikroORM, RequestContext,} from '@mikro-orm/core'; +import { + EntityManager, + EntityRepository, + MikroORM, + RequestContext, +} from '@mikro-orm/core'; import cors from 'cors'; import express from 'express'; import path from 'path'; @@ -80,7 +85,7 @@ const initializeApp = async () => { const mikroOrmConfig = { ...(await import( `./mikro-orm.${process.env.MIKRO_ORM_DRIVER || 'sqlite'}${isTypescript ? '.ts' : '.js'}` - ).then((module) => module.default)), + ).then((module) => module.default)), }; DI.orm = await MikroORM.init(mikroOrmConfig); @@ -117,8 +122,10 @@ const initializeApp = async () => { cors({ origin: (origin, callback) => { const allowedOrigins = [getWebsiteDomain()]; - const regex = /^(https:\/\/[a-zA-Z0-9-]+\.narrative\.tech|https?:\/\/localhost(:\d+)?)$/; - if (!origin || allowedOrigins.includes(origin) || regex.test(origin)) callback(null, true); + const regex = + /^(https:\/\/[a-zA-Z0-9-]+\.narrative\.tech|https?:\/\/localhost(:\d+)?)$/; + if (!origin || allowedOrigins.includes(origin) || regex.test(origin)) + callback(null, true); else callback(new Error('Not allowed by CORS')); }, allowedHeaders: [ diff --git a/frontend/package-lock.json b/frontend/package-lock.json index 43a62b6..ec55a24 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -1,12 +1,12 @@ { "name": "frontend", - "version": "1.0.0-beta.2", + "version": "1.0.0-beta.3", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "frontend", - "version": "1.0.0-beta.2", + "version": "1.0.0-beta.3", "dependencies": { "@apollo/explorer": "^3.7.0", "@apollo/sandbox": "^2.6.0", diff --git a/frontend/src/components/ui/home.tsx b/frontend/src/components/ui/home.tsx index a08237c..1e186c8 100644 --- a/frontend/src/components/ui/home.tsx +++ b/frontend/src/components/ui/home.tsx @@ -88,6 +88,8 @@ import {Tabs, TabsContent, TabsList, TabsTrigger} from './tabs'; import {Textarea} from './textarea'; import {Toaster} from './toaster'; import {toast} from './use-toast'; +import {Badge} from './badge'; +import {SettingsBadge} from './settings-badge'; const Home = () => { const navigate = useNavigate(); @@ -121,6 +123,17 @@ const Home = () => { const serverBaseUrl = getApiBaseUrl(); + const [hasApolloKey, setHasApolloKey] = useState(true); + + useEffect(() => { + fetch(`${serverBaseUrl}/api/apollo-api-key`) + .then((response) => response.json()) + .then((data) => { + setHasApolloKey(!!data.key); + }) + .catch(() => setHasApolloKey(false)); + }, [serverBaseUrl]); + const handleSettingsClick = () => navigate('/settings'); const handleCreateSeedClick = () => { populateSeedForm(); @@ -527,8 +540,8 @@ const Home = () => {
- diff --git a/frontend/src/components/ui/settings-badge.tsx b/frontend/src/components/ui/settings-badge.tsx new file mode 100644 index 0000000..9149a74 --- /dev/null +++ b/frontend/src/components/ui/settings-badge.tsx @@ -0,0 +1,51 @@ +import {cn} from '../../lib/utils'; +import { + Tooltip, + TooltipContent, + TooltipProvider, + TooltipTrigger, +} from './tooltip'; +import {Settings} from 'lucide-react'; + +interface SettingsBadgeProps { + hasApolloKey: boolean; + onClick: () => void; +} + +export function SettingsBadge({hasApolloKey, onClick}: SettingsBadgeProps) { + return ( + + + +
+ + {!hasApolloKey && ( + + )} +
+
+ +

+ Connect to Apollo Studio + + Enter your user-scoped Apollo API key to instantly build custom + mock seeds against existing supergraph variants and schema + proposals + +

+
+
+
+ ); +} diff --git a/frontend/src/components/ui/settings.tsx b/frontend/src/components/ui/settings.tsx index 55a3c5b..137e287 100644 --- a/frontend/src/components/ui/settings.tsx +++ b/frontend/src/components/ui/settings.tsx @@ -14,6 +14,7 @@ import { } from './card'; import {Input} from './input'; import {useToast} from './use-toast'; +import {useNavigate} from 'react-router-dom'; interface ApiKey { id: string; @@ -28,6 +29,7 @@ export default function SettingsPage() { const [isLoading, setIsLoading] = useState(false); const {toast} = useToast(); const serverBaseUrl = getApiBaseUrl(); + const navigate = useNavigate(); useEffect(() => { fetchApiKey(); @@ -65,6 +67,7 @@ export default function SettingsPage() { title: 'Apollo API Key Saved', description: 'Your Apollo API key has been saved successfully.', }); + navigate('/'); } }); }; diff --git a/frontend/tailwind.config.js b/frontend/tailwind.config.js index 47739fa..551bcd1 100644 --- a/frontend/tailwind.config.js +++ b/frontend/tailwind.config.js @@ -82,7 +82,11 @@ module.exports = { to: { height: '0' } - } + }, + pulse: { + '0%, 100%': { opacity: '0.7', transform: 'scale(1)' }, + '50%': { opacity: '0.3', transform: 'scale(0.95)' }, + }, }, animation: { 'accordion-down': 'accordion-down 0.2s ease-out', From 5f4bebe38d90d027e62fd419d6dc5c5296bdba4b Mon Sep 17 00:00:00 2001 From: Michal Kawka Date: Mon, 2 Dec 2024 12:47:46 +0100 Subject: [PATCH 2/2] es lint --- backend/src/routes/apolloApiKey.ts | 131 ++++++++++++++++------------- 1 file changed, 71 insertions(+), 60 deletions(-) diff --git a/backend/src/routes/apolloApiKey.ts b/backend/src/routes/apolloApiKey.ts index bf7b080..3cd8c97 100644 --- a/backend/src/routes/apolloApiKey.ts +++ b/backend/src/routes/apolloApiKey.ts @@ -1,80 +1,91 @@ -import express, {Response, Request} from 'express'; +import express, {Request, Response} from 'express'; +import {SessionRequest} from 'supertokens-node/framework/express'; +import {ApolloApiKey} from '../models/apolloApiKey'; import {DI} from '../server'; import {logger} from '../utilities/logger'; -import {ApolloApiKey} from '../models/apolloApiKey'; -import {SessionRequest} from 'supertokens-node/framework/express'; -import config from '../config/config'; const router = express.Router(); -const handleRequest = async (req: Request | SessionRequest, defaultUserId = 'anonymous') => { +const handleRequest = async ( + req: Request | SessionRequest, + defaultUserId = 'anonymous' +) => { const userId = (req as SessionRequest).session?.getUserId() || defaultUserId; return userId; }; -router.get('/apollo-api-key', async (req: Request | SessionRequest, res: Response) => { - try { - const userId = await handleRequest(req); - const apiKey = await DI.apolloApiKeys.findOne({userId}); - - if (apiKey) { - const decryptedKey = apiKey.getDecryptedKey(); - const obfuscatedKey = `${decryptedKey.slice(0, 4)}****${decryptedKey.slice(-4)}`; - res.json({key: obfuscatedKey}); - } else { - res.json({key: null}); +router.get( + '/apollo-api-key', + async (req: Request | SessionRequest, res: Response) => { + try { + const userId = await handleRequest(req); + const apiKey = await DI.apolloApiKeys.findOne({userId}); + + if (apiKey) { + const decryptedKey = apiKey.getDecryptedKey(); + const obfuscatedKey = `${decryptedKey.slice(0, 4)}****${decryptedKey.slice(-4)}`; + res.json({key: obfuscatedKey}); + } else { + res.json({key: null}); + } + } catch (error) { + logger.error('Failed to fetch API key', {error}); + res.status(500).json({error: 'Failed to fetch API key'}); } - } catch (error) { - logger.error('Failed to fetch API key', {error}); - res.status(500).json({error: 'Failed to fetch API key'}); } -}); +); -router.post('/apollo-api-key', async (req: Request | SessionRequest, res: Response) => { - try { - const userId = await handleRequest(req); - const {key} = req.body; - - if (!key) { - return res.status(400).json({error: 'API key is required'}); - } +router.post( + '/apollo-api-key', + async (req: Request | SessionRequest, res: Response) => { + try { + const userId = await handleRequest(req); + const {key} = req.body; - let apiKey = await DI.apolloApiKeys.findOne({userId}); - if (apiKey) { - const newApiKey = new ApolloApiKey(key, userId); - apiKey.encryptedKey = newApiKey.encryptedKey; - apiKey.iv = newApiKey.iv; - apiKey.tag = newApiKey.tag; - } else { - apiKey = new ApolloApiKey(key, userId); - DI.em.persist(apiKey); - } + if (!key) { + return res.status(400).json({error: 'API key is required'}); + } - await DI.em.flush(); - await DI.apolloClient.updateApiKey(key, userId); - res.status(200).json({message: 'API key saved successfully'}); - } catch (error) { - logger.error('Failed to save API key', {error}); - res.status(500).json({error: 'Failed to save API key'}); - } -}); + let apiKey = await DI.apolloApiKeys.findOne({userId}); + if (apiKey) { + const newApiKey = new ApolloApiKey(key, userId); + apiKey.encryptedKey = newApiKey.encryptedKey; + apiKey.iv = newApiKey.iv; + apiKey.tag = newApiKey.tag; + } else { + apiKey = new ApolloApiKey(key, userId); + DI.em.persist(apiKey); + } -router.delete('/apollo-api-key', async (req: Request | SessionRequest, res: Response) => { - try { - const userId = await handleRequest(req); - const apiKey = await DI.apolloApiKeys.findOne({userId}); - - if (!apiKey) { - return res.status(404).json({error: 'API key not found'}); + await DI.em.flush(); + await DI.apolloClient.updateApiKey(key, userId); + res.status(200).json({message: 'API key saved successfully'}); + } catch (error) { + logger.error('Failed to save API key', {error}); + res.status(500).json({error: 'Failed to save API key'}); } + } +); - await DI.em.remove(apiKey).flush(); - await DI.apolloClient.updateApiKey('', userId); - res.status(200).json({message: 'API key deleted successfully'}); - } catch (error) { - logger.error('Failed to delete API key', {error}); - res.status(500).json({error: 'Failed to delete API key'}); +router.delete( + '/apollo-api-key', + async (req: Request | SessionRequest, res: Response) => { + try { + const userId = await handleRequest(req); + const apiKey = await DI.apolloApiKeys.findOne({userId}); + + if (!apiKey) { + return res.status(404).json({error: 'API key not found'}); + } + + await DI.em.remove(apiKey).flush(); + await DI.apolloClient.updateApiKey('', userId); + res.status(200).json({message: 'API key deleted successfully'}); + } catch (error) { + logger.error('Failed to delete API key', {error}); + res.status(500).json({error: 'Failed to delete API key'}); + } } -}); +); export default router;