first commit
This commit is contained in:
@@ -0,0 +1,16 @@
|
||||
import { connect } from "mongoose";
|
||||
|
||||
async function connectDB() {
|
||||
try {
|
||||
await connect(process.env.MONGO_URI!, { dbName: "test2" });
|
||||
console.log("MongoDB connected...");
|
||||
} catch (error) {
|
||||
if (error instanceof Error) {
|
||||
console.error(error.message);
|
||||
}
|
||||
|
||||
process.exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
export default connectDB;
|
||||
@@ -0,0 +1,32 @@
|
||||
import "dotenv/config";
|
||||
import express, { json } from "express";
|
||||
import connectDB from "./config/db.js";
|
||||
import cors from "cors";
|
||||
import loginRouter from "./routes/login.js";
|
||||
import registrationRouter from "./routes/registration.js";
|
||||
import authMiddleware from "./middlewares/auth.js";
|
||||
import appRouter from "./routes/app.js";
|
||||
import companiesRouter from "./routes/companies.js";
|
||||
import scheduledSessionsRouter from "./routes/scheduledSessions.js";
|
||||
import usersRouter from "./routes/users.js";
|
||||
import buildsRouter from "./routes/builds.js";
|
||||
|
||||
const app = express();
|
||||
const port = process.env.PORT || 3000;
|
||||
|
||||
connectDB();
|
||||
|
||||
app.use(json());
|
||||
app.use(cors());
|
||||
|
||||
app.use("/login", loginRouter);
|
||||
app.use("/registration", registrationRouter);
|
||||
app.use("/app", authMiddleware, appRouter);
|
||||
app.use("/companies", authMiddleware, companiesRouter);
|
||||
app.use("/users", authMiddleware, usersRouter);
|
||||
app.use("/builds", authMiddleware, buildsRouter);
|
||||
app.use("/scheduled_sessions", authMiddleware, scheduledSessionsRouter);
|
||||
|
||||
app.listen(port, () => {
|
||||
console.log(`Server listening on port ${port}`);
|
||||
});
|
||||
@@ -0,0 +1,32 @@
|
||||
import { Request, Response, NextFunction } from "express";
|
||||
import jwt, { Secret } from "jsonwebtoken";
|
||||
import Token from "../models/Token.js";
|
||||
import User from "../models/User.js";
|
||||
|
||||
async function authMiddleware(req: Request, res: Response, next: NextFunction) {
|
||||
if (!req.headers.authorization || !req.headers.authorization.split(" ")[1]) {
|
||||
return res.status(401).json({ error: 10 });
|
||||
}
|
||||
|
||||
const accessToken = req.headers.authorization.split(" ")[1];
|
||||
|
||||
try {
|
||||
jwt.verify(accessToken, process.env.JWT_SECRET as Secret);
|
||||
} catch (error) {
|
||||
return res.status(401).json({ erorr: 20 });
|
||||
}
|
||||
|
||||
const foundAccessToken = await Token.findOne({ accessToken });
|
||||
|
||||
if (!foundAccessToken) {
|
||||
return res.status(401).json({ error: 30 });
|
||||
}
|
||||
|
||||
const user = await User.findById(foundAccessToken.userId);
|
||||
|
||||
res.locals = { accessToken, user };
|
||||
|
||||
next();
|
||||
}
|
||||
|
||||
export default authMiddleware;
|
||||
@@ -0,0 +1,34 @@
|
||||
import { model, Schema } from "mongoose";
|
||||
|
||||
const buildSchema = new Schema(
|
||||
{
|
||||
companyId: {
|
||||
type: Schema.Types.ObjectId,
|
||||
required: true,
|
||||
},
|
||||
name: {
|
||||
type: String,
|
||||
required: true,
|
||||
unique: true,
|
||||
},
|
||||
sessionLimit: {
|
||||
type: Number,
|
||||
required: true,
|
||||
},
|
||||
},
|
||||
{
|
||||
timestamps: true,
|
||||
toJSON: { virtuals: true },
|
||||
toObject: { virtuals: true },
|
||||
}
|
||||
);
|
||||
|
||||
buildSchema.virtual("scheduledSessions", {
|
||||
ref: "Scheduled_Session",
|
||||
localField: "_id",
|
||||
foreignField: "buildId",
|
||||
});
|
||||
|
||||
const Build = model("Build", buildSchema);
|
||||
|
||||
export default Build;
|
||||
@@ -0,0 +1,38 @@
|
||||
import { model, Schema } from "mongoose";
|
||||
|
||||
const companySchema = new Schema(
|
||||
{
|
||||
name: {
|
||||
type: String,
|
||||
required: true,
|
||||
unique: true,
|
||||
},
|
||||
},
|
||||
{
|
||||
timestamps: true,
|
||||
toJSON: { virtuals: true },
|
||||
toObject: { virtuals: true },
|
||||
}
|
||||
);
|
||||
|
||||
companySchema.virtual("users", {
|
||||
ref: "User",
|
||||
localField: "_id",
|
||||
foreignField: "companyId",
|
||||
});
|
||||
|
||||
companySchema.virtual("scheduledSessions", {
|
||||
ref: "Scheduled_Session",
|
||||
localField: "_id",
|
||||
foreignField: "companyId",
|
||||
});
|
||||
|
||||
companySchema.virtual("builds", {
|
||||
ref: "Build",
|
||||
localField: "_id",
|
||||
foreignField: "companyId",
|
||||
});
|
||||
|
||||
const Company = model("Company", companySchema);
|
||||
|
||||
export default Company;
|
||||
@@ -0,0 +1,45 @@
|
||||
import { model, Schema } from "mongoose";
|
||||
|
||||
const scheduledSessionSchema = new Schema(
|
||||
{
|
||||
companyId: {
|
||||
type: Schema.Types.ObjectId,
|
||||
ref: "Company",
|
||||
required: true,
|
||||
},
|
||||
buildId: {
|
||||
type: Schema.Types.ObjectId,
|
||||
ref: "Build",
|
||||
required: true,
|
||||
},
|
||||
userId: {
|
||||
type: Schema.Types.ObjectId,
|
||||
ref: "User",
|
||||
},
|
||||
startAt: {
|
||||
type: Date,
|
||||
required: true,
|
||||
},
|
||||
clientName: {
|
||||
type: String,
|
||||
required: true,
|
||||
},
|
||||
clientPhone: {
|
||||
type: String,
|
||||
required: true,
|
||||
},
|
||||
clientEmail: {
|
||||
type: String,
|
||||
required: true,
|
||||
},
|
||||
},
|
||||
{
|
||||
timestamps: true,
|
||||
toJSON: { virtuals: true },
|
||||
toObject: { virtuals: true },
|
||||
}
|
||||
);
|
||||
|
||||
const ScheduledSession = model("Scheduled_Session", scheduledSessionSchema);
|
||||
|
||||
export default ScheduledSession;
|
||||
@@ -0,0 +1,24 @@
|
||||
import { model, Schema } from "mongoose";
|
||||
|
||||
const tokenSchema = new Schema(
|
||||
{
|
||||
userId: {
|
||||
type: Schema.Types.ObjectId,
|
||||
ref: "User",
|
||||
required: true,
|
||||
},
|
||||
accessToken: {
|
||||
type: String,
|
||||
required: true,
|
||||
},
|
||||
},
|
||||
{
|
||||
timestamps: true,
|
||||
toJSON: { virtuals: true },
|
||||
toObject: { virtuals: true },
|
||||
}
|
||||
);
|
||||
|
||||
const Token = model("Token", tokenSchema);
|
||||
|
||||
export default Token;
|
||||
@@ -0,0 +1,40 @@
|
||||
import { model, Schema } from "mongoose";
|
||||
|
||||
const userSchema = new Schema(
|
||||
{
|
||||
username: {
|
||||
type: String,
|
||||
required: true,
|
||||
unique: true,
|
||||
},
|
||||
password: {
|
||||
type: String,
|
||||
required: true,
|
||||
},
|
||||
companyId: {
|
||||
type: Schema.Types.ObjectId,
|
||||
ref: "Company",
|
||||
required: true,
|
||||
},
|
||||
name: {
|
||||
type: String,
|
||||
required: true,
|
||||
},
|
||||
role: {
|
||||
type: String,
|
||||
required: true,
|
||||
},
|
||||
avatar: {
|
||||
type: String,
|
||||
},
|
||||
},
|
||||
{
|
||||
timestamps: true,
|
||||
toJSON: { virtuals: true },
|
||||
toObject: { virtuals: true },
|
||||
}
|
||||
);
|
||||
|
||||
const User = model("User", userSchema);
|
||||
|
||||
export default User;
|
||||
@@ -0,0 +1,13 @@
|
||||
import { Router } from "express";
|
||||
import Company from "../models/Company.js";
|
||||
|
||||
const appRouter = Router();
|
||||
|
||||
appRouter.post("/", async (_req, res) => {
|
||||
const companies = await Company.find();
|
||||
console.log(companies);
|
||||
|
||||
res.json({ route: "app" });
|
||||
});
|
||||
|
||||
export default appRouter;
|
||||
@@ -0,0 +1,18 @@
|
||||
import { Router } from "express";
|
||||
import Build from "../models/Build.js";
|
||||
|
||||
const buildsRouter = Router();
|
||||
|
||||
buildsRouter.get("/", async (_req, res) => {
|
||||
await Build.find();
|
||||
|
||||
res.json({ ok: 1 });
|
||||
});
|
||||
|
||||
buildsRouter.post("/", async (req, res) => {
|
||||
const build = await Build.create(req.body);
|
||||
|
||||
res.json(build);
|
||||
});
|
||||
|
||||
export default buildsRouter;
|
||||
@@ -0,0 +1,85 @@
|
||||
import { Router } from "express";
|
||||
import Company from "../models/Company.js";
|
||||
import { parseISO, startOfDay, endOfDay } from "date-fns";
|
||||
|
||||
const companiesRouter = Router();
|
||||
|
||||
// companiesRouter.get("/", async (_req, res) => {
|
||||
// const companies = await Company.find();
|
||||
|
||||
// res.json(companies);
|
||||
// });
|
||||
|
||||
companiesRouter.get("/:id", async (req, res) => {
|
||||
if (req.params.id != res.locals.user.companyId) {
|
||||
res.json({ error: "Access denied!" });
|
||||
return;
|
||||
}
|
||||
|
||||
const company = await Company.findById(req.params.id);
|
||||
|
||||
res.json(company);
|
||||
});
|
||||
|
||||
companiesRouter.get("/:id/builds", async (req, res) => {
|
||||
if (req.params.id != res.locals.user.companyId) {
|
||||
res.json({ error: "Access denied!" });
|
||||
return;
|
||||
}
|
||||
|
||||
const company: any = await Company.findById(req.params.id).populate("builds");
|
||||
const { builds } = company;
|
||||
|
||||
res.json(builds);
|
||||
});
|
||||
|
||||
companiesRouter.get("/:id/users", async (req, res) => {
|
||||
if (req.params.id != res.locals.user.companyId) {
|
||||
res.json({ error: "Access denied!" });
|
||||
return;
|
||||
}
|
||||
|
||||
const company: any = await Company.findById(req.params.id).populate("users");
|
||||
const { users } = company;
|
||||
|
||||
res.json(users);
|
||||
});
|
||||
|
||||
companiesRouter.get(
|
||||
"/:id/builds/:buildId/scheduled_sessions",
|
||||
async (req, res) => {
|
||||
if (req.params.id != res.locals.user.companyId) {
|
||||
res.json({ error: "Access denied!" });
|
||||
return;
|
||||
}
|
||||
|
||||
if (!req.query.date) {
|
||||
res.json({ error: "req.query.date" });
|
||||
return;
|
||||
}
|
||||
|
||||
const date = parseISO(req.query.date as string);
|
||||
|
||||
const company: any = await Company.findById(req.params.id).populate({
|
||||
path: "builds",
|
||||
match: {
|
||||
_id: req.params.buildId,
|
||||
},
|
||||
populate: {
|
||||
path: "scheduledSessions",
|
||||
match: {
|
||||
startAt: {
|
||||
$gte: startOfDay(date),
|
||||
$lte: endOfDay(date),
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const { scheduledSessions } = company.builds[0];
|
||||
|
||||
res.json(scheduledSessions);
|
||||
}
|
||||
);
|
||||
|
||||
export default companiesRouter;
|
||||
@@ -0,0 +1,39 @@
|
||||
import { Router } from "express";
|
||||
import jwt, { Secret } from "jsonwebtoken";
|
||||
import bcrypt from "bcrypt";
|
||||
import User from "../models/User.js";
|
||||
import Token from "../models/Token.js";
|
||||
|
||||
const router = Router();
|
||||
|
||||
router.post("/", async (req, res) => {
|
||||
const { username, password } = req.body;
|
||||
|
||||
if (!username || !password) {
|
||||
return res.json({
|
||||
error: "You must pass the 'username' and 'password' parameters",
|
||||
});
|
||||
}
|
||||
|
||||
const user = await User.findOne({ username });
|
||||
|
||||
if (!user) {
|
||||
return res.json({ error: "A user with this name was not found" });
|
||||
}
|
||||
|
||||
if (!bcrypt.compareSync(password, user.password)) {
|
||||
return res.json({ error: "Invalid username or password" });
|
||||
}
|
||||
|
||||
const accessToken = jwt.sign({ username }, process.env.JWT_SECRET as Secret, {
|
||||
expiresIn: "365d",
|
||||
});
|
||||
|
||||
await Token.create({ userId: user.id, accessToken });
|
||||
|
||||
res.json({ accessToken, user });
|
||||
});
|
||||
|
||||
const loginRouter = router;
|
||||
|
||||
export default loginRouter;
|
||||
@@ -0,0 +1,40 @@
|
||||
import { Router } from "express";
|
||||
import jwt, { Secret } from "jsonwebtoken";
|
||||
import bcrypt from "bcrypt";
|
||||
import User from "../models/User.js";
|
||||
import Token from "../models/Token.js";
|
||||
|
||||
const router = Router();
|
||||
|
||||
router.post("/", async (req, res) => {
|
||||
const { username, password } = req.body;
|
||||
|
||||
if (!username || !password) {
|
||||
return res.json({ error: 1 });
|
||||
}
|
||||
|
||||
if (await User.exists({ username })) {
|
||||
return res.json({ error: 2 });
|
||||
}
|
||||
|
||||
const passwordHash = bcrypt.hashSync(password, 12);
|
||||
|
||||
if (!passwordHash) {
|
||||
return res.json({ error: 3 });
|
||||
}
|
||||
|
||||
const accessToken = jwt.sign({ username }, process.env.JWT_SECRET as Secret, {
|
||||
expiresIn: "365d",
|
||||
});
|
||||
|
||||
const user = await User.create({ username, password: passwordHash });
|
||||
const userId = user.id;
|
||||
|
||||
await Token.create({ userId, accessToken });
|
||||
|
||||
res.json({ accessToken });
|
||||
});
|
||||
|
||||
const registerRouter = router;
|
||||
|
||||
export default registerRouter;
|
||||
@@ -0,0 +1,12 @@
|
||||
import { Router } from "express";
|
||||
import ScheduledSession from "../models/ScheduledSession.js";
|
||||
|
||||
const scheduledSessionsRouter = Router();
|
||||
|
||||
scheduledSessionsRouter.get("/", async (_req, res) => {
|
||||
await ScheduledSession.find();
|
||||
|
||||
res.json({ ok: 1 });
|
||||
});
|
||||
|
||||
export default scheduledSessionsRouter;
|
||||
@@ -0,0 +1,20 @@
|
||||
import { Router } from "express";
|
||||
// import User from "../models/User.js";
|
||||
|
||||
const usersRouter = Router();
|
||||
|
||||
// usersRouter.get("/", async (_req, res) => {
|
||||
// const users = await User.find();
|
||||
// console.log(users);
|
||||
|
||||
// res.json(users);
|
||||
// });
|
||||
|
||||
// usersRouter.get("/:id", async (req, res) => {
|
||||
// const user = await User.findById(req.params.id);
|
||||
// console.log(user);
|
||||
|
||||
// res.json(user);
|
||||
// });
|
||||
|
||||
export default usersRouter;
|
||||
Vendored
+12
@@ -0,0 +1,12 @@
|
||||
declare global {
|
||||
namespace NodeJS {
|
||||
interface ProcessEnv {
|
||||
PORT?: string;
|
||||
NODE_ENV: "development" | "production";
|
||||
MONGO_URI: string;
|
||||
JWT_SECRET: string;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export {};
|
||||
Reference in New Issue
Block a user