Signup working with new db
This commit is contained in:
6
main.js
6
main.js
@@ -2,12 +2,6 @@
|
|||||||
const { app, BrowserWindow, nativeImage } = require('electron');
|
const { app, BrowserWindow, nativeImage } = require('electron');
|
||||||
const path = require('path');
|
const path = require('path');
|
||||||
|
|
||||||
if (process.platform === 'darwin') {
|
|
||||||
setTimeout(() => {
|
|
||||||
app.dock.bounce('informational'); // or 'critical'
|
|
||||||
}, 1000);
|
|
||||||
}
|
|
||||||
|
|
||||||
function createWindow() {
|
function createWindow() {
|
||||||
const win = new BrowserWindow({
|
const win = new BrowserWindow({
|
||||||
show: false, // window is hidden
|
show: false, // window is hidden
|
||||||
|
|||||||
@@ -14,7 +14,8 @@
|
|||||||
"express": "^4.18.2",
|
"express": "^4.18.2",
|
||||||
"jsonwebtoken": "^9.0.2",
|
"jsonwebtoken": "^9.0.2",
|
||||||
"moment": "^2.30.1",
|
"moment": "^2.30.1",
|
||||||
"ws": "^8.18.3"
|
"ws": "^8.18.3",
|
||||||
|
"zod": "^4.1.12"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"electron": "^25.0.0"
|
"electron": "^25.0.0"
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ export default class AuthHandler {
|
|||||||
|
|
||||||
async login(req, res) {
|
async login(req, res) {
|
||||||
const { email, password } = req.body;
|
const { email, password } = req.body;
|
||||||
let foundUser = global.db.get.userByEmail(email)
|
let foundUser = global.db.members.getByEmail(email)
|
||||||
if(!foundUser) {
|
if(!foundUser) {
|
||||||
res.status(400).json({ error: 'Incorrect email.' });
|
res.status(400).json({ error: 'Incorrect email.' });
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -49,11 +49,7 @@ export default class Database {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
generateUserID() {
|
saveData() {
|
||||||
let id = this.labels["User"].length + 1;
|
|
||||||
while (this.get.user(`user-${id}`)) {
|
|
||||||
id++;
|
|
||||||
}
|
|
||||||
return `user-${id}`; // O(1) most of the time
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,11 +1,33 @@
|
|||||||
import OrderedObject from "./OrderedObject.js"
|
import OrderedObject from "./OrderedObject.js"
|
||||||
|
const argon2 = require("argon2")
|
||||||
|
const { z } = require("zod")
|
||||||
|
|
||||||
export default class Members extends OrderedObject {
|
export default class Members extends OrderedObject {
|
||||||
|
addressSchema = z.object({
|
||||||
|
address1: z.string(),
|
||||||
|
address2: z.string().optional(),
|
||||||
|
zip: z.string().regex(/^\d{5}(-\d{4})?$/),
|
||||||
|
state: z.string(),
|
||||||
|
city: z.string()
|
||||||
|
})
|
||||||
|
schema = z.object({
|
||||||
|
email: z.string().email(),
|
||||||
|
firstName: z.string(),
|
||||||
|
lastName: z.number().int().min(0),
|
||||||
|
password: z.string(),
|
||||||
|
address: this.addressSchema
|
||||||
|
})
|
||||||
|
|
||||||
add(newMember) {
|
isHashed = (s) => {return s.startsWith("$argon2")}
|
||||||
|
|
||||||
|
async add(newMember) {
|
||||||
console.log("adding ", newMember)
|
console.log("adding ", newMember)
|
||||||
let id = `MEMBER-${newMember.email}`
|
let id = `MEMBER-${newMember.email}`
|
||||||
if(this.validate(id, newMember)) {
|
if(this.schema.safeParse(newMember)) {
|
||||||
|
if(!this.isHashed(newMember.password)) {
|
||||||
|
const hash = await argon2.hash(newMember.password);
|
||||||
|
newMember.password = hash
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
super.add(id, newMember)
|
super.add(id, newMember)
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
@@ -17,48 +39,7 @@ export default class Members extends OrderedObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
validate(id, node) {
|
getByEmail(email) {
|
||||||
let idTraits = {
|
return super.get(`MEMBER-${email}`)
|
||||||
firstWord: "MEMBER"
|
|
||||||
}
|
|
||||||
|
|
||||||
let fields = [
|
|
||||||
"firstName",
|
|
||||||
"lastName",
|
|
||||||
"email",
|
|
||||||
"password"
|
|
||||||
]
|
|
||||||
|
|
||||||
let checkID = () => {
|
|
||||||
let split = id.split("-")
|
|
||||||
return (
|
|
||||||
split[0] === idTraits.firstWord
|
|
||||||
&& split[1].includes("@")
|
|
||||||
&& split[1].includes(".")
|
|
||||||
)
|
|
||||||
}
|
|
||||||
let idres = checkID()
|
|
||||||
if(!idres) {
|
|
||||||
console.log("id failed: ", id)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
let checkFields = () => {
|
|
||||||
for(let i = 0; i < fields.length; i++) {
|
|
||||||
if(!node[fields[i]]) {
|
|
||||||
throw new Error(`Member ${node.email} is missing trait ${fields[i]}`)
|
|
||||||
return false
|
|
||||||
} else {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
let fieldres = checkFields()
|
|
||||||
if(!fieldres) {
|
|
||||||
console.log("fields failed")
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
return true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -43,11 +43,11 @@ class Server {
|
|||||||
next()
|
next()
|
||||||
}
|
}
|
||||||
|
|
||||||
newUserSubmission = (req, res) => {
|
newUserSubmission = async (req, res) => {
|
||||||
const { token } = req.query;
|
const { token } = req.query;
|
||||||
try {
|
try {
|
||||||
db.members.add(req.body)
|
await db.members.add(req.body)
|
||||||
return res.redirect(`/signin/?new=true`);
|
return res.status(200).json({});
|
||||||
} catch(e) {
|
} catch(e) {
|
||||||
return res.status(e.status).json({ error: 'Error adding new member' });
|
return res.status(e.status).json({ error: 'Error adding new member' });
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
Sam Russell
|
Sam Russell
|
||||||
Captured Sun
|
Captured Sun
|
||||||
|
11.23.25 - Added onSubmit() event for form submission
|
||||||
11.20.25 - Added "pct" style unit, added alignVertical and alignHorizontal for flex boxes
|
11.20.25 - Added "pct" style unit, added alignVertical and alignHorizontal for flex boxes
|
||||||
11.19.25 - Allowing for "auto" values in otherwise numeric styles, adding vmin and vmax units
|
11.19.25 - Allowing for "auto" values in otherwise numeric styles, adding vmin and vmax units
|
||||||
11.17.25.3 - Adding styles() and fixing dynamic function from earlier
|
11.17.25.3 - Adding styles() and fixing dynamic function from earlier
|
||||||
@@ -907,6 +908,13 @@ HTMLElement.prototype.onInput = function(cb) {
|
|||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
HTMLElement.prototype.onSubmit = function(cb) {
|
||||||
|
if(!this.matches('form'))
|
||||||
|
throw new Error("Can't put form event on non-form element!")
|
||||||
|
this._storeListener("submit", cb);
|
||||||
|
return this;
|
||||||
|
};
|
||||||
|
|
||||||
HTMLElement.prototype.onTouch = function(cb) {
|
HTMLElement.prototype.onTouch = function(cb) {
|
||||||
const onStart = () => cb.call(this, true);
|
const onStart = () => cb.call(this, true);
|
||||||
const onEnd = () => cb.call(this, false);
|
const onEnd = () => cb.call(this, false);
|
||||||
|
|||||||
@@ -1,5 +1,8 @@
|
|||||||
class SignupForm extends Shadow {
|
class SignupForm extends Shadow {
|
||||||
|
|
||||||
|
errorMessage = "Error signing up. Please try again later or email info@hyperia.so if the problem persists."
|
||||||
|
successMessage = "Success! You may now log in."
|
||||||
|
|
||||||
inputStyles(el) {
|
inputStyles(el) {
|
||||||
return el
|
return el
|
||||||
.border("1px solid var(--accent)")
|
.border("1px solid var(--accent)")
|
||||||
@@ -11,6 +14,13 @@ class SignupForm extends Shadow {
|
|||||||
|
|
||||||
VStack(() => {
|
VStack(() => {
|
||||||
|
|
||||||
|
p()
|
||||||
|
.attr({id: "signupMessage"})
|
||||||
|
.display("none")
|
||||||
|
.padding(1, em)
|
||||||
|
.color("var(--main)")
|
||||||
|
.background("var(--accent)")
|
||||||
|
|
||||||
HStack(() => {
|
HStack(() => {
|
||||||
|
|
||||||
VStack(() => {
|
VStack(() => {
|
||||||
@@ -71,10 +81,51 @@ class SignupForm extends Shadow {
|
|||||||
button("Submit")
|
button("Submit")
|
||||||
})
|
})
|
||||||
.gap(2, em)
|
.gap(2, em)
|
||||||
|
|
||||||
console.log(window.location.pathname)
|
|
||||||
})
|
})
|
||||||
.attr({action: window.location.pathname + window.location.search, method: "POST"})
|
.onSubmit(async (e) => {
|
||||||
|
e.preventDefault()
|
||||||
|
console.log("submitting")
|
||||||
|
$("#signupMessage").style.display = "none"
|
||||||
|
|
||||||
|
const formData = new FormData(this.$("form"));
|
||||||
|
const data = Object.fromEntries(formData.entries());
|
||||||
|
let newMember = {
|
||||||
|
"email": data.email,
|
||||||
|
"firstName": data.firstName,
|
||||||
|
"lastName": data.lastName,
|
||||||
|
"password": data.password
|
||||||
|
}
|
||||||
|
let address = {
|
||||||
|
"address1": data.address1,
|
||||||
|
"address2": data.address2,
|
||||||
|
"zip": data.zip,
|
||||||
|
"state": data.state,
|
||||||
|
"city": data.city
|
||||||
|
}
|
||||||
|
newMember.address = address
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(window.location.pathname + window.location.search, {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json"
|
||||||
|
},
|
||||||
|
body: JSON.stringify(newMember)
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
$("#signupMessage").style.display = "block"
|
||||||
|
$("#signupMessage").innerText = this.errorMessage
|
||||||
|
throw new Error(`HTTP error! status: ${response.status}`);
|
||||||
|
} else {
|
||||||
|
$("#signupMessage").style.display = "block"
|
||||||
|
$("#signupMessage").innerText = this.successMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (err) {
|
||||||
|
console.error("Fetch error:", err);
|
||||||
|
}
|
||||||
|
})
|
||||||
.x(50, vw).y(53, vh)
|
.x(50, vw).y(53, vh)
|
||||||
.width(60, vw)
|
.width(60, vw)
|
||||||
.center()
|
.center()
|
||||||
|
|||||||
Reference in New Issue
Block a user