mirror of
https://github.com/outline/outline.git
synced 2026-06-13 11:25:03 +03:00
5584089441
* OAuth * store logo * unfurl support * refresh token * support for list * embed list * mention menu for all embeds in a list * multi-level list * logo * account level connection * tsc * Update Icon.tsx * coderabbit feedback * RFC 6749 suggestion --------- Co-authored-by: Tom Moor <tom@getoutline.com>
105 lines
3.3 KiB
TypeScript
105 lines
3.3 KiB
TypeScript
import Router from "koa-router";
|
|
import { IntegrationService, IntegrationType } from "@shared/types";
|
|
import Logger from "@server/logging/Logger";
|
|
import apexAuthRedirect from "@server/middlewares/apexAuthRedirect";
|
|
import auth from "@server/middlewares/authentication";
|
|
import { transaction } from "@server/middlewares/transaction";
|
|
import validate from "@server/middlewares/validate";
|
|
import { IntegrationAuthentication, Integration } from "@server/models";
|
|
import type { APIContext } from "@server/types";
|
|
import { Linear } from "../linear";
|
|
import UploadIntegrationLogoTask from "@server/queues/tasks/UploadIntegrationLogoTask";
|
|
import * as T from "./schema";
|
|
import { LinearUtils } from "plugins/linear/shared/LinearUtils";
|
|
import { addSeconds } from "date-fns";
|
|
|
|
const router = new Router();
|
|
|
|
router.get(
|
|
"linear.callback",
|
|
auth({
|
|
optional: true,
|
|
}),
|
|
validate(T.LinearCallbackSchema),
|
|
apexAuthRedirect<T.LinearCallbackReq>({
|
|
getTeamId: (ctx) => LinearUtils.parseState(ctx.input.query.state)?.teamId,
|
|
getRedirectPath: (ctx, team) =>
|
|
LinearUtils.callbackUrl({
|
|
baseUrl: team.url,
|
|
params: ctx.request.querystring,
|
|
}),
|
|
getErrorPath: () => LinearUtils.errorUrl("unauthenticated"),
|
|
}),
|
|
transaction(),
|
|
async (ctx: APIContext<T.LinearCallbackReq>) => {
|
|
const { code, error } = ctx.input.query;
|
|
const { user } = ctx.state.auth;
|
|
const { transaction } = ctx.state;
|
|
|
|
// Check error after any sub-domain redirection. Otherwise, the user will be redirected to the root domain.
|
|
if (error) {
|
|
ctx.redirect(LinearUtils.errorUrl(error));
|
|
return;
|
|
}
|
|
|
|
try {
|
|
// validation middleware ensures that code is non-null at this point.
|
|
const oauth = await Linear.oauthAccess(code!);
|
|
const workspace = await Linear.getInstalledWorkspace(oauth.access_token);
|
|
|
|
const authentication = await IntegrationAuthentication.create(
|
|
{
|
|
service: IntegrationService.Linear,
|
|
userId: user.id,
|
|
teamId: user.teamId,
|
|
token: oauth.access_token,
|
|
refreshToken: oauth.refresh_token,
|
|
expiresAt: oauth.expires_in
|
|
? addSeconds(Date.now(), oauth.expires_in)
|
|
: undefined,
|
|
scopes: oauth.scope.split(" "),
|
|
},
|
|
{ transaction }
|
|
);
|
|
const integration = await Integration.create<
|
|
Integration<IntegrationType.Embed>
|
|
>(
|
|
{
|
|
service: IntegrationService.Linear,
|
|
type: IntegrationType.Embed,
|
|
userId: user.id,
|
|
teamId: user.teamId,
|
|
authenticationId: authentication.id,
|
|
settings: {
|
|
linear: {
|
|
workspace: {
|
|
id: workspace.id,
|
|
name: workspace.name,
|
|
key: workspace.urlKey,
|
|
logoUrl: workspace.logoUrl,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
{ transaction }
|
|
);
|
|
|
|
transaction.afterCommit(async () => {
|
|
if (workspace.logoUrl) {
|
|
await new UploadIntegrationLogoTask().schedule({
|
|
integrationId: integration.id,
|
|
logoUrl: workspace.logoUrl,
|
|
});
|
|
}
|
|
});
|
|
|
|
ctx.redirect(LinearUtils.successUrl());
|
|
} catch (err) {
|
|
Logger.error("Encountered error during Linear OAuth callback", err);
|
|
ctx.redirect(LinearUtils.errorUrl("unknown"));
|
|
}
|
|
}
|
|
);
|
|
|
|
export default router;
|