mirror of
https://github.com/Infisical/infisical.git
synced 2026-01-09 15:38:03 -05:00
misc: initial setup for github integration with Github app auth
This commit is contained in:
@@ -36,12 +36,15 @@ CLIENT_ID_HEROKU=
|
|||||||
CLIENT_ID_VERCEL=
|
CLIENT_ID_VERCEL=
|
||||||
CLIENT_ID_NETLIFY=
|
CLIENT_ID_NETLIFY=
|
||||||
CLIENT_ID_GITHUB=
|
CLIENT_ID_GITHUB=
|
||||||
|
CLIENT_ID_GITHUB_APP=
|
||||||
|
CLIENT_SLUG_GITHUB=
|
||||||
CLIENT_ID_GITLAB=
|
CLIENT_ID_GITLAB=
|
||||||
CLIENT_ID_BITBUCKET=
|
CLIENT_ID_BITBUCKET=
|
||||||
CLIENT_SECRET_HEROKU=
|
CLIENT_SECRET_HEROKU=
|
||||||
CLIENT_SECRET_VERCEL=
|
CLIENT_SECRET_VERCEL=
|
||||||
CLIENT_SECRET_NETLIFY=
|
CLIENT_SECRET_NETLIFY=
|
||||||
CLIENT_SECRET_GITHUB=
|
CLIENT_SECRET_GITHUB=
|
||||||
|
CLIENT_SECRET_GITHUB_APP=
|
||||||
CLIENT_SECRET_GITLAB=
|
CLIENT_SECRET_GITLAB=
|
||||||
CLIENT_SECRET_BITBUCKET=
|
CLIENT_SECRET_BITBUCKET=
|
||||||
CLIENT_SLUG_VERCEL=
|
CLIENT_SLUG_VERCEL=
|
||||||
|
|||||||
@@ -117,9 +117,14 @@ const envSchema = z
|
|||||||
// gcp secret manager
|
// gcp secret manager
|
||||||
CLIENT_ID_GCP_SECRET_MANAGER: zpStr(z.string().optional()),
|
CLIENT_ID_GCP_SECRET_MANAGER: zpStr(z.string().optional()),
|
||||||
CLIENT_SECRET_GCP_SECRET_MANAGER: zpStr(z.string().optional()),
|
CLIENT_SECRET_GCP_SECRET_MANAGER: zpStr(z.string().optional()),
|
||||||
// github
|
// github oauth
|
||||||
CLIENT_ID_GITHUB: zpStr(z.string().optional()),
|
CLIENT_ID_GITHUB: zpStr(z.string().optional()),
|
||||||
CLIENT_SECRET_GITHUB: zpStr(z.string().optional()),
|
CLIENT_SECRET_GITHUB: zpStr(z.string().optional()),
|
||||||
|
CLIENT_SLUG_GITHUB: zpStr(z.string().optional()),
|
||||||
|
// github app
|
||||||
|
CLIENT_ID_GITHUB_APP: zpStr(z.string().optional()),
|
||||||
|
CLIENT_SECRET_GITHUB_APP: zpStr(z.string().optional()),
|
||||||
|
|
||||||
// azure
|
// azure
|
||||||
CLIENT_ID_AZURE: zpStr(z.string().optional()),
|
CLIENT_ID_AZURE: zpStr(z.string().optional()),
|
||||||
CLIENT_SECRET_AZURE: zpStr(z.string().optional()),
|
CLIENT_SECRET_AZURE: zpStr(z.string().optional()),
|
||||||
|
|||||||
@@ -189,6 +189,7 @@ export const registerIntegrationAuthRouter = async (server: FastifyZodProvider)
|
|||||||
workspaceId: z.string().trim(),
|
workspaceId: z.string().trim(),
|
||||||
code: z.string().trim(),
|
code: z.string().trim(),
|
||||||
integration: z.string().trim(),
|
integration: z.string().trim(),
|
||||||
|
installationId: z.string().trim().optional(),
|
||||||
url: z.string().trim().url().optional()
|
url: z.string().trim().url().optional()
|
||||||
}),
|
}),
|
||||||
response: {
|
response: {
|
||||||
|
|||||||
@@ -109,7 +109,8 @@ export const integrationAuthServiceFactory = ({
|
|||||||
actorAuthMethod,
|
actorAuthMethod,
|
||||||
integration,
|
integration,
|
||||||
url,
|
url,
|
||||||
code
|
code,
|
||||||
|
installationId
|
||||||
}: TOauthExchangeDTO) => {
|
}: TOauthExchangeDTO) => {
|
||||||
if (!Object.values(Integrations).includes(integration as Integrations))
|
if (!Object.values(Integrations).includes(integration as Integrations))
|
||||||
throw new BadRequestError({ message: "Invalid integration" });
|
throw new BadRequestError({ message: "Invalid integration" });
|
||||||
@@ -123,7 +124,7 @@ export const integrationAuthServiceFactory = ({
|
|||||||
);
|
);
|
||||||
ForbiddenError.from(permission).throwUnlessCan(ProjectPermissionActions.Create, ProjectPermissionSub.Integrations);
|
ForbiddenError.from(permission).throwUnlessCan(ProjectPermissionActions.Create, ProjectPermissionSub.Integrations);
|
||||||
|
|
||||||
const tokenExchange = await exchangeCode({ integration, code, url });
|
const tokenExchange = await exchangeCode({ integration, code, url, installationId });
|
||||||
const updateDoc: TIntegrationAuthsInsert = {
|
const updateDoc: TIntegrationAuthsInsert = {
|
||||||
projectId,
|
projectId,
|
||||||
integration,
|
integration,
|
||||||
@@ -141,6 +142,16 @@ export const integrationAuthServiceFactory = ({
|
|||||||
updateDoc.metadata = {
|
updateDoc.metadata = {
|
||||||
authMethod: "oauth2"
|
authMethod: "oauth2"
|
||||||
};
|
};
|
||||||
|
} else if (integration === Integrations.GITHUB && installationId) {
|
||||||
|
updateDoc.metadata = {
|
||||||
|
installationId,
|
||||||
|
installationName: tokenExchange.installationName,
|
||||||
|
authMethod: "app"
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (installationId && integration === Integrations.GITHUB) {
|
||||||
|
return integrationAuthDAL.create(updateDoc);
|
||||||
}
|
}
|
||||||
|
|
||||||
const { shouldUseSecretV2Bridge, botKey } = await projectBotService.getBotKey(projectId);
|
const { shouldUseSecretV2Bridge, botKey } = await projectBotService.getBotKey(projectId);
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ export type TOauthExchangeDTO = {
|
|||||||
integration: string;
|
integration: string;
|
||||||
code: string;
|
code: string;
|
||||||
url?: string;
|
url?: string;
|
||||||
|
installationId?: string;
|
||||||
} & TProjectPermission;
|
} & TProjectPermission;
|
||||||
|
|
||||||
export type TSaveIntegrationAccessTokenDTO = {
|
export type TSaveIntegrationAccessTokenDTO = {
|
||||||
|
|||||||
@@ -96,7 +96,9 @@ export enum IntegrationUrls {
|
|||||||
GCP_SECRET_MANAGER_SERVICE_NAME = "secretmanager.googleapis.com",
|
GCP_SECRET_MANAGER_SERVICE_NAME = "secretmanager.googleapis.com",
|
||||||
GCP_SECRET_MANAGER_URL = `https://${GCP_SECRET_MANAGER_SERVICE_NAME}`,
|
GCP_SECRET_MANAGER_URL = `https://${GCP_SECRET_MANAGER_SERVICE_NAME}`,
|
||||||
GCP_SERVICE_USAGE_URL = "https://serviceusage.googleapis.com",
|
GCP_SERVICE_USAGE_URL = "https://serviceusage.googleapis.com",
|
||||||
GCP_CLOUD_PLATFORM_SCOPE = "https://www.googleapis.com/auth/cloud-platform"
|
GCP_CLOUD_PLATFORM_SCOPE = "https://www.googleapis.com/auth/cloud-platform",
|
||||||
|
|
||||||
|
GITHUB_USER_INSTALLATIONS = "https://api.github.com/user/installations"
|
||||||
}
|
}
|
||||||
|
|
||||||
export const getIntegrationOptions = async () => {
|
export const getIntegrationOptions = async () => {
|
||||||
@@ -138,6 +140,7 @@ export const getIntegrationOptions = async () => {
|
|||||||
isAvailable: true,
|
isAvailable: true,
|
||||||
type: "oauth",
|
type: "oauth",
|
||||||
clientId: appCfg.CLIENT_ID_GITHUB,
|
clientId: appCfg.CLIENT_ID_GITHUB,
|
||||||
|
clientSlug: appCfg.CLIENT_SLUG_GITHUB,
|
||||||
docsLink: ""
|
docsLink: ""
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import jwt from "jsonwebtoken";
|
|||||||
|
|
||||||
import { getConfig } from "@app/lib/config/env";
|
import { getConfig } from "@app/lib/config/env";
|
||||||
import { request } from "@app/lib/config/request";
|
import { request } from "@app/lib/config/request";
|
||||||
import { BadRequestError, NotFoundError } from "@app/lib/errors";
|
import { BadRequestError, ForbiddenRequestError, InternalServerError, NotFoundError } from "@app/lib/errors";
|
||||||
|
|
||||||
import { Integrations, IntegrationUrls } from "./integration-list";
|
import { Integrations, IntegrationUrls } from "./integration-list";
|
||||||
|
|
||||||
@@ -234,12 +234,73 @@ const exchangeCodeNetlify = async ({ code }: { code: string }) => {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
const exchangeCodeGithub = async ({ code }: { code: string }) => {
|
const exchangeCodeGithub = async ({ code, installationId }: { code: string; installationId?: string }) => {
|
||||||
const appCfg = getConfig();
|
const appCfg = getConfig();
|
||||||
if (!appCfg.CLIENT_ID_GITHUB || !appCfg.CLIENT_SECRET_GITHUB) {
|
|
||||||
throw new BadRequestError({ message: "Missing client id and client secret" });
|
if (!installationId && (!appCfg.CLIENT_ID_GITHUB || !appCfg.CLIENT_SECRET_GITHUB)) {
|
||||||
|
throw new InternalServerError({ message: "Missing client id and client secret" });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (installationId && (!appCfg.CLIENT_ID_GITHUB_APP || !appCfg.CLIENT_SECRET_GITHUB_APP)) {
|
||||||
|
throw new InternalServerError({
|
||||||
|
message: "Missing Github app client ID and client secret"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (installationId) {
|
||||||
|
// handle app installations
|
||||||
|
const res = (
|
||||||
|
await request.get<ExchangeCodeGithubResponse>(IntegrationUrls.GITHUB_TOKEN_URL, {
|
||||||
|
params: {
|
||||||
|
client_id: appCfg.CLIENT_ID_GITHUB_APP,
|
||||||
|
client_secret: appCfg.CLIENT_SECRET_GITHUB_APP,
|
||||||
|
code,
|
||||||
|
redirect_uri: `${appCfg.SITE_URL}/integrations/github/oauth2/callback`
|
||||||
|
},
|
||||||
|
headers: {
|
||||||
|
Accept: "application/json",
|
||||||
|
"Accept-Encoding": "application/json"
|
||||||
|
}
|
||||||
|
})
|
||||||
|
).data;
|
||||||
|
|
||||||
|
// use access token to validate installation ID
|
||||||
|
const installationsRes = (
|
||||||
|
await request.get<{
|
||||||
|
installations: {
|
||||||
|
id: number;
|
||||||
|
account: {
|
||||||
|
login: string;
|
||||||
|
};
|
||||||
|
}[];
|
||||||
|
}>(IntegrationUrls.GITHUB_USER_INSTALLATIONS, {
|
||||||
|
headers: {
|
||||||
|
Accept: "application/json",
|
||||||
|
Authorization: `Bearer ${res.access_token}`,
|
||||||
|
"Accept-Encoding": "application/json"
|
||||||
|
}
|
||||||
|
})
|
||||||
|
).data;
|
||||||
|
|
||||||
|
const matchingInstallation = installationsRes.installations.find(
|
||||||
|
(installation) => installation.id === +installationId
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!matchingInstallation) {
|
||||||
|
throw new ForbiddenRequestError({
|
||||||
|
message: "User has no access to the provided installation"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
accessToken: "",
|
||||||
|
refreshToken: null,
|
||||||
|
accessExpiresAt: null,
|
||||||
|
installationName: matchingInstallation.account.login
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// handle normal oauth
|
||||||
const res = (
|
const res = (
|
||||||
await request.get<ExchangeCodeGithubResponse>(IntegrationUrls.GITHUB_TOKEN_URL, {
|
await request.get<ExchangeCodeGithubResponse>(IntegrationUrls.GITHUB_TOKEN_URL, {
|
||||||
params: {
|
params: {
|
||||||
@@ -346,6 +407,7 @@ type TExchangeReturn = {
|
|||||||
url?: string;
|
url?: string;
|
||||||
teamId?: string;
|
teamId?: string;
|
||||||
accountId?: string;
|
accountId?: string;
|
||||||
|
installationName?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -355,11 +417,13 @@ type TExchangeReturn = {
|
|||||||
export const exchangeCode = async ({
|
export const exchangeCode = async ({
|
||||||
integration,
|
integration,
|
||||||
code,
|
code,
|
||||||
url
|
url,
|
||||||
|
installationId
|
||||||
}: {
|
}: {
|
||||||
integration: string;
|
integration: string;
|
||||||
code: string;
|
code: string;
|
||||||
url?: string;
|
url?: string;
|
||||||
|
installationId?: string;
|
||||||
}): Promise<TExchangeReturn> => {
|
}): Promise<TExchangeReturn> => {
|
||||||
switch (integration) {
|
switch (integration) {
|
||||||
case Integrations.GCP_SECRET_MANAGER:
|
case Integrations.GCP_SECRET_MANAGER:
|
||||||
@@ -384,7 +448,8 @@ export const exchangeCode = async ({
|
|||||||
});
|
});
|
||||||
case Integrations.GITHUB:
|
case Integrations.GITHUB:
|
||||||
return exchangeCodeGithub({
|
return exchangeCodeGithub({
|
||||||
code
|
code,
|
||||||
|
installationId
|
||||||
});
|
});
|
||||||
case Integrations.GITLAB:
|
case Integrations.GITLAB:
|
||||||
return exchangeCodeGitlab({
|
return exchangeCodeGitlab({
|
||||||
|
|||||||
@@ -777,11 +777,13 @@ export const useAuthorizeIntegration = () => {
|
|||||||
workspaceId,
|
workspaceId,
|
||||||
code,
|
code,
|
||||||
integration,
|
integration,
|
||||||
|
installationId,
|
||||||
url
|
url
|
||||||
}: {
|
}: {
|
||||||
workspaceId: string;
|
workspaceId: string;
|
||||||
code: string;
|
code: string;
|
||||||
integration: string;
|
integration: string;
|
||||||
|
installationId?: string;
|
||||||
url?: string;
|
url?: string;
|
||||||
}) => {
|
}) => {
|
||||||
const {
|
const {
|
||||||
@@ -790,6 +792,7 @@ export const useAuthorizeIntegration = () => {
|
|||||||
workspaceId,
|
workspaceId,
|
||||||
code,
|
code,
|
||||||
integration,
|
integration,
|
||||||
|
installationId,
|
||||||
url
|
url
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -9,6 +9,9 @@ export type IntegrationAuth = {
|
|||||||
keyEncoding: string;
|
keyEncoding: string;
|
||||||
url?: string;
|
url?: string;
|
||||||
teamId?: string;
|
teamId?: string;
|
||||||
|
metadata: {
|
||||||
|
installationName?: string;
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
export type App = {
|
export type App = {
|
||||||
|
|||||||
@@ -0,0 +1,89 @@
|
|||||||
|
import crypto from "crypto";
|
||||||
|
|
||||||
|
import Head from "next/head";
|
||||||
|
import Image from "next/image";
|
||||||
|
import Link from "next/link";
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
import { faArrowUpRightFromSquare, faBookOpen } from "@fortawesome/free-solid-svg-icons";
|
||||||
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
|
|
||||||
|
import { Button, Card, CardTitle } from "@app/components/v2";
|
||||||
|
import { useGetCloudIntegrations } from "@app/hooks/api";
|
||||||
|
|
||||||
|
export default function GithubIntegrationAuthModeSelectionPage() {
|
||||||
|
const router = useRouter();
|
||||||
|
const { data: cloudIntegrations } = useGetCloudIntegrations();
|
||||||
|
const githubIntegration = cloudIntegrations?.find((integration) => integration.slug === "github");
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex h-full w-full items-center justify-center">
|
||||||
|
<Head>
|
||||||
|
<title>Select Github Integration Auth</title>
|
||||||
|
<link rel="icon" href="/infisical.ico" />
|
||||||
|
</Head>
|
||||||
|
<Card className="mb-12 max-w-lg rounded-md border border-mineshaft-600">
|
||||||
|
<CardTitle
|
||||||
|
className="px-6 text-left text-xl"
|
||||||
|
subTitle="Select how you'd like to integrate with GitHub. For more precise control, we recommend using the GitHub App method."
|
||||||
|
>
|
||||||
|
<div className="flex flex-row items-center">
|
||||||
|
<div className="flex items-center pb-0.5">
|
||||||
|
<Image
|
||||||
|
src="/images/integrations/GitHub.png"
|
||||||
|
height={30}
|
||||||
|
width={30}
|
||||||
|
alt="Github logo"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span className="ml-2.5">Github Integration </span>
|
||||||
|
<Link href="https://infisical.com/docs/integrations/cicd/githubactions" passHref>
|
||||||
|
<a target="_blank" rel="noopener noreferrer">
|
||||||
|
<div className="ml-2 mb-1 inline-block cursor-default rounded-md bg-yellow/20 px-1.5 pb-[0.03rem] pt-[0.04rem] text-sm text-yellow opacity-80 hover:opacity-100">
|
||||||
|
<FontAwesomeIcon icon={faBookOpen} className="mr-1.5" />
|
||||||
|
Docs
|
||||||
|
<FontAwesomeIcon
|
||||||
|
icon={faArrowUpRightFromSquare}
|
||||||
|
className="ml-1.5 mb-[0.07rem] text-xxs"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
</CardTitle>
|
||||||
|
<div className="mb-7 flex flex-col items-center">
|
||||||
|
<Button
|
||||||
|
colorSchema="primary"
|
||||||
|
variant="outline_bg"
|
||||||
|
className="mt-2 w-3/4"
|
||||||
|
size="sm"
|
||||||
|
type="submit"
|
||||||
|
onClick={() => {
|
||||||
|
router.push("/integrations/select-integration-auth?integrationSlug=github");
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Connect with Github App
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
colorSchema="primary"
|
||||||
|
variant="outline_bg"
|
||||||
|
className="mt-3 w-3/4"
|
||||||
|
size="sm"
|
||||||
|
type="submit"
|
||||||
|
onClick={() => {
|
||||||
|
const state = crypto.randomBytes(16).toString("hex");
|
||||||
|
localStorage.setItem("latestCSRFToken", state);
|
||||||
|
|
||||||
|
window.location.assign(
|
||||||
|
`https://github.com/login/oauth/authorize?client_id=${githubIntegration?.clientId}&response_type=code&scope=repo,admin:org&redirect_uri=${window.location.origin}/integrations/github/oauth2/callback&state=${state}`
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Connect with Github OAuth
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
GithubIntegrationAuthModeSelectionPage.requireAuth = true;
|
||||||
@@ -8,18 +8,23 @@ export default function GitHubOAuth2CallbackPage() {
|
|||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const { mutateAsync } = useAuthorizeIntegration();
|
const { mutateAsync } = useAuthorizeIntegration();
|
||||||
|
|
||||||
const { code, state } = queryString.parse(router.asPath.split("?")[1]);
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
||||||
|
const { code, state, installation_id } = queryString.parse(router.asPath.split("?")[1]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
(async () => {
|
(async () => {
|
||||||
try {
|
try {
|
||||||
// validate state
|
// validate state
|
||||||
if (state !== localStorage.getItem("latestCSRFToken")) return;
|
if (state !== localStorage.getItem("latestCSRFToken")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
localStorage.removeItem("latestCSRFToken");
|
localStorage.removeItem("latestCSRFToken");
|
||||||
|
|
||||||
const integrationAuth = await mutateAsync({
|
const integrationAuth = await mutateAsync({
|
||||||
workspaceId: localStorage.getItem("projectData.id") as string,
|
workspaceId: localStorage.getItem("projectData.id") as string,
|
||||||
code: code as string,
|
code: code as string,
|
||||||
|
installationId: installation_id as string,
|
||||||
integration: "github"
|
integration: "github"
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
121
frontend/src/pages/integrations/select-integration-auth.tsx
Normal file
121
frontend/src/pages/integrations/select-integration-auth.tsx
Normal file
@@ -0,0 +1,121 @@
|
|||||||
|
import crypto from "crypto";
|
||||||
|
|
||||||
|
import { useCallback } from "react";
|
||||||
|
import Head from "next/head";
|
||||||
|
import Image from "next/image";
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
|
||||||
|
import { Button, Card, CardTitle } from "@app/components/v2";
|
||||||
|
import { useWorkspace } from "@app/context";
|
||||||
|
import { useGetCloudIntegrations, useGetWorkspaceAuthorizations } from "@app/hooks/api";
|
||||||
|
import { IntegrationAuth } from "@app/hooks/api/types";
|
||||||
|
|
||||||
|
export default function SelectIntegrationAuthPage() {
|
||||||
|
const router = useRouter();
|
||||||
|
const { data: cloudIntegrations } = useGetCloudIntegrations();
|
||||||
|
const { currentWorkspace } = useWorkspace();
|
||||||
|
const workspaceId = currentWorkspace?.id || "";
|
||||||
|
|
||||||
|
const integrationSlug = router.query.integrationSlug as string;
|
||||||
|
|
||||||
|
const currentIntegration = cloudIntegrations?.find(
|
||||||
|
(integration) => integration.slug === integrationSlug
|
||||||
|
);
|
||||||
|
|
||||||
|
const { data: integrationAuths, isLoading: isLoadingIntegrationAuths } =
|
||||||
|
useGetWorkspaceAuthorizations(
|
||||||
|
workspaceId,
|
||||||
|
useCallback((data: IntegrationAuth[]) => {
|
||||||
|
const filteredIntegrationAuths = data.filter(
|
||||||
|
(integrationAuth) => integrationAuth.integration === integrationSlug
|
||||||
|
);
|
||||||
|
|
||||||
|
if (integrationSlug === "github") {
|
||||||
|
// for now, we only display the integration auths for Github apps
|
||||||
|
return filteredIntegrationAuths.filter((integrationAuth) =>
|
||||||
|
Boolean(integrationAuth.metadata?.installationName)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return [];
|
||||||
|
}, [])
|
||||||
|
);
|
||||||
|
|
||||||
|
const logo = integrationSlug === "github" ? "/images/integrations/GitHub.png" : "";
|
||||||
|
|
||||||
|
const handleConnectionSelect = (integrationAuthId: string) => {
|
||||||
|
if (integrationSlug === "github") {
|
||||||
|
router.push(`/integrations/github/create?integrationAuthId=${integrationAuthId}`);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleNewConnection = () => {
|
||||||
|
const state = crypto.randomBytes(16).toString("hex");
|
||||||
|
localStorage.setItem("latestCSRFToken", state);
|
||||||
|
|
||||||
|
if (integrationSlug === "github") {
|
||||||
|
// for now we only handle Github apps
|
||||||
|
window.location.assign(
|
||||||
|
`https://github.com/apps/${currentIntegration?.clientSlug}/installations/new?state=${state}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex h-full w-full items-center justify-center">
|
||||||
|
<Head>
|
||||||
|
<title>Select Connection</title>
|
||||||
|
<link rel="icon" href="/infisical.ico" />
|
||||||
|
</Head>
|
||||||
|
<Card className="mb-12 max-w-lg rounded-md border border-mineshaft-600">
|
||||||
|
<CardTitle
|
||||||
|
className="px-6 text-left text-xl"
|
||||||
|
subTitle="Select a connection that you want to use for the new integration."
|
||||||
|
>
|
||||||
|
<div className="flex flex-row items-center">
|
||||||
|
<div className="flex items-center pb-0.5">
|
||||||
|
<Image src={logo} height={30} width={30} alt="Integration logo" />
|
||||||
|
</div>
|
||||||
|
<span className="ml-2.5">Select Connection</span>
|
||||||
|
</div>
|
||||||
|
</CardTitle>
|
||||||
|
<div className="mb-7 flex flex-col items-center">
|
||||||
|
{!isLoadingIntegrationAuths && integrationAuths?.length
|
||||||
|
? integrationAuths.map((integrationAuth) => {
|
||||||
|
let connectionName = "";
|
||||||
|
|
||||||
|
if (integrationAuth.integration === "github") {
|
||||||
|
connectionName = integrationAuth.metadata?.installationName || "";
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
colorSchema="gray"
|
||||||
|
variant="outline"
|
||||||
|
className="mt-3 w-3/4"
|
||||||
|
key={integrationAuth.id}
|
||||||
|
size="sm"
|
||||||
|
type="submit"
|
||||||
|
onClick={() => handleConnectionSelect(integrationAuth.id)}
|
||||||
|
>
|
||||||
|
{connectionName}
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
})
|
||||||
|
: undefined}
|
||||||
|
<Button
|
||||||
|
colorSchema="primary"
|
||||||
|
className="mt-6 w-3/4"
|
||||||
|
size="sm"
|
||||||
|
type="submit"
|
||||||
|
onClick={handleNewConnection}
|
||||||
|
>
|
||||||
|
Create New Connection
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
SelectIntegrationAuthPage.requireAuth = true;
|
||||||
@@ -60,7 +60,7 @@ export const redirectForProviderAuth = (integrationOption: TCloudIntegration) =>
|
|||||||
link = `https://app.netlify.com/authorize?client_id=${integrationOption.clientId}&response_type=code&state=${state}&redirect_uri=${window.location.origin}/integrations/netlify/oauth2/callback`;
|
link = `https://app.netlify.com/authorize?client_id=${integrationOption.clientId}&response_type=code&state=${state}&redirect_uri=${window.location.origin}/integrations/netlify/oauth2/callback`;
|
||||||
break;
|
break;
|
||||||
case "github":
|
case "github":
|
||||||
link = `https://github.com/login/oauth/authorize?client_id=${integrationOption.clientId}&response_type=code&scope=repo,admin:org&redirect_uri=${window.location.origin}/integrations/github/oauth2/callback&state=${state}`;
|
link = `${window.location.origin}/integrations/github/auth-mode-selection`;
|
||||||
break;
|
break;
|
||||||
case "gitlab":
|
case "gitlab":
|
||||||
link = `${window.location.origin}/integrations/gitlab/authorize`;
|
link = `${window.location.origin}/integrations/gitlab/authorize`;
|
||||||
|
|||||||
Reference in New Issue
Block a user