-
-
Notifications
You must be signed in to change notification settings - Fork 7
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
18 changed files
with
456 additions
and
64 deletions.
There are no files selected for viewing
81 changes: 81 additions & 0 deletions
81
apps/nextjs/src/app/[locale]/manage/tools/api/components/api-keys.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,81 @@ | ||
"use client"; | ||
|
||
import { useMemo } from "react"; | ||
import { Button, Group, Stack, Text, Title } from "@mantine/core"; | ||
import type { MRT_ColumnDef } from "mantine-react-table"; | ||
import { MantineReactTable, useMantineReactTable } from "mantine-react-table"; | ||
|
||
import type { RouterOutputs } from "@homarr/api"; | ||
import { clientApi } from "@homarr/api/client"; | ||
import { revalidatePathActionAsync } from "@homarr/common/client"; | ||
import { useModalAction } from "@homarr/modals"; | ||
import { useScopedI18n } from "@homarr/translation/client"; | ||
import { UserAvatar } from "@homarr/ui"; | ||
|
||
import { CopyApiKeyModal } from "~/app/[locale]/manage/tools/api/components/copy-api-key-modal"; | ||
|
||
interface ApiKeysManagementProps { | ||
apiKeys: RouterOutputs["apiKeys"]["getAll"]; | ||
} | ||
|
||
export const ApiKeysManagement = ({ apiKeys }: ApiKeysManagementProps) => { | ||
const { openModal } = useModalAction(CopyApiKeyModal); | ||
const { mutate, isPending } = clientApi.apiKeys.create.useMutation({ | ||
onSettled: async (data) => { | ||
if (!data) { | ||
return; | ||
} | ||
openModal({ | ||
apiKey: data.randomToken, | ||
}); | ||
await revalidatePathActionAsync("/manage/tools/api"); | ||
}, | ||
}); | ||
const t = useScopedI18n("management.page.tool.api.tab.apiKey"); | ||
|
||
const columns = useMemo<MRT_ColumnDef<RouterOutputs["apiKeys"]["getAll"][number]>[]>( | ||
() => [ | ||
{ | ||
accessorKey: "id", | ||
header: t("table.header.id"), | ||
}, | ||
{ | ||
accessorKey: "user", | ||
header: t("table.header.createdBy"), | ||
Cell: ({ row }) => ( | ||
<Group gap={"xs"}> | ||
<UserAvatar user={row.original.user} size={"sm"} /> | ||
<Text>{row.original.user.name}</Text> | ||
</Group> | ||
), | ||
}, | ||
], | ||
[], | ||
); | ||
|
||
const table = useMantineReactTable({ | ||
columns, | ||
data: apiKeys, | ||
renderTopToolbarCustomActions: () => ( | ||
<Button | ||
onClick={() => { | ||
mutate(); | ||
}} | ||
loading={isPending} | ||
> | ||
{t("button.createApiToken")} | ||
</Button> | ||
), | ||
enableDensityToggle: false, | ||
state: { | ||
density: "xs", | ||
}, | ||
}); | ||
|
||
return ( | ||
<Stack> | ||
<Title>{t("title")}</Title> | ||
<MantineReactTable table={table} /> | ||
</Stack> | ||
); | ||
}; |
34 changes: 34 additions & 0 deletions
34
apps/nextjs/src/app/[locale]/manage/tools/api/components/copy-api-key-modal.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
import { Button, CopyButton, PasswordInput, Stack, Text } from "@mantine/core"; | ||
import { useDisclosure } from "@mantine/hooks"; | ||
|
||
import { createModal } from "@homarr/modals"; | ||
import { useScopedI18n } from "@homarr/translation/client"; | ||
|
||
export const CopyApiKeyModal = createModal<{ apiKey: string }>(({ actions, innerProps }) => { | ||
const t = useScopedI18n("management.page.tool.api.modal.createApiToken"); | ||
const [visible, { toggle }] = useDisclosure(false); | ||
return ( | ||
<Stack> | ||
<Text>{t("description")}</Text> | ||
<PasswordInput value={innerProps.apiKey} visible={visible} onVisibilityChange={toggle} readOnly /> | ||
<CopyButton value={innerProps.apiKey}> | ||
{({ copy }) => ( | ||
<Button | ||
onClick={() => { | ||
copy(); | ||
actions.closeModal(); | ||
}} | ||
variant="default" | ||
fullWidth | ||
> | ||
{t("button")} | ||
</Button> | ||
)} | ||
</CopyButton> | ||
</Stack> | ||
); | ||
}).withOptions({ | ||
defaultTitle(t) { | ||
return t("management.page.tool.api.modal.createApiToken.title"); | ||
}, | ||
}); |
23 changes: 23 additions & 0 deletions
23
apps/nextjs/src/app/[locale]/manage/tools/api/components/swagger-ui.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
"use client"; | ||
|
||
import type { OpenAPIV3 } from "openapi-types"; | ||
import SwaggerUI from "swagger-ui-react"; | ||
|
||
// workaround for CSS that cannot be processed by next.js, https://github.com/swagger-api/swagger-ui/issues/10045 | ||
import "../swagger-ui-dark.css"; | ||
import "../swagger-ui-overrides.css"; | ||
import "../swagger-ui.css"; | ||
|
||
interface SwaggerUIClientProps { | ||
document: OpenAPIV3.Document; | ||
} | ||
|
||
export const SwaggerUIClient = ({ document }: SwaggerUIClientProps) => { | ||
// eslint-disable-next-line @typescript-eslint/no-explicit-any | ||
const requestInterceptor = (req: Record<string, any>) => { | ||
req.credentials = "omit"; | ||
return req; | ||
}; | ||
|
||
return <SwaggerUI requestInterceptor={requestInterceptor} spec={document} />; | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,14 +1,59 @@ | ||
import { createOpenApiFetchHandler } from "trpc-swagger/build/index.mjs"; | ||
import {createOpenApiFetchHandler} from "trpc-swagger/build/index.mjs"; | ||
|
||
import { appRouter, createTRPCContext } from "@homarr/api"; | ||
import {appRouter, createTRPCContext} from "@homarr/api"; | ||
import type {Session} from "@homarr/auth"; | ||
import {createSessionAsync} from "@homarr/auth/server"; | ||
import {db, eq} from "@homarr/db"; | ||
import {apiKeys} from "@homarr/db/schema/sqlite"; | ||
import {logger} from "@homarr/log"; | ||
|
||
const handlerAsync = async (req: Request) => { | ||
const apiKeyHeaderValue = req.headers.get("ApiKey"); | ||
const session: Session | null = await getSessionOrDefaultFromHeadersAsync(apiKeyHeaderValue); | ||
|
||
const handler = (req: Request) => { | ||
return createOpenApiFetchHandler({ | ||
req, | ||
endpoint: "/", | ||
router: appRouter, | ||
createContext: () => createTRPCContext({ session: null, headers: req.headers }), | ||
createContext: () => createTRPCContext({session, headers: req.headers}), | ||
}); | ||
}; | ||
|
||
export { handler as GET, handler as POST }; | ||
const getSessionOrDefaultFromHeadersAsync = async (apiKeyHeaderValue: string | null): Promise<Session | null> => { | ||
logger.info( | ||
`Creating OpenAPI fetch handler for user ${apiKeyHeaderValue ? "with an api key" : "without an api key"}`, | ||
); | ||
|
||
if (apiKeyHeaderValue === null) { | ||
return null; | ||
} | ||
|
||
const apiKeyFromDb = await db.query.apiKeys.findFirst({ | ||
where: eq(apiKeys.apiKey, apiKeyHeaderValue), | ||
columns: { | ||
id: true, | ||
apiKey: false, | ||
salt: false, | ||
}, | ||
with: { | ||
user: { | ||
columns: { | ||
id: true, | ||
name: true, | ||
email: true, | ||
emailVerified: true, | ||
}, | ||
}, | ||
}, | ||
}); | ||
|
||
if (apiKeyFromDb === undefined) { | ||
logger.warn(`An attempt to authenticate over API has failed`); | ||
return null; | ||
} | ||
|
||
logger.info(`Read session from API request and found user ${apiKeyFromDb.user.name} (${apiKeyFromDb.user.id})`); | ||
return await createSessionAsync(db, apiKeyFromDb.user); | ||
} | ||
|
||
export {handlerAsync as GET, handlerAsync as POST}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,41 @@ | ||
import { createSaltAsync, hashPasswordAsync } from "@homarr/auth"; | ||
import { generateSecureRandomToken } from "@homarr/common/server"; | ||
import { createId, db } from "@homarr/db"; | ||
import { apiKeys } from "@homarr/db/schema/sqlite"; | ||
|
||
import { createTRPCRouter, permissionRequiredProcedure } from "../trpc"; | ||
|
||
export const apiKeysRouter = createTRPCRouter({ | ||
getAll: permissionRequiredProcedure.requiresPermission("admin").query(() => { | ||
return db.query.apiKeys.findMany({ | ||
columns: { | ||
id: true, | ||
apiKey: false, | ||
salt: false, | ||
}, | ||
with: { | ||
user: { | ||
columns: { | ||
id: true, | ||
name: true, | ||
image: true, | ||
}, | ||
}, | ||
}, | ||
}); | ||
}), | ||
create: permissionRequiredProcedure.requiresPermission("admin").mutation(async ({ ctx }) => { | ||
const salt = await createSaltAsync(); | ||
const randomToken = generateSecureRandomToken(64); | ||
const hashedRandomToken = await hashPasswordAsync(randomToken, salt); | ||
await db.insert(apiKeys).values({ | ||
id: createId(), | ||
apiKey: hashedRandomToken, | ||
salt, | ||
userId: ctx.session.user.id, | ||
}); | ||
return { | ||
randomToken, | ||
}; | ||
}), | ||
}); |
Oops, something went wrong.