* chore: Separate signin/auth middleware fix: Email signin token parsed by JWT middleware fix: Email signin marked as active when logging in as suspended fix: Suspended email signin correctly redirected to login screen closes #1740 * refactor middleware -> lib * lint
97 lines
2.5 KiB
JavaScript
97 lines
2.5 KiB
JavaScript
// @flow
|
|
import { AuthenticationError, UserSuspendedError } from "../errors";
|
|
import { User, Team, ApiKey } from "../models";
|
|
import type { ContextWithState } from "../types";
|
|
import { getUserForJWT } from "../utils/jwt";
|
|
|
|
export default function auth(options?: { required?: boolean } = {}) {
|
|
return async function authMiddleware(
|
|
ctx: ContextWithState,
|
|
next: () => Promise<mixed>
|
|
) {
|
|
let token;
|
|
|
|
const authorizationHeader = ctx.request.get("authorization");
|
|
if (authorizationHeader) {
|
|
const parts = authorizationHeader.split(" ");
|
|
if (parts.length === 2) {
|
|
const scheme = parts[0];
|
|
const credentials = parts[1];
|
|
|
|
if (/^Bearer$/i.test(scheme)) {
|
|
token = credentials;
|
|
}
|
|
} else {
|
|
throw new AuthenticationError(
|
|
`Bad Authorization header format. Format is "Authorization: Bearer <token>"`
|
|
);
|
|
}
|
|
} else if (ctx.body && ctx.body.token) {
|
|
token = ctx.body.token;
|
|
} else if (ctx.request.query.token) {
|
|
token = ctx.request.query.token;
|
|
} else {
|
|
token = ctx.cookies.get("accessToken");
|
|
}
|
|
|
|
if (!token && options.required !== false) {
|
|
throw new AuthenticationError("Authentication required");
|
|
}
|
|
|
|
let user;
|
|
if (token) {
|
|
if (String(token).match(/^[\w]{38}$/)) {
|
|
ctx.state.authType = "api";
|
|
|
|
let apiKey;
|
|
try {
|
|
apiKey = await ApiKey.findOne({
|
|
where: {
|
|
secret: token,
|
|
},
|
|
});
|
|
} catch (err) {
|
|
throw new AuthenticationError("Invalid API key");
|
|
}
|
|
|
|
if (!apiKey) {
|
|
throw new AuthenticationError("Invalid API key");
|
|
}
|
|
|
|
user = await User.findByPk(apiKey.userId, {
|
|
include: [
|
|
{
|
|
model: Team,
|
|
as: "team",
|
|
required: true,
|
|
},
|
|
],
|
|
});
|
|
if (!user) {
|
|
throw new AuthenticationError("Invalid API key");
|
|
}
|
|
} else {
|
|
ctx.state.authType = "app";
|
|
|
|
user = await getUserForJWT(String(token));
|
|
}
|
|
|
|
if (user.isSuspended) {
|
|
const suspendingAdmin = await User.findOne({
|
|
where: { id: user.suspendedById },
|
|
paranoid: false,
|
|
});
|
|
throw new UserSuspendedError({ adminEmail: suspendingAdmin.email });
|
|
}
|
|
|
|
// not awaiting the promise here so that the request is not blocked
|
|
user.updateActiveAt(ctx.request.ip);
|
|
|
|
ctx.state.token = String(token);
|
|
ctx.state.user = user;
|
|
}
|
|
|
|
return next();
|
|
};
|
|
}
|