Skip to content

Commit

Permalink
Merge branch 'main' of https://github.com/unkeyed/unkey
Browse files Browse the repository at this point in the history
  • Loading branch information
chronark committed Oct 20, 2024
2 parents 4a6d726 + c657aff commit 011bc54
Show file tree
Hide file tree
Showing 7 changed files with 38 additions and 12 deletions.
16 changes: 13 additions & 3 deletions apps/api/src/pkg/keys/service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -57,7 +57,11 @@ type InvalidResponse = {
| "DISABLED"
| "INSUFFICIENT_PERMISSIONS";
key: Key;
identity: { id: string; externalId: string; meta: Record<string, unknown> | null } | null;
identity: {
id: string;
externalId: string;
meta: Record<string, unknown> | null;
} | null;
api: Api;
ratelimit?: {
remaining: number;
Expand All @@ -73,7 +77,11 @@ type ValidResponse = {
code?: never;
valid: true;
key: Key;
identity: { id: string; externalId: string; meta: Record<string, unknown> | null } | null;
identity: {
id: string;
externalId: string;
meta: Record<string, unknown> | null;
} | null;
api: Api;
ratelimit?: {
remaining: number;
Expand Down Expand Up @@ -451,6 +459,7 @@ export class KeyService {

if (data.api.ipWhitelist) {
const ip = c.req.header("True-Client-IP") ?? c.req.header("CF-Connecting-IP");

if (!ip) {
return Ok({
key: data.key,
Expand All @@ -461,7 +470,8 @@ export class KeyService {
permissions: data.permissions,
});
}
const ipWhitelist = JSON.parse(data.api.ipWhitelist) as string[];

const ipWhitelist = data.api.ipWhitelist.split(",").map((s) => s.trim());
if (!ipWhitelist.includes(ip)) {
return Ok({
key: data.key,
Expand Down
4 changes: 2 additions & 2 deletions apps/api/src/routes/legacy_keys_verifyKey.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -129,7 +129,7 @@ describe("with ip whitelist", () => {
name: "test",
authType: "key",
keyAuthId: keyAuthId,
ipWhitelist: JSON.stringify(["100.100.100.100"]),
ipWhitelist: ["100.100.100.100"].join(","),
createdAt: new Date(),
deletedAt: null,
});
Expand Down Expand Up @@ -177,7 +177,7 @@ describe("with ip whitelist", () => {
name: "test",
authType: "key",
keyAuthId: keyAuthid,
ipWhitelist: JSON.stringify(["100.100.100.100"]),
ipWhitelist: ["100.100.100.100"].join(","),
createdAt: new Date(),
deletedAt: null,
});
Expand Down
2 changes: 1 addition & 1 deletion apps/api/src/routes/v1_apis_getApi.happy.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ test("with ip whitelist", async (t) => {
id: newId("api"),
name: "with ip whitelist",
workspaceId: h.resources.userWorkspace.id,
ipWhitelist: JSON.stringify(["127.0.0.1"]),
ipWhitelist: ["127.0.0.1"].join(","),
createdAt: new Date(),
deletedAt: null,
};
Expand Down
5 changes: 3 additions & 2 deletions apps/api/src/routes/v1_keys_verifyKey.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -349,6 +349,7 @@ describe("when ratelimited", () => {
expect(res.body.identity!.externalId).toEqual(externalId);
});
});

describe("with ratelimit override", () => {
test("deducts the correct number of tokens", { timeout: 20000 }, async (t) => {
const h = await IntegrationHarness.init(t);
Expand Down Expand Up @@ -517,7 +518,7 @@ describe("with ip whitelist", () => {
name: "test",
authType: "key",
keyAuthId: keyAuthId,
ipWhitelist: JSON.stringify(["100.100.100.100"]),
ipWhitelist: ["100.100.100.100"].join(","),
createdAt: new Date(),
});

Expand Down Expand Up @@ -565,7 +566,7 @@ describe("with ip whitelist", () => {
name: "test",
authType: "key",
keyAuthId: keyAuthid,
ipWhitelist: JSON.stringify(["100.100.100.100"]),
ipWhitelist: ["100.100.100.100"].join(","),
createdAt: new Date(),
});

Expand Down
1 change: 1 addition & 0 deletions apps/dashboard/app/(app)/apis/[apiId]/settings/page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ export default async function SettingsPage(props: Props) {
if (!workspace || workspace.tenantId !== tenantId) {
return redirect("/new");
}

const api = workspace.apis.find((api) => api.id === props.params.apiId);
if (!api) {
return notFound();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ const formSchema = z.object({

type Props = {
workspace: {
plan: Workspace["plan"];
features: Workspace["features"];
};
api: {
id: string;
Expand All @@ -42,7 +42,7 @@ type Props = {

export const UpdateIpWhitelist: React.FC<Props> = ({ api, workspace }) => {
const router = useRouter();
const isEnabled = workspace.plan === "enterprise";
const isEnabled = workspace.features.ipWhitelist;

const form = useForm<z.infer<typeof formSchema>>({
resolver: zodResolver(formSchema),
Expand Down Expand Up @@ -79,7 +79,7 @@ export const UpdateIpWhitelist: React.FC<Props> = ({ api, workspace }) => {
</CardDescription>
</CardHeader>
<CardContent>
{workspace.plan === "enterprise" ? (
{isEnabled ? (
<div className="flex flex-col space-y-2">
<input type="hidden" name="workspaceId" value={api.workspaceId} />
<input type="hidden" name="apiId" value={api.id} />
Expand Down
16 changes: 15 additions & 1 deletion apps/dashboard/lib/trpc/routers/api/updateIpWhitelist.ts
Original file line number Diff line number Diff line change
Expand Up @@ -43,14 +43,27 @@ export const updateApiIpWhitelist = rateLimitedProcedure(ratelimit.update)
"We are unable to update the API whitelist. Please try again or contact [email protected]",
});
});
if (!api || api.workspace.tenantId !== ctx.tenant.id) {

if (
!api ||
api.workspace.tenantId !== ctx.tenant.id ||
input.workspaceId !== api.workspace.id
) {
throw new TRPCError({
code: "NOT_FOUND",
message:
"We are unable to find the correct API. Please try again or contact [email protected].",
});
}

if (!api.workspace.features.ipWhitelist) {
throw new TRPCError({
code: "FORBIDDEN",
message:
"IP Whitelisting is only available for enterprise plans. Please contact [email protected].",
});
}

const newIpWhitelist = input.ipWhitelist === null ? null : input.ipWhitelist.join(",");

await db
Expand All @@ -68,6 +81,7 @@ export const updateApiIpWhitelist = rateLimitedProcedure(ratelimit.update)
"We are unable to update the API whitelist. Please try again or contact [email protected]",
});
});

await insertAuditLogs(tx, {
workspaceId: api.workspace.id,
actor: {
Expand Down

0 comments on commit 011bc54

Please sign in to comment.