Home

js-spectre @main - refs - log -
-
https://git.jolheiser.com/js-spectre.git
JS implementation for spectre/masterpassword
tree log patch
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-----
jolheiser <git@jolheiser.com>
1 month ago
7 changed files, 67 additions(+), 35 deletions(-)
IREADME.md
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
M eslint.config.jseslint.config.js
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,
M example/index.htmlexample/index.html
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>
M example/sakura.cssexample/sakura.css
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
+}
M lib/spectre.tslib/spectre.ts
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
M package.jsonpackage.json
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
+}
M tests/index.test.tstests/index.test.ts
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)
         })
     }