diff --git a/src/api/oauth/auth.ts b/src/api/oauth/auth.ts index dbcce8d..ec513c3 100644 --- a/src/api/oauth/auth.ts +++ b/src/api/oauth/auth.ts @@ -12,6 +12,8 @@ import { ObjectID } from "bson"; const AuthRoute = Stacker(GetUserMiddleware(true), async (req: Request, res: Response) => { let { response_type, client_id, redirect_uri, scope, state, nored } = req.query; const sendError = (type) => { + if (redirect_uri === "$local") + redirect_uri = "/code"; res.redirect(redirect_uri += `?error=${type}&state=${state}`); } /** @@ -63,7 +65,9 @@ const AuthRoute = Stacker(GetUserMiddleware(true), async (req: Request, res: Res }); await ClientCode.save(code); - let ruri = client.redirect_url + `?code=${code.code}&state=${state}`; + let redir = client.redirect_url === "$local" ? "/code" : client.redirect_url; + + let ruri = redir + `?code=${code.code}&state=${state}`; if (nored === "true") { res.json({ redirect_uri: ruri diff --git a/src/api/oauth/refresh.ts b/src/api/oauth/refresh.ts index ccea5ce..84d6648 100644 --- a/src/api/oauth/refresh.ts +++ b/src/api/oauth/refresh.ts @@ -13,12 +13,24 @@ import { JWTExpDur } from "../../keys"; import RefreshToken from "../../models/refresh_token"; import { getEncryptionKey } from "../../helper/user_key"; +// TODO: +/* +For example, the authorization server could employ refresh token +rotation in which a new refresh token is issued with every access +token refresh response. The previous refresh token is invalidated but retained by the authorization server. If a refresh token is +compromised and subsequently used by both the attacker and the +legitimate client, one of them will present an invalidated refresh +token, which will inform the authorization server of the breach. +*/ + +const refreshTokenValidTime = moment.duration(6, "month"); + const RefreshTokenRoute = Stacker(GetClientAuthMiddleware(false, false, true), async (req: Request, res: Response) => { let grant_type = req.query.grant_type || req.body.grant_type; if (!grant_type || grant_type === "authorization_code") { let code = req.query.code || req.body.code; let c = await ClientCode.findOne({ code: code }) - if (!c) { + if (!c || moment(c.validTill).isBefore()) { throw new RequestError(req.__("Invalid code"), HttpStatusCode.BAD_REQUEST); } @@ -33,7 +45,7 @@ const RefreshTokenRoute = Stacker(GetClientAuthMiddleware(false, false, true), a permissions: c.permissions, token: randomBytes(16).toString("hex"), valid: true, - validTill: moment().add(6, "months").toDate() + validTill: moment().add(refreshTokenValidTime).toDate() }); await RefreshToken.save(token); await ClientCode.delete(c); @@ -63,7 +75,11 @@ const RefreshTokenRoute = Stacker(GetClientAuthMiddleware(false, false, true), a if (!refresh_token) throw new RequestError(req.__("refresh_token not set"), HttpStatusCode.BAD_REQUEST); let token = await RefreshToken.findOne({ token: refresh_token }); - if (!token) throw new RequestError(req.__("Invalid token"), HttpStatusCode.BAD_REQUEST); + if (!token || !token.valid || moment(token.validTill).isBefore()) + throw new RequestError(req.__("Invalid token"), HttpStatusCode.BAD_REQUEST); + + token.validTill = moment().add(refreshTokenValidTime).toDate(); + await RefreshToken.save(token); let user = await User.findById(token.user); let client = await Client.findById(token.client) diff --git a/src/views/views.ts b/src/views/views.ts index f3f1541..104966e 100644 --- a/src/views/views.ts +++ b/src/views/views.ts @@ -1,111 +1,119 @@ -import { Router, IRouter, Request } from "express" -import GetLoginPage from "./login"; -import GetAuthPage from "./authorize"; -import promiseMiddleware from "../helper/promiseMiddleware"; -import config from "../config"; -import * as Handlebars from "handlebars"; -import GetRegistrationPage from "./register"; -import GetAdminPage from "./admin"; -import { HttpStatusCode } from "../helper/request_error"; -import * as moment from "moment"; -import Permission, { IPermission } from "../models/permissions"; -import Client from "../models/client"; -import { Logging } from "@hibas123/nodelogging"; -import Stacker from "../api/middlewares/stacker"; -import { UserMiddleware, GetUserMiddleware } from "../api/middlewares/user"; -// import GetUserPage from "./user"; - -Handlebars.registerHelper("appname", () => config.core.name); - -const cacheTime = config.core.dev ? moment.duration(1, "month").asSeconds() : 10; - -const ViewRouter: IRouter = Router(); -ViewRouter.get("/", UserMiddleware, (req, res) => { - res.send("This is the main page") -}) - -ViewRouter.get("/register", (req, res) => { - res.setHeader("Cache-Control", "public, max-age=" + cacheTime); - res.send(GetRegistrationPage(req.__)); -}) - -ViewRouter.get("/login", (req, res) => { - res.setHeader("Cache-Control", "public, max-age=" + cacheTime); - res.send(GetLoginPage(req.__)) -}) - -ViewRouter.get("/admin", GetUserMiddleware(false, true), (req: Request, res, next) => { - if (!req.isAdmin) res.sendStatus(HttpStatusCode.FORBIDDEN) - else next() -}, (req, res) => { - res.send(GetAdminPage(req.__)) -}) - -// ViewRouter.get("/user", Stacker(GetUserMiddleware(false, true), (req, res) => { -// res.setHeader("Cache-Control", "public, max-age=" + cacheTime); -// res.send(GetUserPage(req.__)); -// })); - -ViewRouter.get("/auth", Stacker(GetUserMiddleware(false, true), async (req, res) => { - let { scope, redirect_uri, state, client_id }: { [key: string]: string } = req.query; - const sendError = (type) => { - res.redirect(redirect_uri += `?error=${type}&state=${state}`); - } - let client = await Client.findOne({ client_id: client_id }) - if (!client) { - return sendError("unauthorized_client") - } - - let permissions: IPermission[] = []; - let proms: PromiseLike[] = []; - if (scope) { - for (let perm of scope.split(";").filter(e => e !== "read_user")) { - proms.push(Permission.findById(perm).then(p => { - if (!p) return Promise.reject(new Error()); - permissions.push(p); - })); - } - } - let err = false; - await Promise.all(proms).catch(e => { - err = true; - }) - Logging.debug(err); - if (err) { - return sendError("invalid_scope") - } - let scopes = await Promise.all(permissions.map(async perm => { - let client = await Client.findById(perm.client); - return { - name: perm.name, - description: perm.description, - logo: client.logo - } - })) - res.send(GetAuthPage(req.__, client.name, scopes)); -})); - -if (config.core.dev) { - const logo = "" - ViewRouter.get("/devauth", (req, res) => { - res.send(GetAuthPage(req.__, "Test 05265", [ - { - name: "Access Profile", - description: "It allows the application to know who you are. Required for all applications. And a lot of more Text, because why not? This will not stop, till it is multiple lines long and maybe kill the layout, so keep reading as long as you like, but I promise it will get boring after some time. So this should be enougth.", - logo: logo - }, - { - name: "Test 1", - description: "This is not an real permission. This is used just to verify the layout", - logo: logo - }, - { - name: "Test 2", - description: "This is not an real permission. This is used just to verify the layout", - logo: logo - } - ])) - }) -} - +import { Router, IRouter, Request } from "express" +import GetLoginPage from "./login"; +import GetAuthPage from "./authorize"; +import promiseMiddleware from "../helper/promiseMiddleware"; +import config from "../config"; +import * as Handlebars from "handlebars"; +import GetRegistrationPage from "./register"; +import GetAdminPage from "./admin"; +import { HttpStatusCode } from "../helper/request_error"; +import * as moment from "moment"; +import Permission, { IPermission } from "../models/permissions"; +import Client from "../models/client"; +import { Logging } from "@hibas123/nodelogging"; +import Stacker from "../api/middlewares/stacker"; +import { UserMiddleware, GetUserMiddleware } from "../api/middlewares/user"; +// import GetUserPage from "./user"; + +Handlebars.registerHelper("appname", () => config.core.name); + +const cacheTime = config.core.dev ? moment.duration(1, "month").asSeconds() : 10; + +const ViewRouter: IRouter = Router(); +ViewRouter.get("/", UserMiddleware, (req, res) => { + res.send("This is the main page") +}) + +ViewRouter.get("/register", (req, res) => { + res.setHeader("Cache-Control", "public, max-age=" + cacheTime); + res.send(GetRegistrationPage(req.__)); +}) + +ViewRouter.get("/login", (req, res) => { + res.setHeader("Cache-Control", "public, max-age=" + cacheTime); + res.send(GetLoginPage(req.__)) +}) + +ViewRouter.get("/code", (req, res) => { + res.setHeader("Cache-Control", "no-cache"); + if (req.query.error) + res.send("Some error occured: " + req.query.error); + else + res.send(`Your code is: ${req.query.code}`); +}) + +ViewRouter.get("/admin", GetUserMiddleware(false, true), (req: Request, res, next) => { + if (!req.isAdmin) res.sendStatus(HttpStatusCode.FORBIDDEN) + else next() +}, (req, res) => { + res.send(GetAdminPage(req.__)) +}) + +// ViewRouter.get("/user", Stacker(GetUserMiddleware(false, true), (req, res) => { +// res.setHeader("Cache-Control", "public, max-age=" + cacheTime); +// res.send(GetUserPage(req.__)); +// })); + +ViewRouter.get("/auth", Stacker(GetUserMiddleware(false, true), async (req, res) => { + let { scope, redirect_uri, state, client_id }: { [key: string]: string } = req.query; + const sendError = (type) => { + res.redirect(redirect_uri += `?error=${type}&state=${state}`); + } + let client = await Client.findOne({ client_id: client_id }) + if (!client) { + return sendError("unauthorized_client") + } + + let permissions: IPermission[] = []; + let proms: PromiseLike[] = []; + if (scope) { + for (let perm of scope.split(";").filter(e => e !== "read_user")) { + proms.push(Permission.findById(perm).then(p => { + if (!p) return Promise.reject(new Error()); + permissions.push(p); + })); + } + } + let err = false; + await Promise.all(proms).catch(e => { + err = true; + }) + Logging.debug(err); + if (err) { + return sendError("invalid_scope") + } + let scopes = await Promise.all(permissions.map(async perm => { + let client = await Client.findById(perm.client); + return { + name: perm.name, + description: perm.description, + logo: client.logo + } + })) + res.send(GetAuthPage(req.__, client.name, scopes)); +})); + +if (config.core.dev) { + const logo = "" + ViewRouter.get("/devauth", (req, res) => { + res.send(GetAuthPage(req.__, "Test 05265", [ + { + name: "Access Profile", + description: "It allows the application to know who you are. Required for all applications. And a lot of more Text, because why not? This will not stop, till it is multiple lines long and maybe kill the layout, so keep reading as long as you like, but I promise it will get boring after some time. So this should be enougth.", + logo: logo + }, + { + name: "Test 1", + description: "This is not an real permission. This is used just to verify the layout", + logo: logo + }, + { + name: "Test 2", + description: "This is not an real permission. This is used just to verify the layout", + logo: logo + } + ])) + }) +} + export default ViewRouter; \ No newline at end of file diff --git a/views/src/authorize/authorize.js b/views/src/authorize/authorize.js index 8c6ccb2..248682c 100644 --- a/views/src/authorize/authorize.js +++ b/views/src/authorize/authorize.js @@ -7,9 +7,12 @@ function submit() { document.getElementById("cancel").onclick = () => { let u = new URL(window.location); let uri = u.searchParams.get("redirect_uri"); + if (uri === "$local") { + uri = "/code"; + } window.location.href = uri + "?error=access_denied&state=" + u.searchParams.get("state"); } document.getElementById("allow").onclick = () => { submit() -} +} \ No newline at end of file