mirror of
https://github.com/outline/outline.git
synced 2025-12-31 07:49:47 -06:00
120 lines
3.5 KiB
TypeScript
120 lines
3.5 KiB
TypeScript
import Router from "koa-router";
|
|
import find from "lodash/find";
|
|
import { IntegrationService, IntegrationType } from "@shared/types";
|
|
import { parseDomain } from "@shared/utils/domains";
|
|
import Logger from "@server/logging/Logger";
|
|
import auth from "@server/middlewares/authentication";
|
|
import { transaction } from "@server/middlewares/transaction";
|
|
import validate from "@server/middlewares/validate";
|
|
import { IntegrationAuthentication, Integration, Team } from "@server/models";
|
|
import { APIContext } from "@server/types";
|
|
import { GitHubUtils } from "../../shared/GitHubUtils";
|
|
import { GitHub } from "../github";
|
|
import * as T from "./schema";
|
|
|
|
const router = new Router();
|
|
|
|
router.get(
|
|
"github.callback",
|
|
auth({
|
|
optional: true,
|
|
}),
|
|
validate(T.GitHubCallbackSchema),
|
|
transaction(),
|
|
async (ctx: APIContext<T.GitHubCallbackReq>) => {
|
|
const {
|
|
code,
|
|
state: teamId,
|
|
error,
|
|
installation_id: installationId,
|
|
setup_action: setupAction,
|
|
} = ctx.input.query;
|
|
const { user } = ctx.state.auth;
|
|
const { transaction } = ctx.state;
|
|
|
|
if (error) {
|
|
ctx.redirect(GitHubUtils.errorUrl(error));
|
|
return;
|
|
}
|
|
|
|
if (setupAction === T.SetupAction.request) {
|
|
ctx.redirect(GitHubUtils.installRequestUrl());
|
|
return;
|
|
}
|
|
|
|
// this code block accounts for the root domain being unable to
|
|
// access authentication for subdomains. We must forward to the appropriate
|
|
// subdomain to complete the oauth flow
|
|
if (!user) {
|
|
if (teamId) {
|
|
try {
|
|
const team = await Team.findByPk(teamId, {
|
|
rejectOnEmpty: true,
|
|
transaction,
|
|
});
|
|
return parseDomain(ctx.host).teamSubdomain === team.subdomain
|
|
? ctx.redirect("/")
|
|
: ctx.redirectOnClient(
|
|
GitHubUtils.callbackUrl({
|
|
baseUrl: team.url,
|
|
params: ctx.request.querystring,
|
|
})
|
|
);
|
|
} catch (err) {
|
|
Logger.error(`Error fetching team for teamId: ${teamId}!`, err);
|
|
return ctx.redirect(GitHubUtils.errorUrl("unauthenticated"));
|
|
}
|
|
} else {
|
|
return ctx.redirect(GitHubUtils.errorUrl("unauthenticated"));
|
|
}
|
|
}
|
|
|
|
const client = await GitHub.authenticateAsUser(code!, teamId);
|
|
const installationsByUser = await client.requestAppInstallations();
|
|
const installation = find(
|
|
installationsByUser,
|
|
(i) => i.id === installationId
|
|
);
|
|
|
|
if (!installation) {
|
|
return ctx.redirect(GitHubUtils.errorUrl("unauthenticated"));
|
|
}
|
|
|
|
const authentication = await IntegrationAuthentication.create(
|
|
{
|
|
service: IntegrationService.GitHub,
|
|
userId: user.id,
|
|
teamId: user.teamId,
|
|
},
|
|
{ transaction }
|
|
);
|
|
await Integration.create(
|
|
{
|
|
service: IntegrationService.GitHub,
|
|
type: IntegrationType.Embed,
|
|
userId: user.id,
|
|
teamId: user.teamId,
|
|
authenticationId: authentication.id,
|
|
settings: {
|
|
github: {
|
|
installation: {
|
|
id: installationId!,
|
|
account: {
|
|
id: installation.account?.id,
|
|
name:
|
|
// @ts-expect-error Property 'login' does not exist on type
|
|
installation.account?.login,
|
|
avatarUrl: installation.account?.avatar_url,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
{ transaction }
|
|
);
|
|
ctx.redirect(GitHubUtils.url);
|
|
}
|
|
);
|
|
|
|
export default router;
|