js-spectre @main -
refs -
log -
-
https://git.jolheiser.com/js-spectre.git
JS implementation for spectre/masterpassword
lint and add readme
Signed-off-by: jolheiser <git@jolheiser.com>
Signature
-----BEGIN SSH SIGNATURE-----
U1NIU0lHAAAAAQAAADMAAAALc3NoLWVkMjU1MTkAAAAgBTEvCQk6VqUAdN2RuH6bj1dNkY
oOpbPWj+jw4ua1B1cAAAADZ2l0AAAAAAAAAAZzaGE1MTIAAABTAAAAC3NzaC1lZDI1NTE5
AAAAQG9hCdpvXkICHJvP0GjrsKEk/FTW3ujGmvEUG2/wPbvWa3rCKglq2fhjr1x26cb84E
WJ/o92uXGGT6Dv7P4fJAo=
-----END SSH SIGNATURE-----
7 changed files, 67 additions(+), 35 deletions(-)
diff --git a/README.md b/README.md
new file mode 100644
index 0000000000000000000000000000000000000000..334d8b2252afc826386347459ca8b8b21ca402bb
--- /dev/null
+++ b/README.md
@@ -0,0 +1,13 @@
+# js-spectre
+
+JavaScript implementation of [Spectre](https://spectre.app).
+
+## Code Quality
+
+1. `npm run lint`
+2. `npm run lint:fix`
+3. `npm run test`
+
+## License
+
+[GPLv3](LICENSE) - Same as original algorithm
\ No newline at end of file
diff --git a/eslint.config.js b/eslint.config.js
index 25170ccfd3f103c027968d9c64414d29ad81c016..143cd670e6e25d7469996aa1e41eb67e08a9958b 100644
--- a/eslint.config.js
+++ b/eslint.config.js
@@ -3,7 +3,7 @@ import tseslint from "typescript-eslint"
import stylistic from "@stylistic/eslint-plugin"
export default tseslint.config(
- { ignores: ["dist/"] },
+ { ignores: ["dist/", "spectre.js"] },
eslint.configs.recommended,
tseslint.configs.recommendedTypeChecked,
diff --git a/example/index.html b/example/index.html
index 80a669ff950e50b67dfadcb13172f75e5c066b7d..0fade851bbc4c830f553a9a29bbe0e62d5a7840a 100644
--- a/example/index.html
+++ b/example/index.html
@@ -1,4 +1,4 @@
-<!DOCTYPE html>
+<!doctype html>
<head>
<script src="../spectre.js"></script>
@@ -12,7 +12,11 @@ </head>
<body>
<input id="userKey" type="text" placeholder="Robert Lee Mitchell" /><br />
- <input id="userSecret" type="text" placeholder="banana colored duckling" /><br />
+ <input
+ id="userSecret"
+ type="text"
+ placeholder="banana colored duckling"
+ /><br />
<input id="siteKey" type="text" placeholder="masterpasswordapp.com" /><br />
<strong id="result"></strong>
</body>
@@ -39,4 +43,4 @@ if (s === null || $siteKey.value === "") return;
const pw = s.site($siteKey.value);
$result.innerText = pw;
}
-</script>
\ No newline at end of file
+</script>
diff --git a/example/sakura.css b/example/sakura.css
index 1297f86ddf91e9e5c9a2e781321c4ddfe796a818..53b47996dc0901c6b6ce7bead96da85b55cacbe5 100644
--- a/example/sakura.css
+++ b/example/sakura.css
@@ -6,7 +6,8 @@ */
/* Body */
html {
font-size: 62.5%;
- font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif;
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto,
+ "Helvetica Neue", Arial, "Noto Sans", sans-serif;
}
body {
@@ -38,7 +39,8 @@ h4,
h5,
h6 {
line-height: 1.1;
- font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif;
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto,
+ "Helvetica Neue", Arial, "Noto Sans", sans-serif;
font-weight: 700;
margin-top: 3rem;
margin-bottom: 1.5rem;
@@ -155,7 +157,7 @@ background-color: #f1f1f1;
white-space: pre-wrap;
}
-pre>code {
+pre > code {
padding: 0;
background-color: transparent;
white-space: pre;
@@ -193,10 +195,10 @@ }
.button,
button,
-input[type=submit],
-input[type=reset],
-input[type=button],
-input[type=file]::file-selector-button {
+input[type="submit"],
+input[type="reset"],
+input[type="button"],
+input[type="file"]::file-selector-button {
display: inline-block;
padding: 5px 10px;
text-align: center;
@@ -212,20 +214,20 @@ }
.button[disabled],
button[disabled],
-input[type=submit][disabled],
-input[type=reset][disabled],
-input[type=button][disabled],
-input[type=file]::file-selector-button[disabled] {
+input[type="submit"][disabled],
+input[type="reset"][disabled],
+input[type="button"][disabled],
+input[type="file"]::file-selector-button[disabled] {
cursor: default;
opacity: 0.5;
}
.button:hover,
button:hover,
-input[type=submit]:hover,
-input[type=reset]:hover,
-input[type=button]:hover,
-input[type=file]::file-selector-button:hover {
+input[type="submit"]:hover,
+input[type="reset"]:hover,
+input[type="button"]:hover,
+input[type="file"]::file-selector-button:hover {
background-color: #982c61;
color: #f9f9f9;
outline: 0;
@@ -233,10 +235,10 @@ }
.button:focus-visible,
button:focus-visible,
-input[type=submit]:focus-visible,
-input[type=reset]:focus-visible,
-input[type=button]:focus-visible,
-input[type=file]::file-selector-button:focus-visible {
+input[type="submit"]:focus-visible,
+input[type="reset"]:focus-visible,
+input[type="button"]:focus-visible,
+input[type="file"]::file-selector-button:focus-visible {
outline-style: solid;
outline-width: 2px;
}
@@ -262,7 +264,7 @@ border: 1px solid #1d7484;
outline: 0;
}
-input[type=checkbox]:focus {
+input[type="checkbox"]:focus {
outline: 1px dotted #1d7484;
}
@@ -272,4 +274,4 @@ fieldset {
display: block;
margin-bottom: 0.5rem;
font-weight: 600;
-}
\ No newline at end of file
+}
diff --git a/lib/spectre.ts b/lib/spectre.ts
index 85880b156b652b2abf36f61015ac25aca7aaf258..ac27e48871ed64b6686e0980a1bef7b413e049af 100644
--- a/lib/spectre.ts
+++ b/lib/spectre.ts
@@ -5,10 +5,10 @@ import { sha256 } from "@noble/hashes/sha256"
import { characters, Template, templates } from "./template.js"
export class Spectre {
- private name: string
- private secret: string
- private scoper: Scoper
- private key: Uint8Array
+ private readonly name: string
+ private readonly secret: string
+ private readonly scoper: Scoper
+ private readonly key: Uint8Array
constructor(name: string, secret: string, scoper: Scoper = defaultScoper) {
this.name = name
diff --git a/package.json b/package.json
index 6da8e58952caa113a9f123ef868edfd33b4aa883..00aaa59de6c2612634d950dff971b22aea92d431 100644
--- a/package.json
+++ b/package.json
@@ -14,6 +14,7 @@ "description": "JS implementation of Spectre",
"scripts": {
"test": "vitest --run",
"lint": "eslint .",
+ "lint:fix": "npm run lint -- --fix",
"build:tsc": "tsc -p tsconfig.build.json",
"build:esbuild": "esbuild --target=es2022 --bundle --minify --outfile=spectre.js --global-name=spectre spectre",
"build": "npm run build:tsc && npm run build:esbuild"
@@ -40,4 +41,4 @@ },
"dependencies": {
"@noble/hashes": "^1.7.1"
}
-}
\ No newline at end of file
+}
diff --git a/tests/index.test.ts b/tests/index.test.ts
index 6a7acbbfeceb4daa90b569a8b6f5753f7415e793..41d23cbd55ac284380927c32f7307c57354efa51 100644
--- a/tests/index.test.ts
+++ b/tests/index.test.ts
@@ -14,14 +14,18 @@ assert.equal(pw, "Jejr5[RepuSosp")
})
it("should generate the custom example password", () => {
const scoper = new SimpleScoper("com.jojodev.jolheiser")
- const s = new Spectre("Robert Lee Mitchell", "banana colored duckling", scoper)
+ const s = new Spectre(
+ "Robert Lee Mitchell",
+ "banana colored duckling",
+ scoper,
+ )
const pw = s.site("jojodev.com", Template.MAXIMUM, 2, Scope.IDENTIFICATION)
assert.equal(pw, "Ig^JIcxD!*)TbefJBi6-")
})
})
describe("spectre", () => {
- const tests = readFileSync(join(__dirname, "spectre_tests.xml"), "utf8")
+ const tests = readFileSync(join(__dirname, "spectre_tests.xml"), "utf8")
const parser = new DOMParser()
const xml = parser.parseFromString(tests, "text/xml")
const cases = xml.getElementsByTagName("case")
@@ -29,7 +33,7 @@ const parseCase = (c: Element): Record<string, string> => {
const caseData: Record<string, string> = {}
const id = c.getAttribute("id")
caseData.id = id ?? ""
- Array.from(c.childNodes).forEach(child => {
+ Array.from(c.childNodes).forEach((child) => {
if (child.textContent === null) return
caseData[child.nodeName] = child.textContent.trim()
})
@@ -43,8 +47,16 @@ }
for (const c of Array.from(cases).slice(1)) {
const test = parseCase(c)
it(`should generate the password for ${test.id}`, () => {
- const s = new Spectre(getDef(test, "userName"), getDef(test, "userSecret"))
- const pw = s.site(getDef(test, "siteName"), getDef(test, "resultType") as Template, Number.parseInt(getDef(test, "keyCounter")), getDef(test, "keyPurpose") as Scope)
+ const s = new Spectre(
+ getDef(test, "userName"),
+ getDef(test, "userSecret"),
+ )
+ const pw = s.site(
+ getDef(test, "siteName"),
+ getDef(test, "resultType") as Template,
+ Number.parseInt(getDef(test, "keyCounter")),
+ getDef(test, "keyPurpose") as Scope,
+ )
assert.equal(pw, test.result)
})
}