Adding a popup authentication option.
All checks were successful
continuous-integration/drone/push Build is passing
All checks were successful
continuous-integration/drone/push Build is passing
This commit is contained in:
parent
2c4c87927d
commit
6b4ad81940
@ -6,6 +6,7 @@ steps:
|
|||||||
- name: Build with node
|
- name: Build with node
|
||||||
image: node:12
|
image: node:12
|
||||||
commands:
|
commands:
|
||||||
|
- npm config set registry https://npm.hibas123.de
|
||||||
- npm install
|
- npm install
|
||||||
- npm run install
|
- npm run install
|
||||||
- npm run build
|
- npm run build
|
||||||
@ -21,7 +22,7 @@ steps:
|
|||||||
registry: hibas123.azurecr.io
|
registry: hibas123.azurecr.io
|
||||||
debug: true
|
debug: true
|
||||||
when:
|
when:
|
||||||
branch: master
|
branch: [master]
|
||||||
event:
|
event:
|
||||||
exclude:
|
exclude:
|
||||||
- pull_request
|
- pull_request
|
||||||
|
12
Dockerfile
12
Dockerfile
@ -2,20 +2,12 @@ FROM node:12
|
|||||||
|
|
||||||
LABEL maintainer="Fabian Stamm <dev@fabianstamm.de>"
|
LABEL maintainer="Fabian Stamm <dev@fabianstamm.de>"
|
||||||
|
|
||||||
# RUN apt-get update
|
|
||||||
|
|
||||||
# # for https
|
|
||||||
# RUN apt-get install -yyq ca-certificates
|
|
||||||
# # install libraries
|
|
||||||
# RUN apt-get install -yyq libappindicator1 libasound2 libatk1.0-0 libc6 libcairo2 libcups2 libdbus-1-3 libexpat1 libfontconfig1 libgcc1 libgconf-2-4 libgdk-pixbuf2.0-0 libglib2.0-0 libgtk-3-0 libnspr4 libnss3 libpango-1.0-0 libpangocairo-1.0-0 libstdc++6 libx11-6 libx11-xcb1 libxcb1 libxcomposite1 libxcursor1 libxdamage1 libxext6 libxfixes3 libxi6 libxrandr2 libxrender1 libxss1 libxtst6
|
|
||||||
# # tools
|
|
||||||
# RUN apt-get install -yyq gconf-service lsb-release wget xdg-utils
|
|
||||||
# # and fonts
|
|
||||||
# RUN apt-get install -yyq fonts-liberation
|
|
||||||
|
|
||||||
RUN mkdir -p /usr/src/app
|
RUN mkdir -p /usr/src/app
|
||||||
WORKDIR /usr/src/app
|
WORKDIR /usr/src/app
|
||||||
|
|
||||||
|
RUN npm config set registry https://npm.hibas123.de
|
||||||
|
|
||||||
COPY ["package.json", "package-lock.json", "tsconfig.json", "/usr/src/app/"]
|
COPY ["package.json", "package-lock.json", "tsconfig.json", "/usr/src/app/"]
|
||||||
|
|
||||||
ENV NODE_ENV=production
|
ENV NODE_ENV=production
|
||||||
|
@ -38,5 +38,6 @@
|
|||||||
"Login token invalid": "Login token invalid",
|
"Login token invalid": "Login token invalid",
|
||||||
"No login token": "No login token",
|
"No login token": "No login token",
|
||||||
"You are not logged in or your login is expired (Login token invalid)": "You are not logged in or your login is expired (Login token invalid)",
|
"You are not logged in or your login is expired (Login token invalid)": "You are not logged in or your login is expired (Login token invalid)",
|
||||||
"You are not logged in or your login is expired (No special token)": "You are not logged in or your login is expired (No special token)"
|
"You are not logged in or your login is expired (No special token)": "You are not logged in or your login is expired (No special token)",
|
||||||
|
"Special token invalid": "Special token invalid"
|
||||||
}
|
}
|
@ -1,10 +1,11 @@
|
|||||||
import { Router } from "express";
|
import { Router } from "express";
|
||||||
import Register from "./register";
|
|
||||||
import Login from "./login";
|
|
||||||
import TwoFactorRoute from "./twofactor";
|
|
||||||
import { GetToken, DeleteToken } from "./token";
|
|
||||||
import { GetAccount } from "./account";
|
import { GetAccount } from "./account";
|
||||||
import { GetContactInfos } from "./contact";
|
import { GetContactInfos } from "./contact";
|
||||||
|
import { GetJWTByUser } from "./jwt";
|
||||||
|
import Login from "./login";
|
||||||
|
import Register from "./register";
|
||||||
|
import { DeleteToken, GetToken } from "./token";
|
||||||
|
import TwoFactorRoute from "./twofactor";
|
||||||
|
|
||||||
const UserRoute: Router = Router();
|
const UserRoute: Router = Router();
|
||||||
|
|
||||||
@ -125,4 +126,7 @@ UserRoute.get("/account", GetAccount);
|
|||||||
* @apiSuccess {Object[]} user.phone Phone numbers
|
* @apiSuccess {Object[]} user.phone Phone numbers
|
||||||
*/
|
*/
|
||||||
UserRoute.get("/contact", GetContactInfos);
|
UserRoute.get("/contact", GetContactInfos);
|
||||||
|
|
||||||
|
UserRoute.get("/jwt", GetJWTByUser);
|
||||||
|
|
||||||
export default UserRoute;
|
export default UserRoute;
|
||||||
|
37
src/api/user/jwt.ts
Normal file
37
src/api/user/jwt.ts
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
import { Request, Response } from "express";
|
||||||
|
import Stacker from "../middlewares/stacker";
|
||||||
|
import { GetUserMiddleware } from "../middlewares/user";
|
||||||
|
import { URL } from "url";
|
||||||
|
import Client from "../../models/client";
|
||||||
|
import RequestError, { HttpStatusCode } from "../../helper/request_error";
|
||||||
|
import { getAccessTokenJWT } from "../../helper/jwt";
|
||||||
|
|
||||||
|
export const GetJWTByUser = Stacker(
|
||||||
|
GetUserMiddleware(true, false),
|
||||||
|
async (req: Request, res: Response) => {
|
||||||
|
const { client_id, origin } = req.query as { [key: string]: string };
|
||||||
|
|
||||||
|
const client = await Client.findOne({
|
||||||
|
client_id,
|
||||||
|
});
|
||||||
|
|
||||||
|
const clientNotFoundError = new RequestError(
|
||||||
|
"Client not found!",
|
||||||
|
HttpStatusCode.BAD_REQUEST
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!client) throw clientNotFoundError;
|
||||||
|
|
||||||
|
const clientUrl = new URL(client.redirect_url);
|
||||||
|
|
||||||
|
if (clientUrl.hostname !== origin) throw clientNotFoundError;
|
||||||
|
|
||||||
|
const jwt = await getAccessTokenJWT({
|
||||||
|
user: req.user,
|
||||||
|
client: client,
|
||||||
|
permissions: [],
|
||||||
|
});
|
||||||
|
|
||||||
|
res.json({ jwt });
|
||||||
|
}
|
||||||
|
);
|
@ -62,7 +62,7 @@ export default async function TestData() {
|
|||||||
await Client.save(c);
|
await Client.save(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
let perm = await Permission.findOne({ id: 0 });
|
let perm = await Permission.findById("507f1f77bcf86cd799439011");
|
||||||
if (!perm) {
|
if (!perm) {
|
||||||
Logging.log("Adding test permission");
|
Logging.log("Adding test permission");
|
||||||
perm = Permission.new({
|
perm = Permission.new({
|
||||||
|
21
src/views/popup.ts
Normal file
21
src/views/popup.ts
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import * as handlebars from "handlebars";
|
||||||
|
import { readFileSync } from "fs";
|
||||||
|
import { __ as i__ } from "i18n";
|
||||||
|
import config from "../config";
|
||||||
|
|
||||||
|
let template: handlebars.TemplateDelegate<any>;
|
||||||
|
function loadStatic() {
|
||||||
|
let html = readFileSync("./views/out/popup/popup.html").toString();
|
||||||
|
template = handlebars.compile(html);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function GetPopupPage(__: typeof i__): string {
|
||||||
|
if (config.core.dev) {
|
||||||
|
loadStatic();
|
||||||
|
}
|
||||||
|
|
||||||
|
let data = {};
|
||||||
|
return template(data, { helpers: { i18n: __ } });
|
||||||
|
}
|
||||||
|
|
||||||
|
loadStatic();
|
@ -1,9 +1,9 @@
|
|||||||
import {
|
import {
|
||||||
IRouter,
|
IRouter,
|
||||||
Request,
|
Request,
|
||||||
|
RequestHandler,
|
||||||
Router,
|
Router,
|
||||||
static as ServeStatic,
|
static as ServeStatic,
|
||||||
RequestHandler,
|
|
||||||
} from "express";
|
} from "express";
|
||||||
import * as Handlebars from "handlebars";
|
import * as Handlebars from "handlebars";
|
||||||
import * as moment from "moment";
|
import * as moment from "moment";
|
||||||
@ -13,6 +13,7 @@ import config from "../config";
|
|||||||
import { HttpStatusCode } from "../helper/request_error";
|
import { HttpStatusCode } from "../helper/request_error";
|
||||||
import GetAdminPage from "./admin";
|
import GetAdminPage from "./admin";
|
||||||
import GetAuthPage from "./authorize";
|
import GetAuthPage from "./authorize";
|
||||||
|
import GetPopupPage from "./popup";
|
||||||
import GetRegistrationPage from "./register";
|
import GetRegistrationPage from "./register";
|
||||||
|
|
||||||
Handlebars.registerHelper("appname", () => config.core.name);
|
Handlebars.registerHelper("appname", () => config.core.name);
|
||||||
@ -68,33 +69,37 @@ ViewRouter.get(
|
|||||||
|
|
||||||
ViewRouter.get("/auth", GetAuthRoute(true));
|
ViewRouter.get("/auth", GetAuthRoute(true));
|
||||||
|
|
||||||
if (config.core.dev) {
|
ViewRouter.get("/popup", UserMiddleware, (req, res) => {
|
||||||
const logo =
|
res.send(GetPopupPage(req.__));
|
||||||
"";
|
|
||||||
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,
|
|
||||||
},
|
|
||||||
])
|
|
||||||
);
|
|
||||||
});
|
});
|
||||||
}
|
|
||||||
|
// 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;
|
export default ViewRouter;
|
||||||
|
24
views/src/popup/popup.hbs
Normal file
24
views/src/popup/popup.hbs
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<html>
|
||||||
|
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<title>Popup Auth</title>
|
||||||
|
<meta charset="utf8" />
|
||||||
|
<meta name="viewport" content="width=device-width,initial-scale=1" />
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<div class="card">
|
||||||
|
<div class="title">
|
||||||
|
<h1>Allow <span id="hostname" style="font-weight: bold;">LOADING...</span> to authenticate you?</h1>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<div style="text-align: right;">
|
||||||
|
<button onclick="allow()" class="mdc-button mdc-button--raised blue-button">Allow</button>
|
||||||
|
<button onclick="deny()" class="mdc-button mdc-button--raised blue-button">Deny</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
32
views/src/popup/popup.js
Normal file
32
views/src/popup/popup.js
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
async function getJWT(client_id, hostname) {
|
||||||
|
hostname = encodeURIComponent(hostname);
|
||||||
|
client_id = encodeURIComponent(client_id);
|
||||||
|
|
||||||
|
const res = await fetch(
|
||||||
|
`/api/user/jwt?client_id=${client_id}&origin=${hostname}`
|
||||||
|
).then((res) => res.json());
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
let acceptPromise;
|
||||||
|
|
||||||
|
window.allow = () => acceptPromise();
|
||||||
|
window.deny = () => window.close();
|
||||||
|
|
||||||
|
function start() {
|
||||||
|
let started = false;
|
||||||
|
window.addEventListener("message", async (msg) => {
|
||||||
|
if (!started) {
|
||||||
|
started = true;
|
||||||
|
const url = new URL(msg.origin);
|
||||||
|
document.getElementById("hostname").innerText = url.hostname;
|
||||||
|
await new Promise((yes) => (acceptPromise = yes));
|
||||||
|
const res = await getJWT(msg.data.client_id, url.hostname);
|
||||||
|
msg.source.postMessage(res, msg.origin);
|
||||||
|
window.close();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
start();
|
63
views/src/popup/popup.scss
Normal file
63
views/src/popup/popup.scss
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
@import "@material/button/mdc-button";
|
||||||
|
.blue-button {
|
||||||
|
background: #4a89dc !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grey-button {
|
||||||
|
background: #797979 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
// display: block;
|
||||||
|
// height: 1px;
|
||||||
|
border: 0;
|
||||||
|
border-top: 1px solid #b8b8b8;
|
||||||
|
// margin: 1em 0;
|
||||||
|
// padding: 0;
|
||||||
|
}
|
||||||
|
body {
|
||||||
|
font-family: Helvetica;
|
||||||
|
background: #eee;
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1,
|
||||||
|
h3 {
|
||||||
|
font-weight: 300;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
color: #636363;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul {
|
||||||
|
list-style: none;
|
||||||
|
padding-left: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.scope_title {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0;
|
||||||
|
padding-left: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.scope_description {
|
||||||
|
margin-top: 0;
|
||||||
|
padding-left: 15px;
|
||||||
|
font-size: 13px;
|
||||||
|
color: #202020;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card {
|
||||||
|
max-width: 480px;
|
||||||
|
margin: 4em auto;
|
||||||
|
padding: 3em 2em 2em 2em;
|
||||||
|
background: #fafafa;
|
||||||
|
border: 1px solid #ebebeb;
|
||||||
|
box-shadow: rgba(0, 0, 0, 0.14902) 0px 1px 1px 0px,
|
||||||
|
rgba(0, 0, 0, 0.09804) 0px 1px 2px 0px;
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user