Move Secrets home to its own page and move secrets pages into their own folder.
This commit is contained in:
parent
c71e530f3f
commit
db02e0e91b
|
@ -14,7 +14,7 @@ The resulting built files will be in `dist/`
|
||||||
Use whatever browser you want, I personally test on latest Microsoft Edge Dev build.
|
Use whatever browser you want, I personally test on latest Microsoft Edge Dev build.
|
||||||
```
|
```
|
||||||
npm install --save-dev
|
npm install --save-dev
|
||||||
WEBPACK_MODE=development BROWSER=google-chrome npx webpack serve
|
BROWSER=google-chrome npx webpack serve --config webpack-dev.config.js
|
||||||
```
|
```
|
||||||
|
|
||||||
## Translating
|
## Translating
|
||||||
|
|
|
@ -1,28 +1,29 @@
|
||||||
import { HomePage } from "./pages/Home";
|
import { HomePage } from "./pages/Home";
|
||||||
import { KeyValueDeletePage } from "./pages/KeyValue/KeyValueDelete";
|
import { KeyValueDeletePage } from "./pages/Secrets/KeyValue/KeyValueDelete";
|
||||||
import { KeyValueNewPage } from "./pages/KeyValue/KeyValueNew";
|
import { KeyValueNewPage } from "./pages/Secrets/KeyValue/KeyValueNew";
|
||||||
import { KeyValueSecretEditPage } from "./pages/KeyValue/KeyValueSecretsEdit";
|
import { KeyValueSecretEditPage } from "./pages/Secrets/KeyValue/KeyValueSecretsEdit";
|
||||||
import { KeyValueSecretPage } from "./pages/KeyValue/KeyValueSecret";
|
import { KeyValueSecretPage } from "./pages/Secrets/KeyValue/KeyValueSecret";
|
||||||
import { KeyValueVersionsPage } from "./pages/KeyValue/KeyValueVersions";
|
import { KeyValueVersionsPage } from "./pages/Secrets/KeyValue/KeyValueVersions";
|
||||||
import { KeyValueViewPage } from "./pages/KeyValue/KeyValueView";
|
import { KeyValueViewPage } from "./pages/Secrets/KeyValue/KeyValueView";
|
||||||
import { LoginPage } from "./pages/Login";
|
import { LoginPage } from "./pages/Login";
|
||||||
import { MePage } from "./pages/Me";
|
import { MePage } from "./pages/Me";
|
||||||
import { NewKVEnginePage } from "./pages/NewEngines/NewKVEngine";
|
import { NewKVEnginePage } from "./pages/Secrets/NewEngines/NewKVEngine";
|
||||||
import { NewSecretsEnginePage } from "./pages/NewSecretsEngine";
|
import { NewSecretsEnginePage } from "./pages/Secrets/NewSecretsEngine";
|
||||||
import { NewTOTPEnginePage } from "./pages/NewEngines/NewTOTPEngine";
|
import { NewTOTPEnginePage } from "./pages/Secrets/NewEngines/NewTOTPEngine";
|
||||||
import { NewTOTPPage } from "./pages/TOTP/NewTOTP";
|
import { NewTOTPPage } from "./pages/Secrets/TOTP/NewTOTP";
|
||||||
import { NewTransitEnginePage } from "./pages/NewEngines/NewTransitEngine";
|
import { NewTransitEnginePage } from "./pages/Secrets/NewEngines/NewTransitEngine";
|
||||||
import { NewTransitKeyPage } from "./pages/Transit/NewTransitKey";
|
import { NewTransitKeyPage } from "./pages/Secrets/Transit/NewTransitKey";
|
||||||
import { Page } from "./types/Page";
|
import { Page } from "./types/Page";
|
||||||
import { PwGenPage } from "./pages/PwGen";
|
import { PwGenPage } from "./pages/PwGen";
|
||||||
|
import { SecretsHomePage } from "./pages/Secrets/SecretsHome";
|
||||||
import { SetLanguagePage } from "./pages/SetLanguage";
|
import { SetLanguagePage } from "./pages/SetLanguage";
|
||||||
import { SetVaultURLPage } from "./pages/SetVaultURL";
|
import { SetVaultURLPage } from "./pages/SetVaultURL";
|
||||||
import { TOTPViewPage } from "./pages/TOTP/TOTPView";
|
import { TOTPViewPage } from "./pages/Secrets/TOTP/TOTPView";
|
||||||
import { TransitDecryptPage } from "./pages/Transit/TransitDecrypt";
|
import { TransitDecryptPage } from "./pages/Secrets/Transit/TransitDecrypt";
|
||||||
import { TransitEncryptPage } from "./pages/Transit/TransitEncrypt";
|
import { TransitEncryptPage } from "./pages/Secrets/Transit/TransitEncrypt";
|
||||||
import { TransitRewrapPage } from "./pages/Transit/TransitRewrap";
|
import { TransitRewrapPage } from "./pages/Secrets/Transit/TransitRewrap";
|
||||||
import { TransitViewPage } from "./pages/Transit/TransitView";
|
import { TransitViewPage } from "./pages/Secrets/Transit/TransitView";
|
||||||
import { TransitViewSecretPage } from "./pages/Transit/TransitViewSecret";
|
import { TransitViewSecretPage } from "./pages/Secrets/Transit/TransitViewSecret";
|
||||||
import { UnsealPage } from "./pages/Unseal";
|
import { UnsealPage } from "./pages/Unseal";
|
||||||
import { getObjectKeys } from "./utils";
|
import { getObjectKeys } from "./utils";
|
||||||
|
|
||||||
|
@ -36,6 +37,7 @@ type pagesList = {
|
||||||
|
|
||||||
export const allPages: pagesList = {
|
export const allPages: pagesList = {
|
||||||
HOME: new HomePage(),
|
HOME: new HomePage(),
|
||||||
|
SECRETS_HOME: new SecretsHomePage(),
|
||||||
ME: new MePage(),
|
ME: new MePage(),
|
||||||
TOTP: new TOTPViewPage(),
|
TOTP: new TOTPViewPage(),
|
||||||
NEW_TOTP: new NewTOTPPage(),
|
NEW_TOTP: new NewTOTPPage(),
|
||||||
|
|
|
@ -1,10 +1,8 @@
|
||||||
import { MountType, getMounts } from "../api/sys/getMounts";
|
|
||||||
import { Page } from "../types/Page";
|
import { Page } from "../types/Page";
|
||||||
import { getCapabilitiesPath } from "../api/sys/getCapabilities";
|
import { Tile } from "../elements/Tile";
|
||||||
import { lookupSelf } from "../api/sys/lookupSelf";
|
import { lookupSelf } from "../api/sys/lookupSelf";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { prePageChecks, setErrorText } from "../pageUtils";
|
import { prePageChecks, setErrorText } from "../pageUtils";
|
||||||
import { sortedObjectMap } from "../utils";
|
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class HomePage extends Page {
|
export class HomePage extends Page {
|
||||||
|
@ -25,14 +23,14 @@ export class HomePage extends Page {
|
||||||
tag: "li",
|
tag: "li",
|
||||||
children: makeElement({
|
children: makeElement({
|
||||||
tag: "span",
|
tag: "span",
|
||||||
html: i18next.t("vaulturl_text", { text: this.state.apiURL }),
|
html: i18next.t("home_vaulturl_text", { text: this.state.apiURL }),
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
makeElement({
|
makeElement({
|
||||||
tag: "li",
|
tag: "li",
|
||||||
children: makeElement({
|
children: makeElement({
|
||||||
tag: "a",
|
tag: "a",
|
||||||
text: i18next.t("password_generator_btn"),
|
text: i18next.t("home_password_generator_btn"),
|
||||||
onclick: async () => {
|
onclick: async () => {
|
||||||
await this.router.changePage("PW_GEN");
|
await this.router.changePage("PW_GEN");
|
||||||
},
|
},
|
||||||
|
@ -47,7 +45,7 @@ export class HomePage extends Page {
|
||||||
textList.appendChild(
|
textList.appendChild(
|
||||||
makeElement({
|
makeElement({
|
||||||
tag: "li",
|
tag: "li",
|
||||||
text: i18next.t("your_token_expires_in", {
|
text: i18next.t("home_your_token_expires_in", {
|
||||||
date: new Date(selfTokenInfo.expire_time),
|
date: new Date(selfTokenInfo.expire_time),
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
|
@ -61,74 +59,24 @@ export class HomePage extends Page {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const mountsCapabilities = await getCapabilitiesPath("/sys/mounts");
|
textList.appendChild(
|
||||||
if (mountsCapabilities.includes("sudo") && mountsCapabilities.includes("create")) {
|
makeElement({
|
||||||
textList.appendChild(
|
tag: "div",
|
||||||
makeElement({
|
class:
|
||||||
tag: "button",
|
"uk-child-width-1-1@s uk-child-width-1-2@m uk-grid-small uk-grid-match uk-margin-top",
|
||||||
text: i18next.t("home_new_secrets_engine_button"),
|
attributes: { "uk-grid": "" },
|
||||||
class: ["uk-button", "uk-button-primary", "uk-margin-top"],
|
children: [
|
||||||
onclick: async () => {
|
Tile({
|
||||||
await this.router.changePage("NEW_SECRETS_ENGINE");
|
title: i18next.t("home_secrets_title"),
|
||||||
},
|
description: i18next.t("home_secrets_description"),
|
||||||
}),
|
icon: "file-edit",
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.state.currentBaseMount = "";
|
|
||||||
this.state.currentSecretPath = [];
|
|
||||||
this.state.currentSecret = "";
|
|
||||||
this.state.currentSecretVersion = null;
|
|
||||||
|
|
||||||
const navList = makeElement({
|
|
||||||
tag: "ul",
|
|
||||||
class: ["uk-nav", "uk-nav-default", "uk-margin-top"],
|
|
||||||
});
|
|
||||||
homePageContent.appendChild(navList);
|
|
||||||
|
|
||||||
const mounts = await getMounts();
|
|
||||||
// sort it by secretPath so it's in alphabetical order consistantly.
|
|
||||||
const mountsMap = sortedObjectMap(mounts);
|
|
||||||
|
|
||||||
mountsMap.forEach((mount: MountType, baseMount) => {
|
|
||||||
if (typeof mount != "object") return;
|
|
||||||
if (mount == null) return;
|
|
||||||
if (!("type" in mount)) return;
|
|
||||||
if (!["kv", "totp", "transit", "cubbyhole"].includes(mount.type)) return;
|
|
||||||
|
|
||||||
const mountType = mount.type == "kv" ? "kv-v" + String(mount.options.version) : mount.type;
|
|
||||||
|
|
||||||
let linkText = "";
|
|
||||||
let linkPage: string;
|
|
||||||
if (mount.type == "kv") {
|
|
||||||
linkText = `K/V (v${mount.options.version}) - ${baseMount}`;
|
|
||||||
linkPage = "KEY_VALUE_VIEW";
|
|
||||||
} else if (mount.type == "totp") {
|
|
||||||
linkText = `TOTP - ${baseMount}`;
|
|
||||||
linkPage = "TOTP";
|
|
||||||
} else if (mount.type == "transit") {
|
|
||||||
linkText = `Transit - ${baseMount}`;
|
|
||||||
linkPage = "TRANSIT_VIEW";
|
|
||||||
} else if (mount.type == "cubbyhole") {
|
|
||||||
linkText = `Cubbyhole - ${baseMount}`;
|
|
||||||
linkPage = "KEY_VALUE_VIEW";
|
|
||||||
}
|
|
||||||
|
|
||||||
navList.appendChild(
|
|
||||||
makeElement({
|
|
||||||
tag: "li",
|
|
||||||
children: makeElement({
|
|
||||||
tag: "a",
|
|
||||||
text: linkText,
|
|
||||||
onclick: async () => {
|
onclick: async () => {
|
||||||
this.state.currentBaseMount = baseMount;
|
await this.router.changePage("SECRETS_HOME");
|
||||||
this.state.currentMountType = mountType;
|
|
||||||
await this.router.changePage(linkPage);
|
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
}),
|
],
|
||||||
);
|
}),
|
||||||
});
|
);
|
||||||
}
|
}
|
||||||
get name(): string {
|
get name(): string {
|
||||||
return i18next.t("home_page_title");
|
return i18next.t("home_page_title");
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { deleteSecret } from "../../api/kv/deleteSecret";
|
import { deleteSecret } from "../../../api/kv/deleteSecret";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { createOrUpdateSecret } from "../../api/kv/createOrUpdateSecret";
|
import { createOrUpdateSecret } from "../../../api/kv/createOrUpdateSecret";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class KeyValueNewPage extends Page {
|
export class KeyValueNewPage extends Page {
|
|
@ -1,11 +1,11 @@
|
||||||
import { CopyableInputBox } from "../../elements/CopyableInputBox";
|
import { CopyableInputBox } from "../../../elements/CopyableInputBox";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { getCapabilities } from "../../api/sys/getCapabilities";
|
import { getCapabilities } from "../../../api/sys/getCapabilities";
|
||||||
import { getSecret } from "../../api/kv/getSecret";
|
import { getSecret } from "../../../api/kv/getSecret";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { sortedObjectMap } from "../../utils";
|
import { sortedObjectMap } from "../../../utils";
|
||||||
import { undeleteSecret } from "../../api/kv/undeleteSecret";
|
import { undeleteSecret } from "../../../api/kv/undeleteSecret";
|
||||||
import Prism from "prismjs";
|
import Prism from "prismjs";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
import { CodeJar } from "codejar";
|
import { CodeJar } from "codejar";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { createOrUpdateSecret } from "../../api/kv/createOrUpdateSecret";
|
import { createOrUpdateSecret } from "../../../api/kv/createOrUpdateSecret";
|
||||||
import { getSecret } from "../../api/kv/getSecret";
|
import { getSecret } from "../../../api/kv/getSecret";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import { sortedObjectMap, verifyJSONString } from "../../utils";
|
import { sortedObjectMap, verifyJSONString } from "../../../utils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class KeyValueSecretEditPage extends Page {
|
export class KeyValueSecretEditPage extends Page {
|
|
@ -1,8 +1,8 @@
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { getSecretMetadata } from "../../api/kv/getSecretMetadata";
|
import { getSecretMetadata } from "../../../api/kv/getSecretMetadata";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { objectToMap } from "../../utils";
|
import { objectToMap } from "../../../utils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class KeyValueVersionsPage extends Page {
|
export class KeyValueVersionsPage extends Page {
|
|
@ -1,9 +1,9 @@
|
||||||
import { DoesNotExistError } from "../../types/internalErrors";
|
import { DoesNotExistError } from "../../../types/internalErrors";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { getSecrets } from "../../api/kv/getSecrets";
|
import { getSecrets } from "../../../api/kv/getSecrets";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class KeyValueViewPage extends Page {
|
export class KeyValueViewPage extends Page {
|
||||||
|
@ -15,7 +15,7 @@ export class KeyValueViewPage extends Page {
|
||||||
this.state.popCurrentSecretPath();
|
this.state.popCurrentSecretPath();
|
||||||
await this.router.changePage("KEY_VALUE_VIEW");
|
await this.router.changePage("KEY_VALUE_VIEW");
|
||||||
} else {
|
} else {
|
||||||
await this.router.changePage("HOME");
|
await this.router.changePage("SECRETS_HOME");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
async render(): Promise<void> {
|
async render(): Promise<void> {
|
|
@ -1,15 +1,18 @@
|
||||||
import { Margin } from "../../elements/Margin";
|
import { Margin } from "../../../elements/Margin";
|
||||||
import { Option } from "../../elements/Option";
|
import { Option } from "../../../elements/Option";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { newMount } from "../../api/sys/newMount";
|
import { newMount } from "../../../api/sys/newMount";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class NewKVEnginePage extends Page {
|
export class NewKVEnginePage extends Page {
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
}
|
}
|
||||||
|
async goBack(): Promise<void> {
|
||||||
|
await this.router.changePage("SECRETS_HOME");
|
||||||
|
}
|
||||||
async render(): Promise<void> {
|
async render(): Promise<void> {
|
||||||
const newEngineForm = makeElement({
|
const newEngineForm = makeElement({
|
||||||
tag: "form",
|
tag: "form",
|
|
@ -1,14 +1,17 @@
|
||||||
import { Margin } from "../../elements/Margin";
|
import { Margin } from "../../../elements/Margin";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { newMount } from "../../api/sys/newMount";
|
import { newMount } from "../../../api/sys/newMount";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class NewTOTPEnginePage extends Page {
|
export class NewTOTPEnginePage extends Page {
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
}
|
}
|
||||||
|
async goBack(): Promise<void> {
|
||||||
|
await this.router.changePage("SECRETS_HOME");
|
||||||
|
}
|
||||||
async render(): Promise<void> {
|
async render(): Promise<void> {
|
||||||
const newEngineForm = makeElement({
|
const newEngineForm = makeElement({
|
||||||
tag: "form",
|
tag: "form",
|
|
@ -1,14 +1,17 @@
|
||||||
import { Margin } from "../../elements/Margin";
|
import { Margin } from "../../../elements/Margin";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { newMount } from "../../api/sys/newMount";
|
import { newMount } from "../../../api/sys/newMount";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class NewTransitEnginePage extends Page {
|
export class NewTransitEnginePage extends Page {
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
}
|
}
|
||||||
|
async goBack(): Promise<void> {
|
||||||
|
await this.router.changePage("SECRETS_HOME");
|
||||||
|
}
|
||||||
async render(): Promise<void> {
|
async render(): Promise<void> {
|
||||||
const newEngineForm = makeElement({
|
const newEngineForm = makeElement({
|
||||||
tag: "form",
|
tag: "form",
|
|
@ -1,5 +1,5 @@
|
||||||
import { Page } from "../types/Page";
|
import { Page } from "../../types/Page";
|
||||||
import { Tile } from "../elements/Tile";
|
import { Tile } from "../../elements/Tile";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
95
src/pages/Secrets/SecretsHome.ts
Normal file
95
src/pages/Secrets/SecretsHome.ts
Normal file
|
@ -0,0 +1,95 @@
|
||||||
|
import { MountType, getMounts } from "../../api/sys/getMounts";
|
||||||
|
import { Page } from "../../types/Page";
|
||||||
|
import { getCapabilitiesPath } from "../../api/sys/getCapabilities";
|
||||||
|
import { makeElement } from "z-makeelement";
|
||||||
|
import { prePageChecks } from "../../pageUtils";
|
||||||
|
import { sortedObjectMap } from "../../utils";
|
||||||
|
import i18next from "i18next";
|
||||||
|
|
||||||
|
export class SecretsHomePage extends Page {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
async goBack(): Promise<void> {
|
||||||
|
await this.router.changePage("HOME");
|
||||||
|
}
|
||||||
|
async render(): Promise<void> {
|
||||||
|
await this.router.setPageContent("");
|
||||||
|
if (!(await prePageChecks(this.router))) return;
|
||||||
|
|
||||||
|
const homePageContent = makeElement({ tag: "div" });
|
||||||
|
await this.router.setPageContent(homePageContent);
|
||||||
|
|
||||||
|
const mountsCapabilities = await getCapabilitiesPath("/sys/mounts");
|
||||||
|
if (mountsCapabilities.includes("sudo") && mountsCapabilities.includes("create")) {
|
||||||
|
homePageContent.appendChild(
|
||||||
|
makeElement({
|
||||||
|
tag: "button",
|
||||||
|
text: i18next.t("secrets_home_new_secrets_engine_button"),
|
||||||
|
class: ["uk-button", "uk-button-primary"],
|
||||||
|
onclick: async () => {
|
||||||
|
await this.router.changePage("NEW_SECRETS_ENGINE");
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.state.currentBaseMount = "";
|
||||||
|
this.state.currentSecretPath = [];
|
||||||
|
this.state.currentSecret = "";
|
||||||
|
this.state.currentSecretVersion = null;
|
||||||
|
|
||||||
|
const navList = makeElement({
|
||||||
|
tag: "ul",
|
||||||
|
class: ["uk-nav", "uk-nav-default", "uk-margin-top"],
|
||||||
|
});
|
||||||
|
homePageContent.appendChild(navList);
|
||||||
|
|
||||||
|
const mounts = await getMounts();
|
||||||
|
// sort it by secretPath so it's in alphabetical order consistantly.
|
||||||
|
const mountsMap = sortedObjectMap(mounts);
|
||||||
|
|
||||||
|
mountsMap.forEach((mount: MountType, baseMount) => {
|
||||||
|
if (typeof mount != "object") return;
|
||||||
|
if (mount == null) return;
|
||||||
|
if (!("type" in mount)) return;
|
||||||
|
if (!["kv", "totp", "transit", "cubbyhole"].includes(mount.type)) return;
|
||||||
|
|
||||||
|
const mountType = mount.type == "kv" ? "kv-v" + String(mount.options.version) : mount.type;
|
||||||
|
|
||||||
|
let linkText = "";
|
||||||
|
let linkPage: string;
|
||||||
|
if (mount.type == "kv") {
|
||||||
|
linkText = `K/V (v${mount.options.version}) - ${baseMount}`;
|
||||||
|
linkPage = "KEY_VALUE_VIEW";
|
||||||
|
} else if (mount.type == "totp") {
|
||||||
|
linkText = `TOTP - ${baseMount}`;
|
||||||
|
linkPage = "TOTP";
|
||||||
|
} else if (mount.type == "transit") {
|
||||||
|
linkText = `Transit - ${baseMount}`;
|
||||||
|
linkPage = "TRANSIT_VIEW";
|
||||||
|
} else if (mount.type == "cubbyhole") {
|
||||||
|
linkText = `Cubbyhole - ${baseMount}`;
|
||||||
|
linkPage = "KEY_VALUE_VIEW";
|
||||||
|
}
|
||||||
|
|
||||||
|
navList.appendChild(
|
||||||
|
makeElement({
|
||||||
|
tag: "li",
|
||||||
|
children: makeElement({
|
||||||
|
tag: "a",
|
||||||
|
text: linkText,
|
||||||
|
onclick: async () => {
|
||||||
|
this.state.currentBaseMount = baseMount;
|
||||||
|
this.state.currentMountType = mountType;
|
||||||
|
await this.router.changePage(linkPage);
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
get name(): string {
|
||||||
|
return i18next.t("secrets_home_page_title");
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,10 +1,10 @@
|
||||||
import { Margin } from "../../elements/Margin";
|
import { Margin } from "../../../elements/Margin";
|
||||||
import { MarginInline } from "../../elements/MarginInline";
|
import { MarginInline } from "../../../elements/MarginInline";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { addNewTOTP } from "../../api/totp/addNewTOTP";
|
import { addNewTOTP } from "../../../api/totp/addNewTOTP";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
function replaceAll(str: string, replace: string, replaceWith: string): string {
|
function replaceAll(str: string, replace: string, replaceWith: string): string {
|
|
@ -1,12 +1,12 @@
|
||||||
import { CopyableInputBox } from "../../elements/CopyableInputBox";
|
import { CopyableInputBox } from "../../../elements/CopyableInputBox";
|
||||||
import { DoesNotExistError } from "../../types/internalErrors";
|
import { DoesNotExistError } from "../../../types/internalErrors";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { getTOTPCode } from "../../api/totp/getTOTPCode";
|
import { getTOTPCode } from "../../../api/totp/getTOTPCode";
|
||||||
import { getTOTPKeys } from "../../api/totp/getTOTPKeys";
|
import { getTOTPKeys } from "../../../api/totp/getTOTPKeys";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { objectToMap } from "../../utils";
|
import { objectToMap } from "../../../utils";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export interface TOTPListElement extends HTMLElement {
|
export interface TOTPListElement extends HTMLElement {
|
||||||
|
@ -23,6 +23,10 @@ export class TOTPViewPage extends Page {
|
||||||
refresher: number;
|
refresher: number;
|
||||||
totpListElements: Record<string, TOTPListElement>;
|
totpListElements: Record<string, TOTPListElement>;
|
||||||
|
|
||||||
|
async goBack(): Promise<void> {
|
||||||
|
await this.router.changePage("SECRETS_HOME");
|
||||||
|
}
|
||||||
|
|
||||||
async render(): Promise<void> {
|
async render(): Promise<void> {
|
||||||
const totpList = makeElement({ tag: "div" });
|
const totpList = makeElement({ tag: "div" });
|
||||||
await this.router.setPageContent(
|
await this.router.setPageContent(
|
|
@ -1,10 +1,10 @@
|
||||||
import { Margin } from "../../elements/Margin";
|
import { Margin } from "../../../elements/Margin";
|
||||||
import { Option } from "../../elements/Option";
|
import { Option } from "../../../elements/Option";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { newTransitKey } from "../../api/transit/newTransitKey";
|
import { newTransitKey } from "../../../api/transit/newTransitKey";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class NewTransitKeyPage extends Page {
|
export class NewTransitKeyPage extends Page {
|
|
@ -1,12 +1,12 @@
|
||||||
import { CopyableModal } from "../../elements/CopyableModal";
|
import { CopyableModal } from "../../../elements/CopyableModal";
|
||||||
import { FileUploadInput } from "../../elements/FileUploadInput";
|
import { FileUploadInput } from "../../../elements/FileUploadInput";
|
||||||
import { Margin } from "../../elements/Margin";
|
import { Margin } from "../../../elements/Margin";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { fileToBase64 } from "../../htmlUtils";
|
import { fileToBase64 } from "../../../htmlUtils";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import { transitDecrypt } from "../../api/transit/transitDecrypt";
|
import { transitDecrypt } from "../../../api/transit/transitDecrypt";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class TransitDecryptPage extends Page {
|
export class TransitDecryptPage extends Page {
|
|
@ -1,12 +1,12 @@
|
||||||
import { CopyableModal } from "../../elements/CopyableModal";
|
import { CopyableModal } from "../../../elements/CopyableModal";
|
||||||
import { FileUploadInput } from "../../elements/FileUploadInput";
|
import { FileUploadInput } from "../../../elements/FileUploadInput";
|
||||||
import { Margin } from "../../elements/Margin";
|
import { Margin } from "../../../elements/Margin";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { fileToBase64 } from "../../htmlUtils";
|
import { fileToBase64 } from "../../../htmlUtils";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import { transitEncrypt } from "../../api/transit/transitEncrypt";
|
import { transitEncrypt } from "../../../api/transit/transitEncrypt";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class TransitEncryptPage extends Page {
|
export class TransitEncryptPage extends Page {
|
|
@ -1,13 +1,13 @@
|
||||||
import { CopyableModal } from "../../elements/CopyableModal";
|
import { CopyableModal } from "../../../elements/CopyableModal";
|
||||||
import { Margin } from "../../elements/Margin";
|
import { Margin } from "../../../elements/Margin";
|
||||||
import { Option } from "../../elements/Option";
|
import { Option } from "../../../elements/Option";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { getTransitKey } from "../../api/transit/getTransitKey";
|
import { getTransitKey } from "../../../api/transit/getTransitKey";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { objectToMap } from "../../utils";
|
import { objectToMap } from "../../../utils";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import { transitRewrap } from "../../api/transit/transitRewrap";
|
import { transitRewrap } from "../../../api/transit/transitRewrap";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
type versionOption = { version: string; label: string };
|
type versionOption = { version: string; label: string };
|
|
@ -1,9 +1,9 @@
|
||||||
import { DoesNotExistError } from "../../types/internalErrors";
|
import { DoesNotExistError } from "../../../types/internalErrors";
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { getTransitKeys } from "../../api/transit/getTransitKeys";
|
import { getTransitKeys } from "../../../api/transit/getTransitKeys";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import { setErrorText } from "../../pageUtils";
|
import { setErrorText } from "../../../pageUtils";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
||||||
export class TransitViewPage extends Page {
|
export class TransitViewPage extends Page {
|
||||||
|
@ -12,7 +12,7 @@ export class TransitViewPage extends Page {
|
||||||
}
|
}
|
||||||
|
|
||||||
async goBack(): Promise<void> {
|
async goBack(): Promise<void> {
|
||||||
await this.router.changePage("HOME");
|
await this.router.changePage("SECRETS_HOME");
|
||||||
}
|
}
|
||||||
|
|
||||||
async render(): Promise<void> {
|
async render(): Promise<void> {
|
|
@ -1,7 +1,7 @@
|
||||||
import { Page } from "../../types/Page";
|
import { Page } from "../../../types/Page";
|
||||||
import { SecretTitleElement } from "../../elements/SecretTitleElement";
|
import { SecretTitleElement } from "../../../elements/SecretTitleElement";
|
||||||
import { Tile } from "../../elements/Tile";
|
import { Tile } from "../../../elements/Tile";
|
||||||
import { getTransitKey } from "../../api/transit/getTransitKey";
|
import { getTransitKey } from "../../../api/transit/getTransitKey";
|
||||||
import { makeElement } from "z-makeelement";
|
import { makeElement } from "z-makeelement";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
|
|
13
src/translations/en.js
vendored
13
src/translations/en.js
vendored
|
@ -34,10 +34,15 @@ module.exports = {
|
||||||
|
|
||||||
// Home Page
|
// Home Page
|
||||||
home_page_title: "Home",
|
home_page_title: "Home",
|
||||||
vaulturl_text: "Vault URL: {{text}}",
|
home_vaulturl_text: "Vault URL: {{text}}",
|
||||||
password_generator_btn: "Password Generator",
|
home_password_generator_btn: "Password Generator",
|
||||||
your_token_expires_in: "Your token expires in {{date, until_date}}",
|
home_your_token_expires_in: "Your token expires in {{date, until_date}}",
|
||||||
home_new_secrets_engine_button: "New Secrets Engine",
|
home_secrets_title: "Secrets",
|
||||||
|
home_secrets_description: "View, create and manage secrets.",
|
||||||
|
|
||||||
|
// Secrets Home Page
|
||||||
|
secrets_home_page_title: "Secrets",
|
||||||
|
secrets_home_new_secrets_engine_button: "New Secrets Engine",
|
||||||
|
|
||||||
// New Secrets Engine Page
|
// New Secrets Engine Page
|
||||||
new_secrets_engine_title: "New Secrets Engine",
|
new_secrets_engine_title: "New Secrets Engine",
|
||||||
|
|
6
src/translations/fr.js
vendored
6
src/translations/fr.js
vendored
|
@ -31,9 +31,9 @@ module.exports = {
|
||||||
|
|
||||||
// Home Page
|
// Home Page
|
||||||
home_page_title: "Accueil",
|
home_page_title: "Accueil",
|
||||||
vaulturl_text: "Adresse du Vault: {{text}}",
|
home_vaulturl_text: "Adresse du Vault: {{text}}",
|
||||||
password_generator_btn: "Générateur de mot de passe",
|
home_password_generator_btn: "Générateur de mot de passe",
|
||||||
your_token_expires_in: "Votre jeton expire dans {{date, until_date}}",
|
home_your_token_expires_in: "Votre jeton expire dans {{date, until_date}}",
|
||||||
|
|
||||||
// Unseal Page
|
// Unseal Page
|
||||||
unseal_vault_text: "Ouvrir le Vault",
|
unseal_vault_text: "Ouvrir le Vault",
|
||||||
|
|
6
src/translations/ru.js
vendored
6
src/translations/ru.js
vendored
|
@ -34,9 +34,9 @@ module.exports = {
|
||||||
|
|
||||||
// Home Page
|
// Home Page
|
||||||
home_page_title: "Главная страница",
|
home_page_title: "Главная страница",
|
||||||
vaulturl_text: "Адрес хранилища: {{text}}",
|
home_vaulturl_text: "Адрес хранилища: {{text}}",
|
||||||
password_generator_btn: "Генератор паролей",
|
home_password_generator_btn: "Генератор паролей",
|
||||||
your_token_expires_in: "Продолжительность ключа: {{date, until_date}}",
|
home_your_token_expires_in: "Продолжительность ключа: {{date, until_date}}",
|
||||||
|
|
||||||
// Unseal Page
|
// Unseal Page
|
||||||
unseal_vault_text: "Раскрыть хранилище",
|
unseal_vault_text: "Раскрыть хранилище",
|
||||||
|
|
60
webpack-dev.config.js
Normal file
60
webpack-dev.config.js
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
const path = require('path');
|
||||||
|
const webpack = require('webpack');
|
||||||
|
var os = require("os");
|
||||||
|
|
||||||
|
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
|
||||||
|
const HtmlWebpackPlugin = require('html-webpack-plugin');
|
||||||
|
const { GitRevisionPlugin } = require('git-revision-webpack-plugin');
|
||||||
|
const gitRevisionPlugin = new GitRevisionPlugin();
|
||||||
|
let commitHash = gitRevisionPlugin.commithash();
|
||||||
|
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
mode: "development",
|
||||||
|
cache: true,
|
||||||
|
entry: './src/main.ts',
|
||||||
|
output: {
|
||||||
|
path: path.resolve(__dirname, 'dist'),
|
||||||
|
filename: 'bundle.js',
|
||||||
|
},
|
||||||
|
stats: {
|
||||||
|
colors: true,
|
||||||
|
timings: true,
|
||||||
|
},
|
||||||
|
plugins: [
|
||||||
|
new MiniCssExtractPlugin(),
|
||||||
|
new HtmlWebpackPlugin({ title: "VaultUI" }),
|
||||||
|
new webpack.DefinePlugin({
|
||||||
|
BUILD_STRING:
|
||||||
|
JSON.stringify(`Built At: ${new Date().toString()} by ${os.userInfo().username}@${os.hostname()} on commit ${commitHash}`),
|
||||||
|
})
|
||||||
|
],
|
||||||
|
devServer: {
|
||||||
|
open: process.env.BROWSER || "microsoft-edge-dev",
|
||||||
|
},
|
||||||
|
resolve: {
|
||||||
|
modules: ['node_modules'],
|
||||||
|
extensions: ['.tsx', '.ts', '.js', ".mjs"],
|
||||||
|
},
|
||||||
|
|
||||||
|
module: {
|
||||||
|
rules: [
|
||||||
|
{
|
||||||
|
test: /\.(sa|sc|c)ss$/,
|
||||||
|
use: [
|
||||||
|
MiniCssExtractPlugin.loader,
|
||||||
|
"css-loader",
|
||||||
|
"sass-loader"
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
test: /\.ts(x?)$/,
|
||||||
|
use: [
|
||||||
|
{
|
||||||
|
loader: 'ts-loader'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
};
|
|
@ -28,24 +28,17 @@ var babelOptions = {
|
||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
const VERBOSE = Object.getOwnPropertyNames(process.env).includes("VERBOSE") || !true;
|
|
||||||
const MODE = process.env.WEBPACK_MODE || "production"
|
|
||||||
const DEBUG = MODE != "production";
|
|
||||||
|
|
||||||
let commitHash = gitRevisionPlugin.commithash();
|
let commitHash = gitRevisionPlugin.commithash();
|
||||||
|
|
||||||
console.log("DEBUG:", DEBUG);
|
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
mode: MODE,
|
mode: "production",
|
||||||
cache: DEBUG,
|
cache: false,
|
||||||
entry: './src/main.ts',
|
entry: './src/main.ts',
|
||||||
output: {
|
output: {
|
||||||
path: path.resolve(__dirname, 'dist'),
|
path: path.resolve(__dirname, 'dist'),
|
||||||
filename: 'bundle.js',
|
filename: 'bundle.js',
|
||||||
},
|
},
|
||||||
stats: {
|
stats: {
|
||||||
preset: VERBOSE ? "detailed" : "normal",
|
|
||||||
colors: true,
|
colors: true,
|
||||||
timings: true,
|
timings: true,
|
||||||
},
|
},
|
||||||
|
|
Loading…
Reference in a new issue