Skip to content

Commit 7336f6e

Browse files
feat: Envelope encryption for Provider API Keys + Modal (#202)
* chore(server): track empty temp folder * chore(local-kms): add seed yaml * chore(docker-compose): add local-kms * chore: add aws kms sdk client * feat(server): adjust prisma schema to include encrypted data and key * feat(sever): add encryption module and service * feaet(server): encrypt provider API keys * feat(console): ui to consume new encrypted API contract * feat(server): prompt tester uses provider api key * chore(server): add KMS_LOCAL env variable * chore(console): remove unused imports * feat(console): required API key modal for a particular provider * fix(server): formatting * feat(console): fix wording * fix(console): formatting * feat(server): db migration for encrypted provider api keys * fix(server): broken ProviderApiKey migration
1 parent 8c1f691 commit 7336f6e

31 files changed

+1434
-747
lines changed

.gitignore

Lines changed: 3 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@
22

33
# compiled output
44
dist
5-
tmp
65
/out-tsc
76

87
# dependencies
@@ -48,4 +47,6 @@ Thumbs.db
4847
schema.graphql
4948

5049
# temp
51-
temp
50+
temp
51+
52+
kms/data

apps/console/src/app/app.tsx

Lines changed: 6 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@ import { RequestsPage } from "./pages/requests/RequestsPage";
2828
import { DashboardPage } from "./pages/projects/overview/DashboardPage";
2929
import { LoginPage } from "./pages/auth/LoginPage";
3030
import { AuthCallbackPage } from "./pages/auth/AuthCallbackPage";
31+
import { RequiredProviderApiKeyModalProvider } from "./lib/providers/RequiredProviderApiKeyModalProvider";
3132

3233
initSuperTokens();
3334

@@ -142,9 +143,11 @@ export function App() {
142143
path={paths["/projects/:projectId"]}
143144
element={
144145
<CurrentPromptProvider>
145-
<LayoutWrapper withSideNav={true}>
146-
<Outlet />
147-
</LayoutWrapper>
146+
<RequiredProviderApiKeyModalProvider>
147+
<LayoutWrapper withSideNav={true}>
148+
<Outlet />
149+
</LayoutWrapper>
150+
</RequiredProviderApiKeyModalProvider>
148151
</CurrentPromptProvider>
149152
}
150153
>
Lines changed: 68 additions & 54 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,13 @@
1-
import { Avatar, Card, Row, Col, Typography, Button, Input } from "antd";
2-
import styled from "@emotion/styled";
1+
import {
2+
Avatar,
3+
Card,
4+
Row,
5+
Col,
6+
Typography,
7+
Button,
8+
Input,
9+
message,
10+
} from "antd";
311
import { CloseOutlined, EditOutlined, SaveOutlined } from "@ant-design/icons";
412
import { useState } from "react";
513
import { useMutation } from "@tanstack/react-query";
@@ -8,24 +16,25 @@ import { gqlClient, queryClient } from "../../lib/graphql";
816
import { CreateProviderApiKeyInput } from "../../../@generated/graphql/graphql";
917
import { useEffect } from "react";
1018
import { useCurrentOrganization } from "../../lib/hooks/useCurrentOrganization";
11-
12-
const APIKeyContainer = styled.div`
13-
display: flex;
14-
align-items: center;
15-
width: 600px;
16-
`;
19+
import { trackEvent } from "../../lib/utils/analytics";
20+
import { providersList } from "./providers-list";
1721

1822
interface Props {
1923
provider: string;
2024
value: string | null;
21-
iconBase64: string;
25+
onSave?: () => void;
26+
initialIsEditing?: boolean;
27+
canCancelEdit?: boolean;
2228
}
2329

2430
export const ProviderApiKeyListItem = ({
2531
provider,
2632
value,
27-
iconBase64,
33+
onSave,
34+
initialIsEditing = false,
35+
canCancelEdit = true,
2836
}: Props) => {
37+
const [messageApi, contextHolder] = message.useMessage();
2938
const { currentOrgId } = useCurrentOrganization();
3039
const updateKeyMutation = useMutation({
3140
mutationFn: (data: CreateProviderApiKeyInput) =>
@@ -38,10 +47,11 @@ export const ProviderApiKeyListItem = ({
3847
}),
3948
onSuccess: () => {
4049
queryClient.invalidateQueries({ queryKey: ["providerApiKeys"] });
50+
onSave && onSave();
4151
},
4252
});
4353

44-
const [isEditing, setIsEditing] = useState(false);
54+
const [isEditing, setIsEditing] = useState(initialIsEditing);
4555
const [editValue, setEditValue] = useState<string>("");
4656

4757
useEffect(() => {
@@ -60,62 +70,66 @@ export const ProviderApiKeyListItem = ({
6070
value: editValue,
6171
organizationId: currentOrgId,
6272
});
73+
74+
messageApi.success("API key saved successfully");
75+
trackEvent("provider_api_key_set", { provider });
6376
setIsEditing(false);
6477
};
6578

79+
const iconBase64 = providersList.find(
80+
(item) => item.provider === provider
81+
).iconBase64;
82+
6683
return (
6784
<Card size="small" key={provider}>
68-
<APIKeyContainer>
69-
<Row gutter={[12, 12]} align="middle" style={{ width: "100%" }}>
70-
<Col
71-
style={{ display: "flex", alignItems: "center", marginRight: 20 }}
72-
>
73-
<Avatar size="large" shape="square" src={iconBase64} />
74-
<Typography.Text style={{ fontSize: 18, marginLeft: 10 }}>
75-
{provider}
85+
{contextHolder}
86+
<Row gutter={[12, 12]} align="middle" style={{ width: "100%" }}>
87+
<Col style={{ display: "flex", alignItems: "center", marginRight: 20 }}>
88+
<Avatar size="large" shape="square" src={iconBase64} />
89+
<Typography.Text style={{ fontSize: 18, marginLeft: 10 }}>
90+
{provider}
91+
</Typography.Text>
92+
</Col>
93+
<Col
94+
flex="auto"
95+
style={{ display: "flex", justifyContent: "flex-end" }}
96+
>
97+
{isEditing ? (
98+
<Input
99+
placeholder="Paste your API key"
100+
onChange={(e) => setEditValue(e.target.value)}
101+
autoComplete="off"
102+
/>
103+
) : (
104+
<Typography.Text style={{ marginLeft: 10, opacity: 0.5 }}>
105+
{value || "No API key provided"}
76106
</Typography.Text>
77-
</Col>
78-
<Col
79-
flex="auto"
80-
style={{ display: "flex", justifyContent: "flex-end" }}
81-
>
82-
{isEditing ? (
83-
<Input
84-
placeholder="Paste your API key"
85-
onChange={(e) => setEditValue(e.target.value)}
86-
autoComplete="off"
107+
)}
108+
</Col>
109+
<Col style={{ display: "flex", justifyContent: "flex-end" }}>
110+
{isEditing ? (
111+
<>
112+
<Button
113+
type="primary"
114+
onClick={handleSave}
115+
loading={updateKeyMutation.isLoading}
116+
icon={<SaveOutlined height={18} />}
87117
/>
88-
) : (
89-
<Typography.Text style={{ marginLeft: 10, opacity: 0.5 }}>
90-
{value || "No API key provided"}
91-
</Typography.Text>
92-
)}
93-
</Col>
94-
<Col style={{ display: "flex", justifyContent: "flex-end" }}>
95-
{isEditing ? (
96-
<>
97-
<Button
98-
type="primary"
99-
onClick={handleSave}
100-
loading={updateKeyMutation.isLoading}
101-
icon={<SaveOutlined height={18} />}
102-
/>
118+
119+
{canCancelEdit && (
103120
<Button
104121
onClick={() => setIsEditing(false)}
105122
loading={updateKeyMutation.isLoading}
106123
icon={<CloseOutlined />}
107124
style={{ marginLeft: 10 }}
108125
/>
109-
</>
110-
) : (
111-
<Button
112-
onClick={handleEdit}
113-
icon={<EditOutlined height={18} />}
114-
/>
115-
)}
116-
</Col>
117-
</Row>
118-
</APIKeyContainer>
126+
)}
127+
</>
128+
) : (
129+
<Button onClick={handleEdit} icon={<EditOutlined height={18} />} />
130+
)}
131+
</Col>
132+
</Row>
119133
</Card>
120134
);
121135
};
Lines changed: 34 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
import { Space } from "antd";
2+
import { ProviderApiKeyListItem } from "./ProviderApiKeyListItem";
3+
import { useProviderApiKeys } from "../../graphql/hooks/queries";
4+
import { providersList } from "./providers-list";
5+
6+
export const ProviderApiKeysList = () => {
7+
const { providerApiKeys } = useProviderApiKeys();
8+
9+
const renderProviderApiKey = (provider) => {
10+
const apiKey = providerApiKeys.find(
11+
(key) => key.provider === provider.provider
12+
);
13+
14+
const value = apiKey?.censoredValue
15+
? `**********${apiKey?.censoredValue}`
16+
: null;
17+
18+
return (
19+
<ProviderApiKeyListItem
20+
key={provider.provider}
21+
provider={provider.provider}
22+
value={value}
23+
/>
24+
);
25+
};
26+
27+
return (
28+
providerApiKeys && (
29+
<Space direction="vertical" style={{ width: 600 }}>
30+
{providersList.map((item, index) => renderProviderApiKey(item))}
31+
</Space>
32+
)
33+
);
34+
};
Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
export const providersList = [
2+
{
3+
name: "OpenAI",
4+
provider: "OpenAI",
5+
iconBase64:
6+
"",
7+
},
8+
];

apps/console/src/app/components/prompts/editor/ProviderSettingsCard.tsx

Lines changed: 1 addition & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,7 @@
1-
import { Button, Divider, Form } from "antd";
1+
import { Divider, Form } from "antd";
22
import { usePromptVersionEditorContext } from "../../../lib/providers/PromptVersionEditorContext";
33
import { ProviderSelector } from "./ProviderSelector/ProviderSelector";
44
import { PromptService } from "@pezzo/types";
5-
import { SendOutlined } from "@ant-design/icons";
65
import { ProviderSettingsSchemaRenderer } from "./ProviderSettings/ProviderSettingsSchemaRenderer";
76
import { openAIChatCompletionSettingsDefinition } from "./ProviderSettings/providers/openai-chat-completion";
87
import { azureOpenAIChatCompletionSettingsDefinition } from "./ProviderSettings/providers/azure-openai-chat-completion";
@@ -35,12 +34,6 @@ export const ProviderSettingsCard = ({ onOpenFunctionsModal }: Props) => {
3534
<ProviderSettingsSchemaRenderer
3635
schema={providerSettings[service].generateFormSchema(settings)}
3736
/>
38-
{/* {service === PromptService.OpenAIChatCompletion &&
39-
onOpenFunctionsModal && (
40-
<Button onClick={onOpenFunctionsModal} icon={<SendOutlined />}>
41-
Edit Functions
42-
</Button>
43-
)} */}
4437
</>
4538
)}
4639
</>

apps/console/src/app/components/prompts/views/PromptEditView.tsx

Lines changed: 25 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,8 @@ import { PromptTesterModal } from "../prompt-tester/PromptTesterModal";
2323
import { usePromptTester } from "../../../lib/providers/PromptTesterContext";
2424
import { ConsumePromptModal } from "../ConsumePromptModal";
2525
import { trackEvent } from "../../../lib/utils/analytics";
26+
import { useRequiredProviderApiKeyModal } from "../../../lib/providers/RequiredProviderApiKeyModalProvider";
27+
import { useProviderApiKeys } from "../../../graphql/hooks/queries";
2628

2729
const FUNCTIONS_FEATURE_FLAG = true;
2830

@@ -31,6 +33,8 @@ export const PromptEditView = () => {
3133
const { prompt, isLoading: isPromptLoading } = useCurrentPrompt();
3234
const { currentVersion, isPublishEnabled, isDraft, form } =
3335
usePromptVersionEditorContext();
36+
const { providerApiKeys } = useProviderApiKeys();
37+
const { openRequiredProviderApiKeyModal } = useRequiredProviderApiKeyModal();
3438

3539
const [isCommitModalOpen, setIsCommitModalOpen] = useState(false);
3640
const [isConsumePromptModalOpen, setIsConsumePromptModalOpen] =
@@ -40,8 +44,28 @@ export const PromptEditView = () => {
4044

4145
const handleRunTest = () => {
4246
const formValues = form.getFieldsValue();
43-
openTestModal(formValues);
47+
48+
// TODO: make dynamic
49+
const provider = "OpenAI";
50+
const hasProviderApiKey = !!providerApiKeys.find(
51+
(key) => key.provider === provider
52+
);
4453
trackEvent("prompt_run_test_clicked");
54+
55+
if (!hasProviderApiKey) {
56+
openRequiredProviderApiKeyModal({
57+
callback: () => {
58+
openTestModal(formValues);
59+
},
60+
provider,
61+
});
62+
trackEvent("provider_api_keys_modal_due_to_missing_api_key", {
63+
provider,
64+
});
65+
return;
66+
}
67+
68+
openTestModal(formValues);
4569
};
4670

4771
const onConsumeClick = () => {

apps/console/src/app/graphql/definitions/mutations/api-keys.tsx

Lines changed: 0 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,6 @@ export const UPDATE_PROVIDER_API_KEY = graphql(/* GraphQL */ `
44
mutation UpdateProviderAPIKey($data: CreateProviderApiKeyInput!) {
55
updateProviderApiKey(data: $data) {
66
provider
7-
value
87
}
98
}
109
`);

apps/console/src/app/graphql/definitions/queries/api-keys.tsx

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ export const GET_ALL_PROVIDER_API_KEYS = graphql(/* GraphQL */ `
55
providerApiKeys(data: $data) {
66
id
77
provider
8-
value
8+
censoredValue
99
}
1010
}
1111
`);

apps/console/src/app/graphql/hooks/queries.ts

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -32,13 +32,15 @@ import {
3232
export const useProviderApiKeys = () => {
3333
const { organization } = useCurrentOrganization();
3434

35-
return useQuery({
35+
const result = useQuery({
3636
queryKey: ["providerApiKeys", organization?.id],
3737
queryFn: () =>
3838
gqlClient.request(GET_ALL_PROVIDER_API_KEYS, {
3939
data: { organizationId: organization?.id },
4040
}),
4141
});
42+
43+
return { ...result, providerApiKeys: result.data?.providerApiKeys ?? [] };
4244
};
4345

4446
export const usePezzoApiKeys = () => {

0 commit comments

Comments
 (0)