├── .gitignore
├── README.md
├── backend
├── .gitignore
├── README.md
├── index.js
├── middleware
│ └── authentication.js
├── models
│ ├── Blog.js
│ └── User.js
├── package-lock.json
├── package.json
├── routes
│ ├── authRoute.js
│ ├── blogRoute.js
│ └── homeRoute.js
└── yarn.lock
└── frontend
├── .gitignore
├── README.md
├── debug.log
├── package - Copy.json
├── package.json
├── public
├── _redirects
├── android-chrome-192x192.png
├── android-chrome-512x512.png
├── apple-touch-icon.png
├── favicon-16x16.png
├── favicon-32x32.png
├── favicon.ico
├── index.html
├── manifest.json
├── robots.txt
└── site.webmanifest
├── src
├── App.js
├── App.test.js
├── assets
│ ├── Icons
│ │ └── Icons.jsx
│ └── Images
│ │ ├── bg.jpg
│ │ ├── bgp3.png
│ │ ├── home.svg
│ │ └── meta.jpg
├── components
│ ├── common
│ │ ├── AuthorMenu
│ │ │ ├── AuthorMenu.component.jsx
│ │ │ ├── DeleteButton.component.jsx
│ │ │ └── Edit.component.jsx
│ │ ├── Footer
│ │ │ ├── Footer.component.jsx
│ │ │ └── Footer.styles.scss
│ │ ├── Header
│ │ │ ├── Header.component.jsx
│ │ │ ├── Header.styles.css
│ │ │ ├── Header.styles.css.map
│ │ │ └── Header.styles.scss
│ │ ├── SEO
│ │ │ └── SEO.component.jsx
│ │ └── SocialMedia
│ │ │ └── SocialMedia.component.jsx
│ └── pages
│ │ ├── 404
│ │ ├── NotFound.component.jsx
│ │ └── NotFound.styles.scss
│ │ ├── AdminPage
│ │ ├── AdminPage.component.jsx
│ │ └── AdminPage.styles.scss
│ │ ├── Auth
│ │ ├── Login.component.jsx
│ │ ├── Login.styles.scss
│ │ ├── Register.component.jsx
│ │ └── Register.styles.scss
│ │ ├── Blog
│ │ ├── Blog.component.jsx
│ │ ├── Blog.styles.scss
│ │ └── BlogRender.jsx
│ │ ├── BlogCard
│ │ ├── BlogCard.component.jsx
│ │ └── BlogCard.styles.scss
│ │ ├── Blogs
│ │ ├── Blogs.component.jsx
│ │ └── Blogs.styles.scss
│ │ ├── EditPage
│ │ └── EditPage.component.jsx
│ │ ├── Home
│ │ ├── Home.component.jsx
│ │ └── Home.styles.scss
│ │ ├── NewBlog
│ │ ├── NewBlog.component.jsx
│ │ └── NewBlog.styles.scss
│ │ └── User
│ │ ├── User.component.jsx
│ │ └── User.styles.scss
├── context
│ └── userContext.js
├── index.css
├── index.js
├── serviceWorker.js
├── services
│ └── authService.js
├── setupTests.js
├── themes
│ └── theme.js
└── yarn.lock
└── yarn.lock
/.gitignore:
--------------------------------------------------------------------------------
1 | frontend/node_modules/
2 | backend/node_modules/
3 | backend/.env
4 | frontend/.env
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 | Semicolon
7 |
8 |
9 |
10 | A markdown based blogging platform built using React and ExpressJS.
11 |
12 |
13 | Key Features •
14 | How To Use •
15 | Visit •
16 | Credits •
17 | License
18 |
19 |
20 | 
21 | 
22 |
23 | ## Key Features
24 |
25 | - Markdown Editor - Make changes, See changes
26 | - Instantly see what your Markdown documents look like in HTML as you create them,
27 | - Sync Scrolling
28 | - While you type, LivePreview will automatically scroll to the current location you're editing.
29 | - GitHub Flavored Markdown.
30 | - Syntax highlighting and code support
31 | - Dark/Light mode (Under construction)
32 | - Toolbar for basic Markdown formatting.
33 | - Save your blog as a draft (Under construction)
34 | - Emoji support in preview :tada:
35 | - Share your blog on Twitter, WhatsApp and LinkedIn
36 | - Full screen mode
37 | - Write distraction free.
38 |
39 | ## Visit
40 |
41 | You can visit [here](http://semicolon-blog.netlify.app)
42 |
43 | ## How To Use
44 |
45 | To clone and run this application, you'll need [Git](https://git-scm.com) and [Node.js](https://nodejs.org/en/download/) (which comes with [npm](http://npmjs.com)) installed on your computer. From your command line:
46 |
47 | ```bash
48 | # Clone this repository
49 | $ git clone https://github.com/OmkarK45/blog-app
50 |
51 | # Go into the repository
52 | $ cd backend
53 |
54 | # Go into the repository
55 | $ cd frontend
56 |
57 | # Install dependencies in backend by using
58 | $ npm install (wait 5mins)
59 |
60 | # Install the frontend dependencies by using
61 | $ yarn install (wait 5mins)
62 |
63 | # Setup environment variables
64 | $ In backend folder, create a file called '.env'
65 | $ Declare the following variables as env vars : ACCESS_TOKEN, REFRESH_TOKEN_SECRET, JWT_SECRET='some secure string'(required), DB_URI_LOCAL=mongodb://localhost/blogdb (if db doesn't exist, it'll create one.), DB_URI_ATLAS='the url from mongo cloud atlas' (optional)
66 |
67 | # Run the backend
68 | $cd backend
69 | $ npm start
70 |
71 | # Run the frontend by setting proxy to frontend in package.json
72 | $ add this line at line 2 in frontend/package.json >> "proxy" : "https://localhost:3001"
73 |
74 | # Run the frontend
75 | $cd frontend
76 | $ yarn start
77 | ```
78 |
79 | Note: If you're using Linux Bash for Windows, [see this guide](https://www.howtogeek.com/261575/how-to-run-graphical-linux-desktop-applications-from-windows-10s-bash-shell/) or use `node` from the command prompt.
80 |
81 | ## Credits
82 |
83 | This software uses the following open source packages:
84 |
85 | - [NodeJS](http://nodejs.org)
86 | - [ReactJS](http://reactjs.org)
87 | - [ExpressJS](http://expressjs.com)
88 | - [MongoDB](http://mongodb.com)
89 |
90 | ## License
91 |
92 | MIT
93 |
94 | > [omkarkulkarni.netlify.app](https://www.omkarkulkarni.netlify.app) ·
95 | > GitHub [@omkark45](https://github.com/omkark45) ·
96 | > LinkedIn [@omkark45](https://www.linkedin.com/in/omkark45/)
97 |
--------------------------------------------------------------------------------
/backend/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules/
2 | .env
--------------------------------------------------------------------------------
/backend/README.md:
--------------------------------------------------------------------------------
1 | # jwt-auth-boilerplate
2 | Simple boilerplate for backend projects in NodeJS and Express.. (Uses JWT or jsonwebtoken)
3 |
4 |
5 | # How to use
6 | - For projects which use react =>
7 | 1. Create a form in your application which puts POST (using axios etc) request to '/login'
8 | 2. Submit username in request body
9 | 3. Set auth token in frontend
10 |
11 | - Vanilla JS => use axios POST request to '/login'
12 |
--------------------------------------------------------------------------------
/backend/index.js:
--------------------------------------------------------------------------------
1 | require("dotenv").config();
2 | const morgan = require("morgan");
3 | const express = require("express");
4 | const mongoose = require("mongoose");
5 | const bodyParser = require("body-parser");
6 | const cors = require("cors");
7 | const methodOverride = require("method-override");
8 | const homeRoute = require("./routes/homeRoute");
9 | const blogRoute = require("./routes/blogRoute");
10 | const authRoute = require("./routes/authRoute");
11 | const helmet = require("helmet");
12 | const app = express();
13 |
14 | // Express setup
15 | app.use(helmet());
16 | app.use(helmet.xssFilter());
17 | app.disable("x-powered-by");
18 | app.use(
19 | bodyParser.urlencoded({ extended: false }, { useUnifiedTopology: true })
20 | );
21 | app.use(bodyParser.json());
22 | app.use(methodOverride("_method"));
23 |
24 | var corsOptions = {
25 | origin: "https://semicolon-blog.netlify.app",
26 | optionsSuccessStatus: 200, // For legacy browser support
27 | };
28 |
29 | app.use(cors(corsOptions));
30 | app.use(morgan("tiny"));
31 | // Routes
32 | app.use("/", homeRoute);
33 | app.use("/blogs", blogRoute);
34 | app.use("/user", authRoute);
35 |
36 | // DB Configuration
37 | mongoose.connect(
38 | process.env.DB_URI_ATLAS,
39 | { useNewUrlParser: true, useUnifiedTopology: true, useCreateIndex: true },
40 | () => {
41 | console.log("Connected to Cloud database.");
42 | }
43 | );
44 |
45 | // Express server Bootup
46 | const PORT = process.env.PORT || 3001;
47 | app.listen(PORT, () => {
48 | console.log("Server has started on", `https://localhost:${PORT}`);
49 | });
50 |
--------------------------------------------------------------------------------
/backend/middleware/authentication.js:
--------------------------------------------------------------------------------
1 | const jwt = require("jsonwebtoken");
2 |
3 | function authentication(req, res, next) {
4 | const token = req.header("x-auth-token");
5 | if (!token) return res.status(401).json({ msg: "Auth Token not found." });
6 | try {
7 | const decodedToken = jwt.verify(token, process.env.JWT_SECRET);
8 | req.user = decodedToken;
9 | next();
10 | } catch (error) {
11 | res.json({ msg: "Token is invalid." });
12 | }
13 | }
14 | module.exports = authentication;
15 |
--------------------------------------------------------------------------------
/backend/models/Blog.js:
--------------------------------------------------------------------------------
1 | const mongoose = require("mongoose");
2 | const User = require("../models/User").schema;
3 | const slugify = require("slugify");
4 |
5 | const blogSchema = new mongoose.Schema({
6 | title: {
7 | type: String,
8 | required: true,
9 | max: 256,
10 | min: 5,
11 | },
12 | bannerURL: {
13 | type: String,
14 | required: false,
15 | },
16 | subtitle: {
17 | type: String,
18 | min: 3,
19 | max: 1024,
20 | },
21 | author: {
22 | type: String,
23 | },
24 | content: {
25 | type: String,
26 | min: 10,
27 | max: 3048,
28 | },
29 | date: {
30 | type: String,
31 | },
32 | avatar: {
33 | type: String,
34 | },
35 | authorID: {
36 | type: String,
37 | },
38 | isApproved: {
39 | type: Boolean,
40 | default: false,
41 | },
42 | slug: {
43 | type: String,
44 | unique: true,
45 | required: true,
46 | },
47 | });
48 | blogSchema.pre("validate", function (next) {
49 | if (this.title) {
50 | this.slug = slugify(this.title, { lower: true, strict: true });
51 | }
52 | next()
53 | });
54 | module.exports = mongoose.model("Blog", blogSchema);
55 |
--------------------------------------------------------------------------------
/backend/models/User.js:
--------------------------------------------------------------------------------
1 | const mongoose = require("mongoose");
2 | const Blog = require("../models/Blog").schema;
3 | const Joi = require("@hapi/joi");
4 |
5 | const userSchema = new mongoose.Schema({
6 | username: {
7 | type: String,
8 | required: true,
9 | min: 3,
10 | max: 256,
11 | },
12 | name: {
13 | type: String,
14 | require: true,
15 | min: 3,
16 | max: 256,
17 | },
18 | password: {
19 | type: String,
20 | required: true,
21 | min: 6,
22 | max: 1024,
23 | },
24 | email: {
25 | type: String,
26 | required: true,
27 | min: 5,
28 | max: 256,
29 | },
30 | avatar: {
31 | type: String,
32 | required: false,
33 | },
34 | socialMedia: [],
35 | blogs: [
36 | {
37 | type: mongoose.Schema.Types.ObjectId,
38 | ref: "Blog",
39 | },
40 | ],
41 | isAdmin: {
42 | type: Boolean,
43 | default: false,
44 | },
45 | });
46 |
47 | function validateUser(user) {
48 | const schema = {
49 | email: Joi.string().min(3).max(255).required().email(),
50 | };
51 | return schema.validate(user);
52 |
53 | }
54 |
55 | exports.validate = validateUser;
56 | module.exports = mongoose.model("User", userSchema);
57 |
--------------------------------------------------------------------------------
/backend/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "blog-backend",
3 | "version": "1.0.0",
4 | "lockfileVersion": 1,
5 | "requires": true,
6 | "dependencies": {
7 | "accepts": {
8 | "version": "1.3.7",
9 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
10 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
11 | "requires": {
12 | "mime-types": "~2.1.24",
13 | "negotiator": "0.6.2"
14 | }
15 | },
16 | "array-flatten": {
17 | "version": "1.1.1",
18 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
19 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
20 | },
21 | "basic-auth": {
22 | "version": "2.0.1",
23 | "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz",
24 | "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==",
25 | "requires": {
26 | "safe-buffer": "5.1.2"
27 | }
28 | },
29 | "bcryptjs": {
30 | "version": "2.4.3",
31 | "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz",
32 | "integrity": "sha1-mrVie5PmBiH/fNrF2pczAn3x0Ms="
33 | },
34 | "bl": {
35 | "version": "2.2.1",
36 | "resolved": "https://registry.npmjs.org/bl/-/bl-2.2.1.tgz",
37 | "integrity": "sha512-6Pesp1w0DEX1N550i/uGV/TqucVL4AM/pgThFSN/Qq9si1/DF9aIHs1BxD8V/QU0HoeHO6cQRTAuYnLPKq1e4g==",
38 | "requires": {
39 | "readable-stream": "^2.3.5",
40 | "safe-buffer": "^5.1.1"
41 | }
42 | },
43 | "bluebird": {
44 | "version": "3.5.1",
45 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz",
46 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA=="
47 | },
48 | "body-parser": {
49 | "version": "1.19.0",
50 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
51 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
52 | "requires": {
53 | "bytes": "3.1.0",
54 | "content-type": "~1.0.4",
55 | "debug": "2.6.9",
56 | "depd": "~1.1.2",
57 | "http-errors": "1.7.2",
58 | "iconv-lite": "0.4.24",
59 | "on-finished": "~2.3.0",
60 | "qs": "6.7.0",
61 | "raw-body": "2.4.0",
62 | "type-is": "~1.6.17"
63 | }
64 | },
65 | "bson": {
66 | "version": "1.1.5",
67 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.5.tgz",
68 | "integrity": "sha512-kDuEzldR21lHciPQAIulLs1LZlCXdLziXI6Mb/TDkwXhb//UORJNPXgcRs2CuO4H0DcMkpfT3/ySsP3unoZjBg=="
69 | },
70 | "buffer-equal-constant-time": {
71 | "version": "1.0.1",
72 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz",
73 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk="
74 | },
75 | "bytes": {
76 | "version": "3.1.0",
77 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
78 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg=="
79 | },
80 | "content-disposition": {
81 | "version": "0.5.3",
82 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
83 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
84 | "requires": {
85 | "safe-buffer": "5.1.2"
86 | }
87 | },
88 | "content-type": {
89 | "version": "1.0.4",
90 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
91 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
92 | },
93 | "cookie": {
94 | "version": "0.4.0",
95 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
96 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg=="
97 | },
98 | "cookie-signature": {
99 | "version": "1.0.6",
100 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
101 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
102 | },
103 | "core-util-is": {
104 | "version": "1.0.2",
105 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
106 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac="
107 | },
108 | "cors": {
109 | "version": "2.8.5",
110 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
111 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
112 | "requires": {
113 | "object-assign": "^4",
114 | "vary": "^1"
115 | }
116 | },
117 | "debug": {
118 | "version": "2.6.9",
119 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
120 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
121 | "requires": {
122 | "ms": "2.0.0"
123 | }
124 | },
125 | "denque": {
126 | "version": "1.4.1",
127 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.4.1.tgz",
128 | "integrity": "sha512-OfzPuSZKGcgr96rf1oODnfjqBFmr1DVoc/TrItj3Ohe0Ah1C5WX5Baquw/9U9KovnQ88EqmJbD66rKYUQYN1tQ=="
129 | },
130 | "depd": {
131 | "version": "1.1.2",
132 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
133 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
134 | },
135 | "destroy": {
136 | "version": "1.0.4",
137 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
138 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
139 | },
140 | "dotenv": {
141 | "version": "8.2.0",
142 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz",
143 | "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw=="
144 | },
145 | "ecdsa-sig-formatter": {
146 | "version": "1.0.11",
147 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz",
148 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==",
149 | "requires": {
150 | "safe-buffer": "^5.0.1"
151 | }
152 | },
153 | "ee-first": {
154 | "version": "1.1.1",
155 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
156 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
157 | },
158 | "encodeurl": {
159 | "version": "1.0.2",
160 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
161 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
162 | },
163 | "escape-html": {
164 | "version": "1.0.3",
165 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
166 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
167 | },
168 | "etag": {
169 | "version": "1.8.1",
170 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
171 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
172 | },
173 | "express": {
174 | "version": "4.17.1",
175 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
176 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
177 | "requires": {
178 | "accepts": "~1.3.7",
179 | "array-flatten": "1.1.1",
180 | "body-parser": "1.19.0",
181 | "content-disposition": "0.5.3",
182 | "content-type": "~1.0.4",
183 | "cookie": "0.4.0",
184 | "cookie-signature": "1.0.6",
185 | "debug": "2.6.9",
186 | "depd": "~1.1.2",
187 | "encodeurl": "~1.0.2",
188 | "escape-html": "~1.0.3",
189 | "etag": "~1.8.1",
190 | "finalhandler": "~1.1.2",
191 | "fresh": "0.5.2",
192 | "merge-descriptors": "1.0.1",
193 | "methods": "~1.1.2",
194 | "on-finished": "~2.3.0",
195 | "parseurl": "~1.3.3",
196 | "path-to-regexp": "0.1.7",
197 | "proxy-addr": "~2.0.5",
198 | "qs": "6.7.0",
199 | "range-parser": "~1.2.1",
200 | "safe-buffer": "5.1.2",
201 | "send": "0.17.1",
202 | "serve-static": "1.14.1",
203 | "setprototypeof": "1.1.1",
204 | "statuses": "~1.5.0",
205 | "type-is": "~1.6.18",
206 | "utils-merge": "1.0.1",
207 | "vary": "~1.1.2"
208 | }
209 | },
210 | "finalhandler": {
211 | "version": "1.1.2",
212 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
213 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
214 | "requires": {
215 | "debug": "2.6.9",
216 | "encodeurl": "~1.0.2",
217 | "escape-html": "~1.0.3",
218 | "on-finished": "~2.3.0",
219 | "parseurl": "~1.3.3",
220 | "statuses": "~1.5.0",
221 | "unpipe": "~1.0.0"
222 | }
223 | },
224 | "forwarded": {
225 | "version": "0.1.2",
226 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz",
227 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ="
228 | },
229 | "fresh": {
230 | "version": "0.5.2",
231 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
232 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
233 | },
234 | "http-errors": {
235 | "version": "1.7.2",
236 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
237 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
238 | "requires": {
239 | "depd": "~1.1.2",
240 | "inherits": "2.0.3",
241 | "setprototypeof": "1.1.1",
242 | "statuses": ">= 1.5.0 < 2",
243 | "toidentifier": "1.0.0"
244 | }
245 | },
246 | "iconv-lite": {
247 | "version": "0.4.24",
248 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
249 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
250 | "requires": {
251 | "safer-buffer": ">= 2.1.2 < 3"
252 | }
253 | },
254 | "inherits": {
255 | "version": "2.0.3",
256 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
257 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
258 | },
259 | "ipaddr.js": {
260 | "version": "1.9.1",
261 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
262 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="
263 | },
264 | "isarray": {
265 | "version": "1.0.0",
266 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
267 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE="
268 | },
269 | "jsonwebtoken": {
270 | "version": "8.5.1",
271 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz",
272 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==",
273 | "requires": {
274 | "jws": "^3.2.2",
275 | "lodash.includes": "^4.3.0",
276 | "lodash.isboolean": "^3.0.3",
277 | "lodash.isinteger": "^4.0.4",
278 | "lodash.isnumber": "^3.0.3",
279 | "lodash.isplainobject": "^4.0.6",
280 | "lodash.isstring": "^4.0.1",
281 | "lodash.once": "^4.0.0",
282 | "ms": "^2.1.1",
283 | "semver": "^5.6.0"
284 | },
285 | "dependencies": {
286 | "ms": {
287 | "version": "2.1.2",
288 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
289 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
290 | }
291 | }
292 | },
293 | "jwa": {
294 | "version": "1.4.1",
295 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz",
296 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==",
297 | "requires": {
298 | "buffer-equal-constant-time": "1.0.1",
299 | "ecdsa-sig-formatter": "1.0.11",
300 | "safe-buffer": "^5.0.1"
301 | }
302 | },
303 | "jws": {
304 | "version": "3.2.2",
305 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz",
306 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==",
307 | "requires": {
308 | "jwa": "^1.4.1",
309 | "safe-buffer": "^5.0.1"
310 | }
311 | },
312 | "kareem": {
313 | "version": "2.3.1",
314 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.1.tgz",
315 | "integrity": "sha512-l3hLhffs9zqoDe8zjmb/mAN4B8VT3L56EUvKNqLFVs9YlFA+zx7ke1DO8STAdDyYNkeSo1nKmjuvQeI12So8Xw=="
316 | },
317 | "lodash.includes": {
318 | "version": "4.3.0",
319 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz",
320 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8="
321 | },
322 | "lodash.isboolean": {
323 | "version": "3.0.3",
324 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz",
325 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY="
326 | },
327 | "lodash.isinteger": {
328 | "version": "4.0.4",
329 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz",
330 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M="
331 | },
332 | "lodash.isnumber": {
333 | "version": "3.0.3",
334 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz",
335 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w="
336 | },
337 | "lodash.isplainobject": {
338 | "version": "4.0.6",
339 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz",
340 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs="
341 | },
342 | "lodash.isstring": {
343 | "version": "4.0.1",
344 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz",
345 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE="
346 | },
347 | "lodash.once": {
348 | "version": "4.1.1",
349 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz",
350 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w="
351 | },
352 | "media-typer": {
353 | "version": "0.3.0",
354 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
355 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
356 | },
357 | "memory-pager": {
358 | "version": "1.5.0",
359 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz",
360 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==",
361 | "optional": true
362 | },
363 | "merge-descriptors": {
364 | "version": "1.0.1",
365 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
366 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
367 | },
368 | "method-override": {
369 | "version": "3.0.0",
370 | "resolved": "https://registry.npmjs.org/method-override/-/method-override-3.0.0.tgz",
371 | "integrity": "sha512-IJ2NNN/mSl9w3kzWB92rcdHpz+HjkxhDJWNDBqSlas+zQdP8wBiJzITPg08M/k2uVvMow7Sk41atndNtt/PHSA==",
372 | "requires": {
373 | "debug": "3.1.0",
374 | "methods": "~1.1.2",
375 | "parseurl": "~1.3.2",
376 | "vary": "~1.1.2"
377 | },
378 | "dependencies": {
379 | "debug": {
380 | "version": "3.1.0",
381 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz",
382 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==",
383 | "requires": {
384 | "ms": "2.0.0"
385 | }
386 | }
387 | }
388 | },
389 | "methods": {
390 | "version": "1.1.2",
391 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
392 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
393 | },
394 | "mime": {
395 | "version": "1.6.0",
396 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
397 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
398 | },
399 | "mime-db": {
400 | "version": "1.44.0",
401 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz",
402 | "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg=="
403 | },
404 | "mime-types": {
405 | "version": "2.1.27",
406 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz",
407 | "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==",
408 | "requires": {
409 | "mime-db": "1.44.0"
410 | }
411 | },
412 | "mongodb": {
413 | "version": "3.6.2",
414 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.6.2.tgz",
415 | "integrity": "sha512-sSZOb04w3HcnrrXC82NEh/YGCmBuRgR+C1hZgmmv4L6dBz4BkRse6Y8/q/neXer9i95fKUBbFi4KgeceXmbsOA==",
416 | "requires": {
417 | "bl": "^2.2.1",
418 | "bson": "^1.1.4",
419 | "denque": "^1.4.1",
420 | "require_optional": "^1.0.1",
421 | "safe-buffer": "^5.1.2",
422 | "saslprep": "^1.0.0"
423 | }
424 | },
425 | "mongoose": {
426 | "version": "5.10.8",
427 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.10.8.tgz",
428 | "integrity": "sha512-hbpFhOU6rWkWPkekUeSJxqWwzsjVQZ9xPg4WmWA1HJ8YDvjyNye1xbp82fw67BpnyvcjHxyU3/YhujsOCx55yw==",
429 | "requires": {
430 | "bson": "^1.1.4",
431 | "kareem": "2.3.1",
432 | "mongodb": "3.6.2",
433 | "mongoose-legacy-pluralize": "1.0.2",
434 | "mpath": "0.7.0",
435 | "mquery": "3.2.2",
436 | "ms": "2.1.2",
437 | "regexp-clone": "1.0.0",
438 | "safe-buffer": "5.2.1",
439 | "sift": "7.0.1",
440 | "sliced": "1.0.1"
441 | },
442 | "dependencies": {
443 | "ms": {
444 | "version": "2.1.2",
445 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
446 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
447 | },
448 | "safe-buffer": {
449 | "version": "5.2.1",
450 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
451 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ=="
452 | }
453 | }
454 | },
455 | "mongoose-legacy-pluralize": {
456 | "version": "1.0.2",
457 | "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz",
458 | "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ=="
459 | },
460 | "morgan": {
461 | "version": "1.10.0",
462 | "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.10.0.tgz",
463 | "integrity": "sha512-AbegBVI4sh6El+1gNwvD5YIck7nSA36weD7xvIxG4in80j/UoK8AEGaWnnz8v1GxonMCltmlNs5ZKbGvl9b1XQ==",
464 | "requires": {
465 | "basic-auth": "~2.0.1",
466 | "debug": "2.6.9",
467 | "depd": "~2.0.0",
468 | "on-finished": "~2.3.0",
469 | "on-headers": "~1.0.2"
470 | },
471 | "dependencies": {
472 | "depd": {
473 | "version": "2.0.0",
474 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
475 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw=="
476 | }
477 | }
478 | },
479 | "mpath": {
480 | "version": "0.7.0",
481 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.7.0.tgz",
482 | "integrity": "sha512-Aiq04hILxhz1L+f7sjGyn7IxYzWm1zLNNXcfhDtx04kZ2Gk7uvFdgZ8ts1cWa/6d0TQmag2yR8zSGZUmp0tFNg=="
483 | },
484 | "mquery": {
485 | "version": "3.2.2",
486 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.2.tgz",
487 | "integrity": "sha512-XB52992COp0KP230I3qloVUbkLUxJIu328HBP2t2EsxSFtf4W1HPSOBWOXf1bqxK4Xbb66lfMJ+Bpfd9/yZE1Q==",
488 | "requires": {
489 | "bluebird": "3.5.1",
490 | "debug": "3.1.0",
491 | "regexp-clone": "^1.0.0",
492 | "safe-buffer": "5.1.2",
493 | "sliced": "1.0.1"
494 | },
495 | "dependencies": {
496 | "debug": {
497 | "version": "3.1.0",
498 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz",
499 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==",
500 | "requires": {
501 | "ms": "2.0.0"
502 | }
503 | }
504 | }
505 | },
506 | "ms": {
507 | "version": "2.0.0",
508 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
509 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
510 | },
511 | "negotiator": {
512 | "version": "0.6.2",
513 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
514 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
515 | },
516 | "object-assign": {
517 | "version": "4.1.1",
518 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
519 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM="
520 | },
521 | "on-finished": {
522 | "version": "2.3.0",
523 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
524 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
525 | "requires": {
526 | "ee-first": "1.1.1"
527 | }
528 | },
529 | "on-headers": {
530 | "version": "1.0.2",
531 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz",
532 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA=="
533 | },
534 | "parseurl": {
535 | "version": "1.3.3",
536 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
537 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
538 | },
539 | "path-to-regexp": {
540 | "version": "0.1.7",
541 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
542 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
543 | },
544 | "process-nextick-args": {
545 | "version": "2.0.1",
546 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
547 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag=="
548 | },
549 | "proxy-addr": {
550 | "version": "2.0.6",
551 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz",
552 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==",
553 | "requires": {
554 | "forwarded": "~0.1.2",
555 | "ipaddr.js": "1.9.1"
556 | }
557 | },
558 | "qs": {
559 | "version": "6.7.0",
560 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
561 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ=="
562 | },
563 | "range-parser": {
564 | "version": "1.2.1",
565 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
566 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
567 | },
568 | "raw-body": {
569 | "version": "2.4.0",
570 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
571 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
572 | "requires": {
573 | "bytes": "3.1.0",
574 | "http-errors": "1.7.2",
575 | "iconv-lite": "0.4.24",
576 | "unpipe": "1.0.0"
577 | }
578 | },
579 | "readable-stream": {
580 | "version": "2.3.7",
581 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz",
582 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==",
583 | "requires": {
584 | "core-util-is": "~1.0.0",
585 | "inherits": "~2.0.3",
586 | "isarray": "~1.0.0",
587 | "process-nextick-args": "~2.0.0",
588 | "safe-buffer": "~5.1.1",
589 | "string_decoder": "~1.1.1",
590 | "util-deprecate": "~1.0.1"
591 | }
592 | },
593 | "regexp-clone": {
594 | "version": "1.0.0",
595 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz",
596 | "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw=="
597 | },
598 | "require_optional": {
599 | "version": "1.0.1",
600 | "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz",
601 | "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==",
602 | "requires": {
603 | "resolve-from": "^2.0.0",
604 | "semver": "^5.1.0"
605 | }
606 | },
607 | "resolve-from": {
608 | "version": "2.0.0",
609 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz",
610 | "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c="
611 | },
612 | "safe-buffer": {
613 | "version": "5.1.2",
614 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
615 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
616 | },
617 | "safer-buffer": {
618 | "version": "2.1.2",
619 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
620 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
621 | },
622 | "saslprep": {
623 | "version": "1.0.3",
624 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz",
625 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==",
626 | "optional": true,
627 | "requires": {
628 | "sparse-bitfield": "^3.0.3"
629 | }
630 | },
631 | "semver": {
632 | "version": "5.7.1",
633 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz",
634 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ=="
635 | },
636 | "send": {
637 | "version": "0.17.1",
638 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
639 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
640 | "requires": {
641 | "debug": "2.6.9",
642 | "depd": "~1.1.2",
643 | "destroy": "~1.0.4",
644 | "encodeurl": "~1.0.2",
645 | "escape-html": "~1.0.3",
646 | "etag": "~1.8.1",
647 | "fresh": "0.5.2",
648 | "http-errors": "~1.7.2",
649 | "mime": "1.6.0",
650 | "ms": "2.1.1",
651 | "on-finished": "~2.3.0",
652 | "range-parser": "~1.2.1",
653 | "statuses": "~1.5.0"
654 | },
655 | "dependencies": {
656 | "ms": {
657 | "version": "2.1.1",
658 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
659 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
660 | }
661 | }
662 | },
663 | "serve-static": {
664 | "version": "1.14.1",
665 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
666 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
667 | "requires": {
668 | "encodeurl": "~1.0.2",
669 | "escape-html": "~1.0.3",
670 | "parseurl": "~1.3.3",
671 | "send": "0.17.1"
672 | }
673 | },
674 | "setprototypeof": {
675 | "version": "1.1.1",
676 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
677 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
678 | },
679 | "sift": {
680 | "version": "7.0.1",
681 | "resolved": "https://registry.npmjs.org/sift/-/sift-7.0.1.tgz",
682 | "integrity": "sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g=="
683 | },
684 | "sliced": {
685 | "version": "1.0.1",
686 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz",
687 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E="
688 | },
689 | "sparse-bitfield": {
690 | "version": "3.0.3",
691 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz",
692 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=",
693 | "optional": true,
694 | "requires": {
695 | "memory-pager": "^1.0.2"
696 | }
697 | },
698 | "statuses": {
699 | "version": "1.5.0",
700 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
701 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow="
702 | },
703 | "string_decoder": {
704 | "version": "1.1.1",
705 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz",
706 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==",
707 | "requires": {
708 | "safe-buffer": "~5.1.0"
709 | }
710 | },
711 | "toidentifier": {
712 | "version": "1.0.0",
713 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
714 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw=="
715 | },
716 | "type-is": {
717 | "version": "1.6.18",
718 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
719 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
720 | "requires": {
721 | "media-typer": "0.3.0",
722 | "mime-types": "~2.1.24"
723 | }
724 | },
725 | "unpipe": {
726 | "version": "1.0.0",
727 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
728 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
729 | },
730 | "util-deprecate": {
731 | "version": "1.0.2",
732 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
733 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8="
734 | },
735 | "utils-merge": {
736 | "version": "1.0.1",
737 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
738 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
739 | },
740 | "vary": {
741 | "version": "1.1.2",
742 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
743 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
744 | }
745 | }
746 | }
747 |
--------------------------------------------------------------------------------
/backend/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "blog-backend",
3 | "version": "1.0.0",
4 | "description": "Tell the world your story.",
5 | "main": "index.js",
6 | "scripts": {
7 | "test": "echo \"Error: no test specified\" && exit 1",
8 | "start": "nodemon index.js",
9 | "build" : "node index.js",
10 | "prod" : "node index.js"
11 | },
12 | "author": "",
13 | "license": "ISC",
14 | "dependencies": {
15 | "@hapi/joi": "^17.1.1",
16 | "bcryptjs": "^2.4.3",
17 | "body-parser": "^1.19.0",
18 | "cors": "^2.8.5",
19 | "dotenv": "^8.2.0",
20 | "express": "^4.17.1",
21 | "helmet": "^4.1.1",
22 | "jsonwebtoken": "^8.5.1",
23 | "method-override": "^3.0.0",
24 | "mongoose": "^5.10.8",
25 | "morgan": "^1.10.0",
26 | "slugify": "^1.4.5"
27 | }
28 | }
29 |
--------------------------------------------------------------------------------
/backend/routes/authRoute.js:
--------------------------------------------------------------------------------
1 | const express = require("express");
2 | const router = express.Router();
3 | const jwt = require("jsonwebtoken");
4 | const User = require("../models/User");
5 | const bcrypt = require("bcryptjs");
6 | const authentication = require("../middleware/authentication");
7 |
8 | router.get("/login", (req, res) => {});
9 |
10 | router.post("/login", async (req, res) => {
11 | const { email, password } = req.body;
12 |
13 | if (!email || !password)
14 | return res
15 | .json({ error: "Please make sure to enter all fields." })
16 | .status(400);
17 |
18 | await User.findOne({ email }).then((user) => {
19 | if (!user)
20 | return res.status(400).json({
21 | msg: "Invalid Credentials. Please check again.",
22 | });
23 |
24 | bcrypt.compare(password, user.password).then((isMatch) => {
25 | if (!isMatch)
26 | return res
27 | .status(400)
28 | .json({ error: "Invalid Credentials. Please check again." });
29 |
30 | jwt.sign({ id: user._id }, process.env.JWT_SECRET, (err, token) => {
31 | if (err) throw err;
32 |
33 | res.json({
34 | token: token,
35 | user: {
36 | id: user._id,
37 | username: user.username,
38 | email: user.email,
39 | avatar: user.avatar,
40 | blogs: user.blogs,
41 | isAdmin: user.isAdmin
42 | },
43 | });
44 | });
45 | });
46 | });
47 | });
48 |
49 | router.post("/tokenIsValid", async (req, res) => {
50 | try {
51 | const token = req.header("x-auth-token");
52 | if (!token) return res.json(false);
53 |
54 | const verified = jwt.verify(token, process.env.JWT_SECRET);
55 | if (!verified) return res.json(false);
56 |
57 | const user = await User.findById(verified.id);
58 | if (!user) return res.json(false);
59 |
60 | return res.json(true);
61 | } catch (err) {
62 | res.status(500).json({ error: err.message });
63 | }
64 | });
65 |
66 | router.get("/", authentication, async (req, res) => {
67 | const { username, socialMedia, _id, email, blogs } = await User.findById(
68 | req.user.id
69 | ).populate("blogs");
70 |
71 | res.json({
72 | username,
73 | socialMedia,
74 | _id,
75 | email,
76 | blogs,
77 | });
78 | });
79 |
80 | router.get("/register", (req, res) => {
81 | res.json({
82 | msg: "Registration route",
83 | });
84 | });
85 |
86 | router.post("/register", async (req, res) => {
87 | try {
88 | const { username, name, email, password, avatar, socialMedia, isAdmin } = req.body;
89 | if (!username || !email || !password)
90 | return res.status(400).json({ error: "Please fill all the fields." });
91 | const result = await User.findOne({
92 | $or: [{ email: email }, { username: username }],
93 | });
94 |
95 | if (result) {
96 | res
97 | .json({
98 | msg:
99 | "User already exist in database. Please try with a different email.",
100 | })
101 | .status(400);
102 | } else {
103 | const newUser = new User({
104 | username,
105 | name,
106 | email,
107 | password,
108 | avatar,
109 | socialMedia,
110 | isAdmin
111 | });
112 | bcrypt.genSalt(10, (err, salt) => {
113 | bcrypt.hash(newUser.password, salt, (error, hash) => {
114 | if (error) throw error;
115 |
116 | newUser.password = hash;
117 |
118 | newUser.save().then((user) => {
119 | res.json({ msg: "User registered." }).status(200);
120 | });
121 | });
122 | });
123 | }
124 | } catch (error) {
125 | res.json({ error: "Some Error occured." }).status(501);
126 | }
127 | });
128 |
129 |
130 | router.get("/adminPage", (req,res)=>{
131 | res.send('Welcome to admin page')
132 | })
133 |
134 |
135 | module.exports = router;
136 |
--------------------------------------------------------------------------------
/backend/routes/blogRoute.js:
--------------------------------------------------------------------------------
1 | const express = require("express");
2 | const router = express.Router();
3 | const User = require("../models/User");
4 | const Blog = require("../models/Blog");
5 | const authentication = require("../middleware/authentication");
6 |
7 | router.get("/", async (req, res) => {
8 | try {
9 | const blogs = await Blog.find({ isApproved: true });
10 | res.json({ blogs });
11 | } catch (error) {
12 | res.json({
13 | error: "Failed to fetch blogs ! Please try again.",
14 | });
15 | }
16 | });
17 |
18 | router.get("/unapproved", async (req, res) => {
19 | try {
20 | const blogs = await Blog.find({});
21 | res.json(blogs);
22 | } catch (error) {
23 | res.json({ msg: error });
24 | }
25 | });
26 |
27 | router.get("/new", authentication, async (req, res) => {
28 | try {
29 | } catch (error) {
30 | res.json({ error: "Some Error occured. Please try again." }).status(500);
31 | }
32 | });
33 |
34 | router.post("/new", authentication, async (req, res) => {
35 | const newBlog = new Blog({
36 | title: req.body.title,
37 | author: req.body.author,
38 | content: req.body.content,
39 | date: req.body.date,
40 | bannerURL: req.body.bannerURL,
41 | avatar: req.body.avatar,
42 | authorID: req.body.authorID,
43 | });
44 | try {
45 | const savedBlog = await newBlog.save();
46 | res.redirect("/blogs");
47 | } catch (error) {
48 | res.json({
49 | error: "Some error occured while saving your blog. Please try again.",
50 | });
51 | }
52 | });
53 |
54 | router.get("/:username/:slug", async (req, res) => {
55 | try {
56 | const foundBlog = await Blog.findOne({ slug: req.params.slug });
57 | if (foundBlog) {
58 | res.json(foundBlog);
59 | } else {
60 | throw error;
61 | }
62 | } catch (error) {
63 | res
64 | .status(404)
65 | .json({ msg: "Requested blog was not found on the server." });
66 | }
67 | });
68 |
69 | router.delete("/:username/:blogID", authentication, async (req, res) => {
70 | try {
71 | const removedBlog = await Blog.deleteOne({
72 | _id: req.params.blogID,
73 | });
74 | res.redirect("/blogs");
75 | } catch (error) {
76 | res.json(error);
77 | }
78 | });
79 |
80 | router.patch("/:author/:blogID", authentication, async (req, res) => {
81 | try {
82 | const blogToBeEdited = await Blog.updateOne(
83 | {
84 | _id: req.params.blogID,
85 | author: req.params.author,
86 | },
87 | {
88 | $set: {
89 | bannerURL: req.body.bannerURL,
90 | title: req.body.title,
91 | content: req.body.content,
92 | },
93 | }
94 | );
95 | } catch (error) {
96 | res.json({ msg: error });
97 | }
98 | });
99 |
100 | router.patch("/:blogID", authentication, async (req, res) => {
101 | try {
102 | const blogTobeApproved = await Blog.updateOne(
103 | {
104 | _id: req.params.blogID,
105 | },
106 | {
107 | $set: {
108 | isApproved: req.body.isApproved,
109 | },
110 | }
111 | );
112 |
113 | res.json(blogTobeApproved).status(200);
114 | } catch (error) {
115 | res.json({
116 | error:
117 | "Some error occured while editing the blog. Please make sure you are authenticated.",
118 | });
119 | }
120 | });
121 | module.exports = router;
122 |
--------------------------------------------------------------------------------
/backend/routes/homeRoute.js:
--------------------------------------------------------------------------------
1 | const express = require("express");
2 | const router = express.Router();
3 |
4 | router.get("/", (req, res) => {
5 | res.json({ msg: "I'm Home route of this app" });
6 | });
7 |
8 | module.exports = router;
9 |
--------------------------------------------------------------------------------
/backend/yarn.lock:
--------------------------------------------------------------------------------
1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
2 | # yarn lockfile v1
3 |
4 |
5 | "@hapi/address@^4.0.1":
6 | version "4.1.0"
7 | resolved "https://registry.yarnpkg.com/@hapi/address/-/address-4.1.0.tgz#d60c5c0d930e77456fdcde2598e77302e2955e1d"
8 | integrity sha512-SkszZf13HVgGmChdHo/PxchnSaCJ6cetVqLzyciudzZRT0jcOouIF/Q93mgjw8cce+D+4F4C1Z/WrfFN+O3VHQ==
9 | dependencies:
10 | "@hapi/hoek" "^9.0.0"
11 |
12 | "@hapi/formula@^2.0.0":
13 | version "2.0.0"
14 | resolved "https://registry.yarnpkg.com/@hapi/formula/-/formula-2.0.0.tgz#edade0619ed58c8e4f164f233cda70211e787128"
15 | integrity sha512-V87P8fv7PI0LH7LiVi8Lkf3x+KCO7pQozXRssAHNXXL9L1K+uyu4XypLXwxqVDKgyQai6qj3/KteNlrqDx4W5A==
16 |
17 | "@hapi/hoek@^9.0.0":
18 | version "9.1.0"
19 | resolved "https://registry.yarnpkg.com/@hapi/hoek/-/hoek-9.1.0.tgz#6c9eafc78c1529248f8f4d92b0799a712b6052c6"
20 | integrity sha512-i9YbZPN3QgfighY/1X1Pu118VUz2Fmmhd6b2n0/O8YVgGGfw0FbUYoA97k7FkpGJ+pLCFEDLUmAPPV4D1kpeFw==
21 |
22 | "@hapi/joi@^17.1.1":
23 | version "17.1.1"
24 | resolved "https://registry.yarnpkg.com/@hapi/joi/-/joi-17.1.1.tgz#9cc8d7e2c2213d1e46708c6260184b447c661350"
25 | integrity sha512-p4DKeZAoeZW4g3u7ZeRo+vCDuSDgSvtsB/NpfjXEHTUjSeINAi/RrVOWiVQ1isaoLzMvFEhe8n5065mQq1AdQg==
26 | dependencies:
27 | "@hapi/address" "^4.0.1"
28 | "@hapi/formula" "^2.0.0"
29 | "@hapi/hoek" "^9.0.0"
30 | "@hapi/pinpoint" "^2.0.0"
31 | "@hapi/topo" "^5.0.0"
32 |
33 | "@hapi/pinpoint@^2.0.0":
34 | version "2.0.0"
35 | resolved "https://registry.yarnpkg.com/@hapi/pinpoint/-/pinpoint-2.0.0.tgz#805b40d4dbec04fc116a73089494e00f073de8df"
36 | integrity sha512-vzXR5MY7n4XeIvLpfl3HtE3coZYO4raKXW766R6DZw/6aLqR26iuZ109K7a0NtF2Db0jxqh7xz2AxkUwpUFybw==
37 |
38 | "@hapi/topo@^5.0.0":
39 | version "5.0.0"
40 | resolved "https://registry.yarnpkg.com/@hapi/topo/-/topo-5.0.0.tgz#c19af8577fa393a06e9c77b60995af959be721e7"
41 | integrity sha512-tFJlT47db0kMqVm3H4nQYgn6Pwg10GTZHb1pwmSiv1K4ks6drQOtfEF5ZnPjkvC+y4/bUPHK+bc87QvLcL+WMw==
42 | dependencies:
43 | "@hapi/hoek" "^9.0.0"
44 |
45 | accepts@~1.3.7:
46 | version "1.3.7"
47 | resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.7.tgz#531bc726517a3b2b41f850021c6cc15eaab507cd"
48 | integrity sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==
49 | dependencies:
50 | mime-types "~2.1.24"
51 | negotiator "0.6.2"
52 |
53 | array-flatten@1.1.1:
54 | version "1.1.1"
55 | resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2"
56 | integrity sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=
57 |
58 | basic-auth@~2.0.1:
59 | version "2.0.1"
60 | resolved "https://registry.yarnpkg.com/basic-auth/-/basic-auth-2.0.1.tgz#b998279bf47ce38344b4f3cf916d4679bbf51e3a"
61 | integrity sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==
62 | dependencies:
63 | safe-buffer "5.1.2"
64 |
65 | bcryptjs@^2.4.3:
66 | version "2.4.3"
67 | resolved "https://registry.yarnpkg.com/bcryptjs/-/bcryptjs-2.4.3.tgz#9ab5627b93e60621ff7cdac5da9733027df1d0cb"
68 | integrity sha1-mrVie5PmBiH/fNrF2pczAn3x0Ms=
69 |
70 | bl@^2.2.1:
71 | version "2.2.1"
72 | resolved "https://registry.yarnpkg.com/bl/-/bl-2.2.1.tgz#8c11a7b730655c5d56898cdc871224f40fd901d5"
73 | integrity sha512-6Pesp1w0DEX1N550i/uGV/TqucVL4AM/pgThFSN/Qq9si1/DF9aIHs1BxD8V/QU0HoeHO6cQRTAuYnLPKq1e4g==
74 | dependencies:
75 | readable-stream "^2.3.5"
76 | safe-buffer "^5.1.1"
77 |
78 | bluebird@3.5.1:
79 | version "3.5.1"
80 | resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.5.1.tgz#d9551f9de98f1fcda1e683d17ee91a0602ee2eb9"
81 | integrity sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA==
82 |
83 | body-parser@1.19.0, body-parser@^1.19.0:
84 | version "1.19.0"
85 | resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.19.0.tgz#96b2709e57c9c4e09a6fd66a8fd979844f69f08a"
86 | integrity sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==
87 | dependencies:
88 | bytes "3.1.0"
89 | content-type "~1.0.4"
90 | debug "2.6.9"
91 | depd "~1.1.2"
92 | http-errors "1.7.2"
93 | iconv-lite "0.4.24"
94 | on-finished "~2.3.0"
95 | qs "6.7.0"
96 | raw-body "2.4.0"
97 | type-is "~1.6.17"
98 |
99 | bson@^1.1.4:
100 | version "1.1.5"
101 | resolved "https://registry.yarnpkg.com/bson/-/bson-1.1.5.tgz#2aaae98fcdf6750c0848b0cba1ddec3c73060a34"
102 | integrity sha512-kDuEzldR21lHciPQAIulLs1LZlCXdLziXI6Mb/TDkwXhb//UORJNPXgcRs2CuO4H0DcMkpfT3/ySsP3unoZjBg==
103 |
104 | buffer-equal-constant-time@1.0.1:
105 | version "1.0.1"
106 | resolved "https://registry.yarnpkg.com/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz#f8e71132f7ffe6e01a5c9697a4c6f3e48d5cc819"
107 | integrity sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=
108 |
109 | bytes@3.1.0:
110 | version "3.1.0"
111 | resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.0.tgz#f6cf7933a360e0588fa9fde85651cdc7f805d1f6"
112 | integrity sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==
113 |
114 | content-disposition@0.5.3:
115 | version "0.5.3"
116 | resolved "https://registry.yarnpkg.com/content-disposition/-/content-disposition-0.5.3.tgz#e130caf7e7279087c5616c2007d0485698984fbd"
117 | integrity sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==
118 | dependencies:
119 | safe-buffer "5.1.2"
120 |
121 | content-type@~1.0.4:
122 | version "1.0.4"
123 | resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.4.tgz#e138cc75e040c727b1966fe5e5f8c9aee256fe3b"
124 | integrity sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==
125 |
126 | cookie-signature@1.0.6:
127 | version "1.0.6"
128 | resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c"
129 | integrity sha1-4wOogrNCzD7oylE6eZmXNNqzriw=
130 |
131 | cookie@0.4.0:
132 | version "0.4.0"
133 | resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.4.0.tgz#beb437e7022b3b6d49019d088665303ebe9c14ba"
134 | integrity sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==
135 |
136 | core-util-is@~1.0.0:
137 | version "1.0.2"
138 | resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7"
139 | integrity sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=
140 |
141 | cors@^2.8.5:
142 | version "2.8.5"
143 | resolved "https://registry.yarnpkg.com/cors/-/cors-2.8.5.tgz#eac11da51592dd86b9f06f6e7ac293b3df875d29"
144 | integrity sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==
145 | dependencies:
146 | object-assign "^4"
147 | vary "^1"
148 |
149 | debug@2.6.9:
150 | version "2.6.9"
151 | resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f"
152 | integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==
153 | dependencies:
154 | ms "2.0.0"
155 |
156 | debug@3.1.0:
157 | version "3.1.0"
158 | resolved "https://registry.yarnpkg.com/debug/-/debug-3.1.0.tgz#5bb5a0672628b64149566ba16819e61518c67261"
159 | integrity sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==
160 | dependencies:
161 | ms "2.0.0"
162 |
163 | denque@^1.4.1:
164 | version "1.4.1"
165 | resolved "https://registry.yarnpkg.com/denque/-/denque-1.4.1.tgz#6744ff7641c148c3f8a69c307e51235c1f4a37cf"
166 | integrity sha512-OfzPuSZKGcgr96rf1oODnfjqBFmr1DVoc/TrItj3Ohe0Ah1C5WX5Baquw/9U9KovnQ88EqmJbD66rKYUQYN1tQ==
167 |
168 | depd@~1.1.2:
169 | version "1.1.2"
170 | resolved "https://registry.yarnpkg.com/depd/-/depd-1.1.2.tgz#9bcd52e14c097763e749b274c4346ed2e560b5a9"
171 | integrity sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=
172 |
173 | depd@~2.0.0:
174 | version "2.0.0"
175 | resolved "https://registry.yarnpkg.com/depd/-/depd-2.0.0.tgz#b696163cc757560d09cf22cc8fad1571b79e76df"
176 | integrity sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==
177 |
178 | destroy@~1.0.4:
179 | version "1.0.4"
180 | resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.0.4.tgz#978857442c44749e4206613e37946205826abd80"
181 | integrity sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=
182 |
183 | dotenv@^8.2.0:
184 | version "8.2.0"
185 | resolved "https://registry.yarnpkg.com/dotenv/-/dotenv-8.2.0.tgz#97e619259ada750eea3e4ea3e26bceea5424b16a"
186 | integrity sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw==
187 |
188 | ecdsa-sig-formatter@1.0.11:
189 | version "1.0.11"
190 | resolved "https://registry.yarnpkg.com/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz#ae0f0fa2d85045ef14a817daa3ce9acd0489e5bf"
191 | integrity sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==
192 | dependencies:
193 | safe-buffer "^5.0.1"
194 |
195 | ee-first@1.1.1:
196 | version "1.1.1"
197 | resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d"
198 | integrity sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=
199 |
200 | encodeurl@~1.0.2:
201 | version "1.0.2"
202 | resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59"
203 | integrity sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=
204 |
205 | escape-html@~1.0.3:
206 | version "1.0.3"
207 | resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988"
208 | integrity sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=
209 |
210 | etag@~1.8.1:
211 | version "1.8.1"
212 | resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887"
213 | integrity sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=
214 |
215 | express@^4.17.1:
216 | version "4.17.1"
217 | resolved "https://registry.yarnpkg.com/express/-/express-4.17.1.tgz#4491fc38605cf51f8629d39c2b5d026f98a4c134"
218 | integrity sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==
219 | dependencies:
220 | accepts "~1.3.7"
221 | array-flatten "1.1.1"
222 | body-parser "1.19.0"
223 | content-disposition "0.5.3"
224 | content-type "~1.0.4"
225 | cookie "0.4.0"
226 | cookie-signature "1.0.6"
227 | debug "2.6.9"
228 | depd "~1.1.2"
229 | encodeurl "~1.0.2"
230 | escape-html "~1.0.3"
231 | etag "~1.8.1"
232 | finalhandler "~1.1.2"
233 | fresh "0.5.2"
234 | merge-descriptors "1.0.1"
235 | methods "~1.1.2"
236 | on-finished "~2.3.0"
237 | parseurl "~1.3.3"
238 | path-to-regexp "0.1.7"
239 | proxy-addr "~2.0.5"
240 | qs "6.7.0"
241 | range-parser "~1.2.1"
242 | safe-buffer "5.1.2"
243 | send "0.17.1"
244 | serve-static "1.14.1"
245 | setprototypeof "1.1.1"
246 | statuses "~1.5.0"
247 | type-is "~1.6.18"
248 | utils-merge "1.0.1"
249 | vary "~1.1.2"
250 |
251 | finalhandler@~1.1.2:
252 | version "1.1.2"
253 | resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.1.2.tgz#b7e7d000ffd11938d0fdb053506f6ebabe9f587d"
254 | integrity sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==
255 | dependencies:
256 | debug "2.6.9"
257 | encodeurl "~1.0.2"
258 | escape-html "~1.0.3"
259 | on-finished "~2.3.0"
260 | parseurl "~1.3.3"
261 | statuses "~1.5.0"
262 | unpipe "~1.0.0"
263 |
264 | forwarded@~0.1.2:
265 | version "0.1.2"
266 | resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.1.2.tgz#98c23dab1175657b8c0573e8ceccd91b0ff18c84"
267 | integrity sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=
268 |
269 | fresh@0.5.2:
270 | version "0.5.2"
271 | resolved "https://registry.yarnpkg.com/fresh/-/fresh-0.5.2.tgz#3d8cadd90d976569fa835ab1f8e4b23a105605a7"
272 | integrity sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=
273 |
274 | helmet@^4.1.1:
275 | version "4.1.1"
276 | resolved "https://registry.yarnpkg.com/helmet/-/helmet-4.1.1.tgz#751f0e273d809ace9c172073e0003bed27d27a4a"
277 | integrity sha512-Avg4XxSBrehD94mkRwEljnO+6RZx7AGfk8Wa6K1nxaU+hbXlFOhlOIMgPfFqOYQB/dBCsTpootTGuiOG+CHiQA==
278 |
279 | http-errors@1.7.2:
280 | version "1.7.2"
281 | resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.7.2.tgz#4f5029cf13239f31036e5b2e55292bcfbcc85c8f"
282 | integrity sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==
283 | dependencies:
284 | depd "~1.1.2"
285 | inherits "2.0.3"
286 | setprototypeof "1.1.1"
287 | statuses ">= 1.5.0 < 2"
288 | toidentifier "1.0.0"
289 |
290 | http-errors@~1.7.2:
291 | version "1.7.3"
292 | resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.7.3.tgz#6c619e4f9c60308c38519498c14fbb10aacebb06"
293 | integrity sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw==
294 | dependencies:
295 | depd "~1.1.2"
296 | inherits "2.0.4"
297 | setprototypeof "1.1.1"
298 | statuses ">= 1.5.0 < 2"
299 | toidentifier "1.0.0"
300 |
301 | iconv-lite@0.4.24:
302 | version "0.4.24"
303 | resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b"
304 | integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==
305 | dependencies:
306 | safer-buffer ">= 2.1.2 < 3"
307 |
308 | inherits@2.0.3:
309 | version "2.0.3"
310 | resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.3.tgz#633c2c83e3da42a502f52466022480f4208261de"
311 | integrity sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=
312 |
313 | inherits@2.0.4, inherits@~2.0.3:
314 | version "2.0.4"
315 | resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c"
316 | integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==
317 |
318 | ipaddr.js@1.9.1:
319 | version "1.9.1"
320 | resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.9.1.tgz#bff38543eeb8984825079ff3a2a8e6cbd46781b3"
321 | integrity sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==
322 |
323 | isarray@~1.0.0:
324 | version "1.0.0"
325 | resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11"
326 | integrity sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=
327 |
328 | jsonwebtoken@^8.5.1:
329 | version "8.5.1"
330 | resolved "https://registry.yarnpkg.com/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz#00e71e0b8df54c2121a1f26137df2280673bcc0d"
331 | integrity sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==
332 | dependencies:
333 | jws "^3.2.2"
334 | lodash.includes "^4.3.0"
335 | lodash.isboolean "^3.0.3"
336 | lodash.isinteger "^4.0.4"
337 | lodash.isnumber "^3.0.3"
338 | lodash.isplainobject "^4.0.6"
339 | lodash.isstring "^4.0.1"
340 | lodash.once "^4.0.0"
341 | ms "^2.1.1"
342 | semver "^5.6.0"
343 |
344 | jwa@^1.4.1:
345 | version "1.4.1"
346 | resolved "https://registry.yarnpkg.com/jwa/-/jwa-1.4.1.tgz#743c32985cb9e98655530d53641b66c8645b039a"
347 | integrity sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==
348 | dependencies:
349 | buffer-equal-constant-time "1.0.1"
350 | ecdsa-sig-formatter "1.0.11"
351 | safe-buffer "^5.0.1"
352 |
353 | jws@^3.2.2:
354 | version "3.2.2"
355 | resolved "https://registry.yarnpkg.com/jws/-/jws-3.2.2.tgz#001099f3639468c9414000e99995fa52fb478304"
356 | integrity sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==
357 | dependencies:
358 | jwa "^1.4.1"
359 | safe-buffer "^5.0.1"
360 |
361 | kareem@2.3.1:
362 | version "2.3.1"
363 | resolved "https://registry.yarnpkg.com/kareem/-/kareem-2.3.1.tgz#def12d9c941017fabfb00f873af95e9c99e1be87"
364 | integrity sha512-l3hLhffs9zqoDe8zjmb/mAN4B8VT3L56EUvKNqLFVs9YlFA+zx7ke1DO8STAdDyYNkeSo1nKmjuvQeI12So8Xw==
365 |
366 | lodash.includes@^4.3.0:
367 | version "4.3.0"
368 | resolved "https://registry.yarnpkg.com/lodash.includes/-/lodash.includes-4.3.0.tgz#60bb98a87cb923c68ca1e51325483314849f553f"
369 | integrity sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=
370 |
371 | lodash.isboolean@^3.0.3:
372 | version "3.0.3"
373 | resolved "https://registry.yarnpkg.com/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz#6c2e171db2a257cd96802fd43b01b20d5f5870f6"
374 | integrity sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=
375 |
376 | lodash.isinteger@^4.0.4:
377 | version "4.0.4"
378 | resolved "https://registry.yarnpkg.com/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz#619c0af3d03f8b04c31f5882840b77b11cd68343"
379 | integrity sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=
380 |
381 | lodash.isnumber@^3.0.3:
382 | version "3.0.3"
383 | resolved "https://registry.yarnpkg.com/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz#3ce76810c5928d03352301ac287317f11c0b1ffc"
384 | integrity sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=
385 |
386 | lodash.isplainobject@^4.0.6:
387 | version "4.0.6"
388 | resolved "https://registry.yarnpkg.com/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz#7c526a52d89b45c45cc690b88163be0497f550cb"
389 | integrity sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=
390 |
391 | lodash.isstring@^4.0.1:
392 | version "4.0.1"
393 | resolved "https://registry.yarnpkg.com/lodash.isstring/-/lodash.isstring-4.0.1.tgz#d527dfb5456eca7cc9bb95d5daeaf88ba54a5451"
394 | integrity sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=
395 |
396 | lodash.once@^4.0.0:
397 | version "4.1.1"
398 | resolved "https://registry.yarnpkg.com/lodash.once/-/lodash.once-4.1.1.tgz#0dd3971213c7c56df880977d504c88fb471a97ac"
399 | integrity sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=
400 |
401 | media-typer@0.3.0:
402 | version "0.3.0"
403 | resolved "https://registry.yarnpkg.com/media-typer/-/media-typer-0.3.0.tgz#8710d7af0aa626f8fffa1ce00168545263255748"
404 | integrity sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=
405 |
406 | memory-pager@^1.0.2:
407 | version "1.5.0"
408 | resolved "https://registry.yarnpkg.com/memory-pager/-/memory-pager-1.5.0.tgz#d8751655d22d384682741c972f2c3d6dfa3e66b5"
409 | integrity sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==
410 |
411 | merge-descriptors@1.0.1:
412 | version "1.0.1"
413 | resolved "https://registry.yarnpkg.com/merge-descriptors/-/merge-descriptors-1.0.1.tgz#b00aaa556dd8b44568150ec9d1b953f3f90cbb61"
414 | integrity sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=
415 |
416 | method-override@^3.0.0:
417 | version "3.0.0"
418 | resolved "https://registry.yarnpkg.com/method-override/-/method-override-3.0.0.tgz#6ab0d5d574e3208f15b0c9cf45ab52000468d7a2"
419 | integrity sha512-IJ2NNN/mSl9w3kzWB92rcdHpz+HjkxhDJWNDBqSlas+zQdP8wBiJzITPg08M/k2uVvMow7Sk41atndNtt/PHSA==
420 | dependencies:
421 | debug "3.1.0"
422 | methods "~1.1.2"
423 | parseurl "~1.3.2"
424 | vary "~1.1.2"
425 |
426 | methods@~1.1.2:
427 | version "1.1.2"
428 | resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee"
429 | integrity sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=
430 |
431 | mime-db@1.44.0:
432 | version "1.44.0"
433 | resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.44.0.tgz#fa11c5eb0aca1334b4233cb4d52f10c5a6272f92"
434 | integrity sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==
435 |
436 | mime-types@~2.1.24:
437 | version "2.1.27"
438 | resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.27.tgz#47949f98e279ea53119f5722e0f34e529bec009f"
439 | integrity sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==
440 | dependencies:
441 | mime-db "1.44.0"
442 |
443 | mime@1.6.0:
444 | version "1.6.0"
445 | resolved "https://registry.yarnpkg.com/mime/-/mime-1.6.0.tgz#32cd9e5c64553bd58d19a568af452acff04981b1"
446 | integrity sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==
447 |
448 | mongodb@3.6.2:
449 | version "3.6.2"
450 | resolved "https://registry.yarnpkg.com/mongodb/-/mongodb-3.6.2.tgz#1154a4ac107bf1375112d83a29c5cf97704e96b6"
451 | integrity sha512-sSZOb04w3HcnrrXC82NEh/YGCmBuRgR+C1hZgmmv4L6dBz4BkRse6Y8/q/neXer9i95fKUBbFi4KgeceXmbsOA==
452 | dependencies:
453 | bl "^2.2.1"
454 | bson "^1.1.4"
455 | denque "^1.4.1"
456 | require_optional "^1.0.1"
457 | safe-buffer "^5.1.2"
458 | optionalDependencies:
459 | saslprep "^1.0.0"
460 |
461 | mongoose-legacy-pluralize@1.0.2:
462 | version "1.0.2"
463 | resolved "https://registry.yarnpkg.com/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz#3ba9f91fa507b5186d399fb40854bff18fb563e4"
464 | integrity sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ==
465 |
466 | mongoose@^5.10.8:
467 | version "5.10.9"
468 | resolved "https://registry.yarnpkg.com/mongoose/-/mongoose-5.10.9.tgz#23b74debc86d2055cee4fe77f962a9c8a286cdad"
469 | integrity sha512-7dkr1d6Uyk87hELzoc6B7Zo7kkPTx8rKummk51Y0je2V2Ttsw0KFPwTp1G8JIbBta7Wpw8j15PJi0d33Ode2nw==
470 | dependencies:
471 | bson "^1.1.4"
472 | kareem "2.3.1"
473 | mongodb "3.6.2"
474 | mongoose-legacy-pluralize "1.0.2"
475 | mpath "0.7.0"
476 | mquery "3.2.2"
477 | ms "2.1.2"
478 | regexp-clone "1.0.0"
479 | safe-buffer "5.2.1"
480 | sift "7.0.1"
481 | sliced "1.0.1"
482 |
483 | morgan@^1.10.0:
484 | version "1.10.0"
485 | resolved "https://registry.yarnpkg.com/morgan/-/morgan-1.10.0.tgz#091778abc1fc47cd3509824653dae1faab6b17d7"
486 | integrity sha512-AbegBVI4sh6El+1gNwvD5YIck7nSA36weD7xvIxG4in80j/UoK8AEGaWnnz8v1GxonMCltmlNs5ZKbGvl9b1XQ==
487 | dependencies:
488 | basic-auth "~2.0.1"
489 | debug "2.6.9"
490 | depd "~2.0.0"
491 | on-finished "~2.3.0"
492 | on-headers "~1.0.2"
493 |
494 | mpath@0.7.0:
495 | version "0.7.0"
496 | resolved "https://registry.yarnpkg.com/mpath/-/mpath-0.7.0.tgz#20e8102e276b71709d6e07e9f8d4d0f641afbfb8"
497 | integrity sha512-Aiq04hILxhz1L+f7sjGyn7IxYzWm1zLNNXcfhDtx04kZ2Gk7uvFdgZ8ts1cWa/6d0TQmag2yR8zSGZUmp0tFNg==
498 |
499 | mquery@3.2.2:
500 | version "3.2.2"
501 | resolved "https://registry.yarnpkg.com/mquery/-/mquery-3.2.2.tgz#e1383a3951852ce23e37f619a9b350f1fb3664e7"
502 | integrity sha512-XB52992COp0KP230I3qloVUbkLUxJIu328HBP2t2EsxSFtf4W1HPSOBWOXf1bqxK4Xbb66lfMJ+Bpfd9/yZE1Q==
503 | dependencies:
504 | bluebird "3.5.1"
505 | debug "3.1.0"
506 | regexp-clone "^1.0.0"
507 | safe-buffer "5.1.2"
508 | sliced "1.0.1"
509 |
510 | ms@2.0.0:
511 | version "2.0.0"
512 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8"
513 | integrity sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=
514 |
515 | ms@2.1.1:
516 | version "2.1.1"
517 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.1.tgz#30a5864eb3ebb0a66f2ebe6d727af06a09d86e0a"
518 | integrity sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==
519 |
520 | ms@2.1.2, ms@^2.1.1:
521 | version "2.1.2"
522 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009"
523 | integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==
524 |
525 | negotiator@0.6.2:
526 | version "0.6.2"
527 | resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.2.tgz#feacf7ccf525a77ae9634436a64883ffeca346fb"
528 | integrity sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==
529 |
530 | object-assign@^4:
531 | version "4.1.1"
532 | resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863"
533 | integrity sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=
534 |
535 | on-finished@~2.3.0:
536 | version "2.3.0"
537 | resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.3.0.tgz#20f1336481b083cd75337992a16971aa2d906947"
538 | integrity sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=
539 | dependencies:
540 | ee-first "1.1.1"
541 |
542 | on-headers@~1.0.2:
543 | version "1.0.2"
544 | resolved "https://registry.yarnpkg.com/on-headers/-/on-headers-1.0.2.tgz#772b0ae6aaa525c399e489adfad90c403eb3c28f"
545 | integrity sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==
546 |
547 | parseurl@~1.3.2, parseurl@~1.3.3:
548 | version "1.3.3"
549 | resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.3.tgz#9da19e7bee8d12dff0513ed5b76957793bc2e8d4"
550 | integrity sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==
551 |
552 | path-to-regexp@0.1.7:
553 | version "0.1.7"
554 | resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-0.1.7.tgz#df604178005f522f15eb4490e7247a1bfaa67f8c"
555 | integrity sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=
556 |
557 | process-nextick-args@~2.0.0:
558 | version "2.0.1"
559 | resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz#7820d9b16120cc55ca9ae7792680ae7dba6d7fe2"
560 | integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==
561 |
562 | proxy-addr@~2.0.5:
563 | version "2.0.6"
564 | resolved "https://registry.yarnpkg.com/proxy-addr/-/proxy-addr-2.0.6.tgz#fdc2336505447d3f2f2c638ed272caf614bbb2bf"
565 | integrity sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==
566 | dependencies:
567 | forwarded "~0.1.2"
568 | ipaddr.js "1.9.1"
569 |
570 | qs@6.7.0:
571 | version "6.7.0"
572 | resolved "https://registry.yarnpkg.com/qs/-/qs-6.7.0.tgz#41dc1a015e3d581f1621776be31afb2876a9b1bc"
573 | integrity sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==
574 |
575 | range-parser@~1.2.1:
576 | version "1.2.1"
577 | resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.1.tgz#3cf37023d199e1c24d1a55b84800c2f3e6468031"
578 | integrity sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==
579 |
580 | raw-body@2.4.0:
581 | version "2.4.0"
582 | resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.4.0.tgz#a1ce6fb9c9bc356ca52e89256ab59059e13d0332"
583 | integrity sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==
584 | dependencies:
585 | bytes "3.1.0"
586 | http-errors "1.7.2"
587 | iconv-lite "0.4.24"
588 | unpipe "1.0.0"
589 |
590 | readable-stream@^2.3.5:
591 | version "2.3.7"
592 | resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.7.tgz#1eca1cf711aef814c04f62252a36a62f6cb23b57"
593 | integrity sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==
594 | dependencies:
595 | core-util-is "~1.0.0"
596 | inherits "~2.0.3"
597 | isarray "~1.0.0"
598 | process-nextick-args "~2.0.0"
599 | safe-buffer "~5.1.1"
600 | string_decoder "~1.1.1"
601 | util-deprecate "~1.0.1"
602 |
603 | regexp-clone@1.0.0, regexp-clone@^1.0.0:
604 | version "1.0.0"
605 | resolved "https://registry.yarnpkg.com/regexp-clone/-/regexp-clone-1.0.0.tgz#222db967623277056260b992626354a04ce9bf63"
606 | integrity sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw==
607 |
608 | require_optional@^1.0.1:
609 | version "1.0.1"
610 | resolved "https://registry.yarnpkg.com/require_optional/-/require_optional-1.0.1.tgz#4cf35a4247f64ca3df8c2ef208cc494b1ca8fc2e"
611 | integrity sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==
612 | dependencies:
613 | resolve-from "^2.0.0"
614 | semver "^5.1.0"
615 |
616 | resolve-from@^2.0.0:
617 | version "2.0.0"
618 | resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-2.0.0.tgz#9480ab20e94ffa1d9e80a804c7ea147611966b57"
619 | integrity sha1-lICrIOlP+h2egKgEx+oUdhGWa1c=
620 |
621 | safe-buffer@5.1.2, safe-buffer@~5.1.0, safe-buffer@~5.1.1:
622 | version "5.1.2"
623 | resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d"
624 | integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==
625 |
626 | safe-buffer@5.2.1, safe-buffer@^5.0.1, safe-buffer@^5.1.1, safe-buffer@^5.1.2:
627 | version "5.2.1"
628 | resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6"
629 | integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==
630 |
631 | "safer-buffer@>= 2.1.2 < 3":
632 | version "2.1.2"
633 | resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a"
634 | integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==
635 |
636 | saslprep@^1.0.0:
637 | version "1.0.3"
638 | resolved "https://registry.yarnpkg.com/saslprep/-/saslprep-1.0.3.tgz#4c02f946b56cf54297e347ba1093e7acac4cf226"
639 | integrity sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==
640 | dependencies:
641 | sparse-bitfield "^3.0.3"
642 |
643 | semver@^5.1.0, semver@^5.6.0:
644 | version "5.7.1"
645 | resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.1.tgz#a954f931aeba508d307bbf069eff0c01c96116f7"
646 | integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==
647 |
648 | send@0.17.1:
649 | version "0.17.1"
650 | resolved "https://registry.yarnpkg.com/send/-/send-0.17.1.tgz#c1d8b059f7900f7466dd4938bdc44e11ddb376c8"
651 | integrity sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==
652 | dependencies:
653 | debug "2.6.9"
654 | depd "~1.1.2"
655 | destroy "~1.0.4"
656 | encodeurl "~1.0.2"
657 | escape-html "~1.0.3"
658 | etag "~1.8.1"
659 | fresh "0.5.2"
660 | http-errors "~1.7.2"
661 | mime "1.6.0"
662 | ms "2.1.1"
663 | on-finished "~2.3.0"
664 | range-parser "~1.2.1"
665 | statuses "~1.5.0"
666 |
667 | serve-static@1.14.1:
668 | version "1.14.1"
669 | resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.14.1.tgz#666e636dc4f010f7ef29970a88a674320898b2f9"
670 | integrity sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==
671 | dependencies:
672 | encodeurl "~1.0.2"
673 | escape-html "~1.0.3"
674 | parseurl "~1.3.3"
675 | send "0.17.1"
676 |
677 | setprototypeof@1.1.1:
678 | version "1.1.1"
679 | resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.1.1.tgz#7e95acb24aa92f5885e0abef5ba131330d4ae683"
680 | integrity sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==
681 |
682 | sift@7.0.1:
683 | version "7.0.1"
684 | resolved "https://registry.yarnpkg.com/sift/-/sift-7.0.1.tgz#47d62c50b159d316f1372f8b53f9c10cd21a4b08"
685 | integrity sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g==
686 |
687 | sliced@1.0.1:
688 | version "1.0.1"
689 | resolved "https://registry.yarnpkg.com/sliced/-/sliced-1.0.1.tgz#0b3a662b5d04c3177b1926bea82b03f837a2ef41"
690 | integrity sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E=
691 |
692 | slugify@^1.4.5:
693 | version "1.4.5"
694 | resolved "https://registry.yarnpkg.com/slugify/-/slugify-1.4.5.tgz#a7517acf5f4c02a4df41e735354b660a4ed1efcf"
695 | integrity sha512-WpECLAgYaxHoEAJ8Q1Lo8HOs1ngn7LN7QjXgOLbmmfkcWvosyk4ZTXkTzKyhngK640USTZUlgoQJfED1kz5fnQ==
696 |
697 | sparse-bitfield@^3.0.3:
698 | version "3.0.3"
699 | resolved "https://registry.yarnpkg.com/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz#ff4ae6e68656056ba4b3e792ab3334d38273ca11"
700 | integrity sha1-/0rm5oZWBWuks+eSqzM004JzyhE=
701 | dependencies:
702 | memory-pager "^1.0.2"
703 |
704 | "statuses@>= 1.5.0 < 2", statuses@~1.5.0:
705 | version "1.5.0"
706 | resolved "https://registry.yarnpkg.com/statuses/-/statuses-1.5.0.tgz#161c7dac177659fd9811f43771fa99381478628c"
707 | integrity sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=
708 |
709 | string_decoder@~1.1.1:
710 | version "1.1.1"
711 | resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz#9cf1611ba62685d7030ae9e4ba34149c3af03fc8"
712 | integrity sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==
713 | dependencies:
714 | safe-buffer "~5.1.0"
715 |
716 | toidentifier@1.0.0:
717 | version "1.0.0"
718 | resolved "https://registry.yarnpkg.com/toidentifier/-/toidentifier-1.0.0.tgz#7e1be3470f1e77948bc43d94a3c8f4d7752ba553"
719 | integrity sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==
720 |
721 | type-is@~1.6.17, type-is@~1.6.18:
722 | version "1.6.18"
723 | resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.18.tgz#4e552cd05df09467dcbc4ef739de89f2cf37c131"
724 | integrity sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==
725 | dependencies:
726 | media-typer "0.3.0"
727 | mime-types "~2.1.24"
728 |
729 | unpipe@1.0.0, unpipe@~1.0.0:
730 | version "1.0.0"
731 | resolved "https://registry.yarnpkg.com/unpipe/-/unpipe-1.0.0.tgz#b2bf4ee8514aae6165b4817829d21b2ef49904ec"
732 | integrity sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=
733 |
734 | util-deprecate@~1.0.1:
735 | version "1.0.2"
736 | resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf"
737 | integrity sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=
738 |
739 | utils-merge@1.0.1:
740 | version "1.0.1"
741 | resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713"
742 | integrity sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=
743 |
744 | vary@^1, vary@~1.1.2:
745 | version "1.1.2"
746 | resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc"
747 | integrity sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=
748 |
--------------------------------------------------------------------------------
/frontend/.gitignore:
--------------------------------------------------------------------------------
1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
2 |
3 | # dependencies
4 | /node_modules
5 | /.pnp
6 | .pnp.js
7 |
8 | # testing
9 | /coverage
10 |
11 | # production
12 | /build
13 |
14 | # misc
15 | .DS_Store
16 | .env.local
17 | .env.development.local
18 | .env.test.local
19 | .env.production.local
20 |
21 | npm-debug.log*
22 | yarn-debug.log*
23 | yarn-error.log*
24 |
--------------------------------------------------------------------------------
/frontend/README.md:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/README.md
--------------------------------------------------------------------------------
/frontend/debug.log:
--------------------------------------------------------------------------------
1 | [1010/120550.277:ERROR:directory_reader_win.cc(43)] FindFirstFile: The system cannot find the path specified. (0x3)
2 | [1022/172042.093:ERROR:directory_reader_win.cc(43)] FindFirstFile: The system cannot find the path specified. (0x3)
3 |
--------------------------------------------------------------------------------
/frontend/package - Copy.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "frontend",
3 | "proxy": "http://localhost:3001",
4 | "version": "0.1.0",
5 | "private": true,
6 | "dependencies": {
7 | "@chakra-ui/core": "^0.8.0",
8 | "@emotion/core": "^10.0.35",
9 | "@emotion/styled": "^10.0.27",
10 | "@testing-library/jest-dom": "^4.2.4",
11 | "@testing-library/react": "^9.3.2",
12 | "@testing-library/user-event": "^7.1.2",
13 | "@uiw/react-md-editor": "^1.14.4",
14 | "axios": "^0.20.0",
15 | "axios-hooks": "^2.1.0",
16 | "chakra-ui-markdown-renderer": "^0.2.0",
17 | "date-fns": "^2.16.1",
18 | "emotion-theming": "^10.0.27",
19 | "node-sass": "^4.14.1",
20 | "react": "^16.13.1",
21 | "react-dom": "^16.13.1",
22 | "react-icons": "^3.11.0",
23 | "react-loading-skeleton": "^2.1.1",
24 | "react-markdown": "^4.3.1",
25 | "react-router-dom": "^5.2.0",
26 | "react-scripts": "3.4.3"
27 | },
28 | "scripts": {
29 | "start": "react-scripts start",
30 | "build": "react-scripts build",
31 | "test": "react-scripts test",
32 | "eject": "react-scripts eject"
33 | },
34 | "eslintConfig": {
35 | "extends": "react-app"
36 | },
37 | "browserslist": {
38 | "production": [
39 | ">0.2%",
40 | "not dead",
41 | "not op_mini all"
42 | ],
43 | "development": [
44 | "last 1 chrome version",
45 | "last 1 firefox version",
46 | "last 1 safari version"
47 | ]
48 | }
49 | }
50 |
--------------------------------------------------------------------------------
/frontend/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "frontend",
3 | "version": "0.1.0",
4 | "private": true,
5 | "dependencies": {
6 | "@chakra-ui/core": "^0.8.0",
7 | "@emotion/core": "^10.0.35",
8 | "@emotion/styled": "^10.0.27",
9 | "@testing-library/dom": "^7.25.0",
10 | "@testing-library/jest-dom": "^4.2.4",
11 | "@testing-library/react": "^9.3.2",
12 | "@testing-library/user-event": "^7.1.2",
13 | "@types/node": "14.0.5",
14 | "@uiw/react-md-editor": "^1.14.4",
15 | "axios": "0.19.0",
16 | "axios-hooks": "^2.1.0",
17 | "emotion-theming": "^10.0.27",
18 | "node-sass": "^4.14.1",
19 | "package.json": "^2.0.1",
20 | "prop-types": "15.6.2",
21 | "react": "^16.13.1",
22 | "react-dom": "^16.13.1",
23 | "react-helmet": "^6.1.0",
24 | "react-icons": "^3.11.0",
25 | "react-loading-skeleton": "^2.1.1",
26 | "react-markdown": "^4.3.1",
27 | "react-router-dom": "^5.2.0",
28 | "react-scripts": "3.4.3",
29 | "react-share": "^4.3.1",
30 | "react-toastify": "^6.0.9",
31 | "shortid": "^2.2.15"
32 | },
33 | "scripts": {
34 | "start": "react-scripts start",
35 | "build": "react-scripts build",
36 | "test": "react-scripts test",
37 | "eject": "react-scripts eject"
38 | },
39 | "eslintConfig": {
40 | "extends": "react-app"
41 | },
42 | "browserslist": {
43 | "production": [
44 | ">0.2%",
45 | "not dead",
46 | "not op_mini all"
47 | ],
48 | "development": [
49 | "last 1 chrome version",
50 | "last 1 firefox version",
51 | "last 1 safari version"
52 | ]
53 | }
54 | }
55 |
--------------------------------------------------------------------------------
/frontend/public/_redirects:
--------------------------------------------------------------------------------
1 | /* /index.html 200
--------------------------------------------------------------------------------
/frontend/public/android-chrome-192x192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/public/android-chrome-192x192.png
--------------------------------------------------------------------------------
/frontend/public/android-chrome-512x512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/public/android-chrome-512x512.png
--------------------------------------------------------------------------------
/frontend/public/apple-touch-icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/public/apple-touch-icon.png
--------------------------------------------------------------------------------
/frontend/public/favicon-16x16.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/public/favicon-16x16.png
--------------------------------------------------------------------------------
/frontend/public/favicon-32x32.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/public/favicon-32x32.png
--------------------------------------------------------------------------------
/frontend/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/public/favicon.ico
--------------------------------------------------------------------------------
/frontend/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 | Blog App
10 |
11 |
12 |
13 |
14 |
15 |
16 |
--------------------------------------------------------------------------------
/frontend/public/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "Semicolon;",
3 | "short_name": "Semicolon;",
4 | "icons": [
5 | {
6 | "src": "/android-chrome-192x192.png",
7 | "sizes": "192x192",
8 | "type": "image/png"
9 | },
10 | {
11 | "src": "/android-chrome-512x512.png",
12 | "sizes": "512x512",
13 | "type": "image/png"
14 | }
15 | ],
16 | "theme_color": "#ffffff",
17 | "background_color": "#ffffff",
18 | "display": "standalone"
19 | }
20 |
--------------------------------------------------------------------------------
/frontend/public/robots.txt:
--------------------------------------------------------------------------------
1 | # https://www.robotstxt.org/robotstxt.html
2 | User-agent: *
3 | Disallow:
4 |
--------------------------------------------------------------------------------
/frontend/public/site.webmanifest:
--------------------------------------------------------------------------------
1 | {
2 | "name": "Semicolon;",
3 | "short_name": "Semicolon;",
4 | "icons": [
5 | {
6 | "src": "/android-chrome-192x192.png",
7 | "sizes": "192x192",
8 | "type": "image/png"
9 | },
10 | {
11 | "src": "/android-chrome-512x512.png",
12 | "sizes": "512x512",
13 | "type": "image/png"
14 | }
15 | ],
16 | "theme_color": "#ffffff",
17 | "background_color": "#ffffff",
18 | "display": "standalone"
19 | }
20 |
--------------------------------------------------------------------------------
/frontend/src/App.js:
--------------------------------------------------------------------------------
1 | import React, { useState, useEffect } from "react";
2 | import {
3 | BrowserRouter as Router,
4 | Route,
5 | Link,
6 | Switch,
7 | Redirect,
8 | } from "react-router-dom";
9 | import Home from "./components/pages/Home/Home.component";
10 | import Blog from "./components/pages/Blog/Blog.component";
11 | import Blogs from "./components/pages/Blogs/Blogs.component";
12 | import NewBlog from "./components/pages/NewBlog/NewBlog.component";
13 | import Login from "./components/pages/Auth/Login.component";
14 | import Register from "./components/pages/Auth/Register.component";
15 | import NotFound from "./components/pages/404/NotFound.component";
16 | import Header from "./components/common/Header/Header.component";
17 | import AdminPage from "./components/pages/AdminPage/AdminPage.component";
18 | import EditPage from "./components/pages/EditPage/EditPage.component";
19 | import axios from "axios";
20 | import userContext from "./context/userContext";
21 |
22 | const App = () => {
23 | const [userData, setUserData] = useState({
24 | token: undefined,
25 | user: undefined,
26 | isAdmin: false,
27 | });
28 |
29 | useEffect(() => {
30 | const checkLoggedIn = async () => {
31 | let token = localStorage.getItem("x-auth-token");
32 | if (token === null) {
33 | localStorage.setItem("x-auth-token", "");
34 | token = "";
35 | }
36 | try {
37 | const tokenRes = await axios.post(
38 | process.env.REACT_APP_BACKEND + "/user/tokenIsValid",
39 | null,
40 | {
41 | headers: { "x-auth-token": token },
42 | }
43 | );
44 | if (tokenRes.data) {
45 | const userRes = await axios.get(
46 | process.env.REACT_APP_BACKEND + "/user",
47 | {
48 | headers: { "x-auth-token": token },
49 | }
50 | );
51 | setUserData({
52 | token,
53 | user: userRes.data,
54 | });
55 | }
56 | } catch (error) {
57 | console.log("Not logged in");
58 | }
59 | };
60 |
61 | checkLoggedIn();
62 | }, []);
63 |
64 | return (
65 | <>
66 |
67 | {/* Routing of the app */}
68 |
69 |
70 |
71 |
72 |
73 | }
77 | >
78 |
79 |
80 |
81 |
82 | {userData.isAdmin ? (
83 |
84 | ) : (
85 |
86 | )}
87 |
88 |
89 |
90 |
91 | >
92 | );
93 | };
94 |
95 | export default App;
96 |
--------------------------------------------------------------------------------
/frontend/src/App.test.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import { render } from '@testing-library/react';
3 | import App from './App';
4 |
5 | test('renders learn react link', () => {
6 | const { getByText } = render();
7 | const linkElement = getByText(/learn react/i);
8 | expect(linkElement).toBeInTheDocument();
9 | });
10 |
--------------------------------------------------------------------------------
/frontend/src/assets/Icons/Icons.jsx:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/src/assets/Icons/Icons.jsx
--------------------------------------------------------------------------------
/frontend/src/assets/Images/bg.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/src/assets/Images/bg.jpg
--------------------------------------------------------------------------------
/frontend/src/assets/Images/bgp3.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/src/assets/Images/bgp3.png
--------------------------------------------------------------------------------
/frontend/src/assets/Images/home.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/frontend/src/assets/Images/meta.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/src/assets/Images/meta.jpg
--------------------------------------------------------------------------------
/frontend/src/components/common/AuthorMenu/AuthorMenu.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useContext } from "react";
2 | import {
3 | Menu,
4 | MenuButton,
5 | MenuList,
6 | MenuItem,
7 | MenuGroup,
8 | MenuDivider,
9 | MenuOptionGroup,
10 | MenuItemOption,
11 | Text,
12 | Button,
13 | } from "@chakra-ui/core";
14 | import Delete from "./DeleteButton.component";
15 | import userContext from "../../../context/userContext";
16 | import { useEffect } from "react";
17 | import Edit from "./Edit.component";
18 | import { HiMenu, HiOutlineTrash } from "react-icons/hi";
19 | import EditPage from "./../../pages/EditPage/EditPage.component";
20 |
21 | const AuthorMenu = (props) => {
22 | const { userData } = useContext(userContext);
23 |
24 | return (
25 |
26 | {props.blogInfo ? (
27 | props.blogInfo.authorID === (userData.user.id || props.data._id) ? (
28 |
29 |
42 |
43 | ) : (
44 | ""
45 | )
46 | ) : (
47 | ""
48 | )}
49 |
50 | );
51 | };
52 |
53 | export default AuthorMenu;
54 |
--------------------------------------------------------------------------------
/frontend/src/components/common/AuthorMenu/DeleteButton.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useContext } from "react";
2 | import { Text, Button, useToast, Box } from "@chakra-ui/core";
3 | import { Link } from "react-router-dom";
4 | import axios from "axios";
5 | import { useHistory } from "react-router-dom";
6 | import userContext from "../../../context/userContext";
7 | import theme from "../../../themes/theme";
8 | import { HiOutlineTrash } from "react-icons/hi";
9 |
10 | const Delete = (props) => {
11 | const toast = useToast();
12 | const history = useHistory();
13 | const { userData } = useContext(userContext);
14 | const handleClick = () => {
15 | const { author, _id } = props.data.blogInfo;
16 | const reqURL = `/blogs/${author}/${_id}`;
17 | const options = {
18 | headers: {
19 | "x-auth-token": userData.token,
20 | },
21 | };
22 | axios
23 | .delete(process.env.REACT_APP_BACKEND + reqURL, options)
24 | .then((res) => {
25 | toast({
26 | title: "Blog deleted!",
27 | status: "success",
28 | isClosable: true,
29 | });
30 | history.goBack();
31 | })
32 | .catch((err) => {
33 | toast({
34 | title: "Success!",
35 | description: "Blog deleted. Refresh to see changes.",
36 | status: "success",
37 | isClosable: true,
38 | });
39 | history.goBack();
40 | });
41 | };
42 |
43 | return (
44 |
50 |
51 |
52 |
53 | Unpublish
54 |
55 | );
56 | };
57 |
58 | export default Delete;
59 |
--------------------------------------------------------------------------------
/frontend/src/components/common/AuthorMenu/Edit.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useContext } from "react";
2 | import { Box, Text } from "@chakra-ui/core";
3 | import { HiOutlinePencilAlt } from "react-icons/hi";
4 | import userContext from "../../../context/userContext";
5 | import { Link } from "react-router-dom";
6 |
7 | const Edit = (props) => {
8 | return (
9 |
15 |
16 |
17 | Edit
18 |
19 |
20 | );
21 | };
22 |
23 | export default Edit;
24 |
--------------------------------------------------------------------------------
/frontend/src/components/common/Footer/Footer.component.jsx:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 |
3 | const Footer=(props)=> {
4 | return (
5 | Footer
6 | );
7 | }
8 |
9 | export default Footer;
--------------------------------------------------------------------------------
/frontend/src/components/common/Footer/Footer.styles.scss:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/src/components/common/Footer/Footer.styles.scss
--------------------------------------------------------------------------------
/frontend/src/components/common/Header/Header.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useContext } from "react";
2 | import userContext from "../../../context/userContext";
3 | import "./Header.styles.scss";
4 | import {
5 | Heading,
6 | Flex,
7 | Box,
8 | Text,
9 | Button,
10 | useToast,
11 | List,
12 | ListItem,
13 | } from "@chakra-ui/core";
14 | import { Link } from "react-router-dom";
15 | import theme from "../../../themes/theme";
16 | import { AiOutlineFileAdd } from "react-icons/ai";
17 | import { useHistory } from "react-router-dom";
18 |
19 | const Header = (props) => {
20 | const { userData, setUserData } = useContext(userContext);
21 | const history = useHistory();
22 | const toast = useToast();
23 |
24 | const logout = () => {
25 | toast({
26 | title: "Successfully logged out.",
27 | status: "info",
28 | });
29 | setUserData({
30 | token: undefined,
31 | user: undefined,
32 | isAdmin: false,
33 | });
34 | history.push("/");
35 | localStorage.setItem("auth-token", "");
36 | localStorage.setItem("x-auth-token", "");
37 | };
38 |
39 | return (
40 |
41 |
42 |
43 |
44 |
45 |
46 |
53 | Semicolon;
54 |
55 |
56 |
57 |
58 |
59 |
60 | EXPLORE
61 |
62 |
63 | {userData.user ? (
64 |
65 |
85 |
93 |
94 | ) : (
95 | <>
96 |
97 |
98 | REGISTER
99 |
100 |
101 |
102 |
111 |
112 | >
113 | )}
114 |
115 |
116 |
117 |
118 |
119 | );
120 | };
121 |
122 | export default Header;
123 |
--------------------------------------------------------------------------------
/frontend/src/components/common/Header/Header.styles.css:
--------------------------------------------------------------------------------
1 | h1 {
2 | font-size: 3rem;
3 | }
4 | /*# sourceMappingURL=Header.styles.css.map */
--------------------------------------------------------------------------------
/frontend/src/components/common/Header/Header.styles.css.map:
--------------------------------------------------------------------------------
1 | {
2 | "version": 3,
3 | "mappings": "AAAA,AAAA,EAAE,CAAA;EACE,SAAS,EAAE,IAAI;CAClB",
4 | "sources": [
5 | "Header.styles.scss"
6 | ],
7 | "names": [],
8 | "file": "Header.styles.css"
9 | }
--------------------------------------------------------------------------------
/frontend/src/components/common/Header/Header.styles.scss:
--------------------------------------------------------------------------------
1 | .header {
2 | border-bottom: 1px solid #eee;
3 | box-shadow: 0 1px 1px rgba(0, 0, 0, 0.1);
4 | }
5 | .nav {
6 | display: flex;
7 | max-width: 1100px;
8 | margin: 0 auto;
9 | padding: 0.6rem 0;
10 | justify-content: space-between;
11 | align-items: center;
12 | color: black;
13 | }
14 | .links {
15 | display: flex;
16 | justify-content: center;
17 | align-items: center;
18 | a {
19 | margin: 0 2rem;
20 | }
21 |
22 | }
23 | .btn {
24 | background-color: black;
25 | color: white;
26 | padding: 0.4rem 0.8rem;
27 | border-radius: 6px;
28 | }
29 | @media (max-width: 768px) {
30 | .nav {
31 | padding: 0.6rem 0.4rem;
32 | }
33 | .links {
34 | a {
35 | margin: 0 0.6rem;
36 | }
37 | }
38 | }
39 |
40 | @media (max-width: 992px) {
41 | .nav {
42 | padding: 0.5rem 0.2rem;
43 | }
44 | }
45 |
--------------------------------------------------------------------------------
/frontend/src/components/common/SEO/SEO.component.jsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { Helmet } from "react-helmet";
3 |
4 | const SEO = (props) => {
5 | return (
6 |
7 | {props.title}
8 |
9 |
13 |
14 |
15 |
16 |
17 |
21 |
22 |
23 | );
24 | };
25 |
26 | export default SEO;
27 |
--------------------------------------------------------------------------------
/frontend/src/components/common/SocialMedia/SocialMedia.component.jsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { AiFillTwitterCircle, AiFillLinkedin } from "react-icons/ai";
3 | import { FaReddit } from "react-icons/fa";
4 | import { IoLogoWhatsapp } from "react-icons/io";
5 | import { Flex, Box } from "@chakra-ui/core";
6 |
7 | const SocialMedia = (props) => {
8 | return (
9 |
17 |
18 |
27 |
28 |
29 |
30 | {/*
31 |
32 |
33 |
34 | */}
35 |
36 |
44 |
45 |
46 |
47 |
48 |
55 |
56 |
57 |
58 |
59 | );
60 | };
61 |
62 | export default SocialMedia;
63 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/404/NotFound.component.jsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { Heading, Text, Button, Flex } from "@chakra-ui/core";
3 | import { Link } from "react-router-dom";
4 | import SEO from "./../../common/SEO/SEO.component";
5 |
6 | const NotFound = () => {
7 | return (
8 |
9 |
10 |
16 | 404!
17 |
18 | Oops ! Requested resource was not found on this website.
19 |
20 |
21 |
22 |
23 |
24 |
25 | );
26 | };
27 |
28 | export default NotFound;
29 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/404/NotFound.styles.scss:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/src/components/pages/404/NotFound.styles.scss
--------------------------------------------------------------------------------
/frontend/src/components/pages/AdminPage/AdminPage.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useContext, useState } from "react";
2 | import axios from "axios";
3 | import { useEffect } from "react";
4 | import "./AdminPage.styles.scss";
5 | import shortid from "shortid";
6 | import userContext from "../../../context/userContext";
7 | import { useToast } from "@chakra-ui/core";
8 | import SEO from "./../../common/SEO/SEO.component";
9 |
10 | const AdminPage = () => {
11 | const [blogs, setBlogs] = useState();
12 | const { userData } = useContext(userContext);
13 | const toast = useToast();
14 | useEffect(() => {
15 | axios
16 | .get(process.env.REACT_APP_BACKEND + "/blogs/unapproved")
17 | .then((res) => {
18 | setBlogs(res.data);
19 | })
20 | .catch((err) => {
21 | toast({
22 | title: "Error!",
23 | description: "Error while approving the blog",
24 | });
25 | });
26 | }, []);
27 |
28 | const handleApproval = (isApproved, blogObj) => {
29 | const reqURL = `/blogs/${blogObj._id}`;
30 |
31 | const options = {
32 | headers: {
33 | "x-auth-token": userData.token,
34 | },
35 | };
36 | axios
37 | .patch(process.env.REACT_APP_BACKEND + reqURL, { isApproved }, options)
38 | .then((res) => {
39 | toast({
40 | title: "Post Approved!",
41 | });
42 | })
43 | .catch((err) =>
44 | toast({
45 | title: "Some error occured while approving.",
46 | })
47 | );
48 | };
49 | return (
50 |
51 | {/* {JSON.stringify(blogs)} */}
52 |
53 | Admin Page
54 |
55 |
56 |
57 |
58 | BlogID |
59 | Title |
60 | Content |
61 | Author |
62 | Avatar |
63 | Banner URL |
64 | Approved ? |
65 |
66 |
67 |
68 | {blogs &&
69 | blogs.map((b, i) => (
70 |
71 | {b._id} |
72 | {b.title} |
73 | {b.content} |
74 | {b.author} |
75 | {b.avatar} |
76 | {b.bannerURL} |
77 | {JSON.stringify(b.isApproved)} |
78 |
79 |
82 |
85 | |
86 |
87 | ))}
88 |
89 |
90 |
91 |
92 | );
93 | };
94 |
95 | export default AdminPage;
96 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/AdminPage/AdminPage.styles.scss:
--------------------------------------------------------------------------------
1 | table {
2 | border: 2px solid black;
3 | }
4 | .table-container{
5 | overflow-x: auto;
6 | width: 100%;
7 | }
8 | tr td {
9 | white-space:nowrap;
10 | padding: 0 3rem;
11 | max-width: 300px;
12 | overflow: auto;
13 | border: 1px solid black;
14 | }
15 | thead {
16 | border: 2px solid black;
17 | }
18 | th{
19 | border: 1px solid black;
20 | }
21 | .bannerURL{
22 | max-width: 200px;
23 | }
24 | td button {
25 | margin: 0 1rem;
26 | border: 1px solid red;
27 | }
--------------------------------------------------------------------------------
/frontend/src/components/pages/Auth/Login.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useContext } from "react";
2 | import { submitService } from "../../../services/authService";
3 | import {
4 | FormControl,
5 | FormLabel,
6 | FormErrorMessage,
7 | FormHelperText,
8 | Flex,
9 | Box,
10 | Input,
11 | Heading,
12 | useToast,
13 | Text,
14 | Button,
15 | } from "@chakra-ui/core";
16 |
17 | import theme from "../../../themes/theme";
18 | import { useState } from "react";
19 | import { useHistory, Link } from "react-router-dom";
20 | import axios from "axios";
21 | import userContext from "../../../context/userContext";
22 | import SEO from "./../../common/SEO/SEO.component";
23 |
24 | const Login = (props) => {
25 | const [email, setEmail] = useState("");
26 | const [password, setPassword] = useState("");
27 | const { setUserData } = useContext(userContext);
28 | const toast = useToast();
29 | const history = useHistory();
30 | const handleEmail = (e) => {
31 | const userEmail = e.target.value;
32 | setEmail(userEmail);
33 | };
34 | const handlePassword = (e) => {
35 | const userPassword = e.target.value;
36 | setPassword(userPassword);
37 | };
38 |
39 | const handleSubmit = async (e) => {
40 | e.preventDefault();
41 | const loginCreds = { email, password };
42 | if (!email || !password)
43 | return toast({ title: "Please fill all the fields." });
44 | axios
45 | .post(process.env.REACT_APP_BACKEND + "/user/login", loginCreds)
46 | .then((res) => {
47 | setUserData({
48 | token: res.data.token,
49 | user: res.data.user,
50 | isAdmin: res.data.user.isAdmin,
51 | });
52 | localStorage.setItem("x-auth-token", res.data.token);
53 | // localStorage.setItem("auth-token", res.data.token);
54 |
55 | toast({
56 | title: "Logged in !",
57 | status: "success",
58 | });
59 | if (res.data.user.isAdmin) {
60 | history.push("/user/adminPage");
61 | } else {
62 | history.push("/blogs");
63 | }
64 | })
65 | .catch((err) => {
66 | toast({
67 | title: "Error!",
68 | description: err.response.data.msg,
69 | isClosable: true,
70 | status: "warning",
71 | });
72 | });
73 | };
74 |
75 | return (
76 |
77 |
78 |
79 |
87 |
88 |
89 | Login.
90 |
91 |
92 |
130 |
131 | Don't have an account ?
132 |
133 | Register Here.
134 |
135 |
136 |
137 |
138 |
139 | );
140 | };
141 |
142 | export default Login;
143 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/Auth/Login.styles.scss:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/src/components/pages/Auth/Login.styles.scss
--------------------------------------------------------------------------------
/frontend/src/components/pages/Auth/Register.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useState, useContext } from "react";
2 | import userContext from "../../../context/userContext";
3 | import { Link, useHistory } from "react-router-dom";
4 | import {
5 | FormControl,
6 | FormLabel,
7 | Flex,
8 | Box,
9 | Text,
10 | Input,
11 | useToast,
12 | Heading,
13 | Button,
14 | } from "@chakra-ui/core";
15 | import theme from "../../../themes/theme";
16 | import axios from "axios";
17 | import SEO from "./../../common/SEO/SEO.component";
18 |
19 | const Register = () => {
20 | const [email, setEmail] = useState("");
21 | const [password, setPassword] = useState("");
22 | const [username, setUsername] = useState("");
23 | const [avatar, setAvatar] = useState("");
24 | const { setUserData } = useContext(userContext);
25 | const toast = useToast();
26 |
27 | const history = useHistory();
28 | const handleEmail = (e) => {
29 | setEmail(e.target.value);
30 | };
31 | const handlePassword = (e) => {
32 | setPassword(e.target.value);
33 | };
34 | const handleUsername = (e) => {
35 | setUsername(e.target.value);
36 | };
37 | const handleAvatar = (e) => {
38 | setAvatar(e.target.value);
39 | };
40 | const handleSubmit = async (e) => {
41 | e.preventDefault();
42 | const registerCreds = { email, password, username, avatar };
43 | const loginCreds = { email, password };
44 | await axios
45 | .post(process.env.REACT_APP_BACKEND + "/user/register", registerCreds)
46 | .then((res) => {
47 | if (res.status === 200) {
48 | toast({ title: "Registerd!", isClosable: true });
49 | history.push("/");
50 | } else {
51 | throw new Error();
52 | }
53 | })
54 | .catch((err) => {
55 | toast({
56 | title: "Error!",
57 | description: err.msg,
58 | status: "error",
59 | });
60 | history.push("/register");
61 | });
62 | axios
63 | .post(process.env.REACT_APP_BACKEND + "/user/login", loginCreds)
64 | .then((res) => {
65 | setUserData({
66 | token: res.data.token,
67 | user: res.data.user,
68 | });
69 | if (res.data.error) {
70 | history.push("/user/register");
71 | } else {
72 | history.push("/");
73 | }
74 | })
75 | .catch((err) => {
76 | toast({
77 | title: "Error!",
78 | description: err.response.data.error,
79 | });
80 | history.push("/user/register");
81 | });
82 | };
83 |
84 | return (
85 |
86 |
87 |
88 |
96 |
97 |
98 | Register.
99 |
100 |
101 |
160 |
161 | Already have an account ?
162 |
163 | Login Here.
164 |
165 |
166 |
167 |
168 |
169 | );
170 | };
171 |
172 | export default Register;
173 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/Auth/Register.styles.scss:
--------------------------------------------------------------------------------
1 | s
--------------------------------------------------------------------------------
/frontend/src/components/pages/Blog/Blog.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useState, useContext, useEffect } from "react";
2 | import {
3 | Heading,
4 | Box,
5 | Flex,
6 | Grid,
7 | Image,
8 | Text,
9 | Button,
10 | Spinner,
11 | useToast,
12 | } from "@chakra-ui/core";
13 | import ReactMarkdown from "react-markdown";
14 | import theme from "../../../themes/theme";
15 | import Skeleton from "react-loading-skeleton";
16 | import ChakraUIRenderer, { defaults } from "./BlogRender";
17 | import userContext from "../../../context/userContext";
18 | import "./Blog.styles.scss";
19 | import axios from "axios";
20 | import { useHistory } from "react-router-dom";
21 | import AuthorMenu from "./../../common/AuthorMenu/AuthorMenu.component";
22 | import { HiOutlineUserCircle } from "react-icons/hi";
23 | import SEO from "./../../common/SEO/SEO.component";
24 | import {
25 | TwitterShareButton,
26 | TwitterIcon,
27 | FacebookShareButton,
28 | FacebookIcon,
29 | } from "react-share";
30 | import { AiFillTwitterCircle } from "react-icons/ai";
31 | import { FaFacebook, FaReddit, FaWhatsapp } from "react-icons/fa";
32 | import SocialMedia from "../../common/SocialMedia/SocialMedia.component";
33 |
34 | const Blog = (props) => {
35 | const [blog, setBlog] = useState("");
36 | const [image, setImageURL] = useState("");
37 | const { userData } = useContext(userContext);
38 | const toast = useToast();
39 | const history = useHistory();
40 | const reqURL = `/blogs/${props.match.params.username}/${props.match.params.blogID}`;
41 |
42 | useEffect(() => {
43 | axios
44 | .get(process.env.REACT_APP_BACKEND + reqURL)
45 | .then((res) => {
46 | setBlog(res.data);
47 | })
48 | .catch((err) => {
49 | toast({
50 | title: "Some error occured.",
51 | description: err.response.data.msg,
52 | isClosable: true,
53 | });
54 | history.push("/blogs");
55 | });
56 | }, []);
57 |
58 | const handleImageLoad = () => {
59 | setImageURL("loaded");
60 | };
61 |
62 | return (
63 |
64 |
65 |
66 |
67 |
68 |
69 |
70 |
71 |
79 |
83 | {blog.bannerURL !== "" ? (
84 |
93 | ) : (
94 | ""
95 | )}
96 |
97 |
98 |
102 |
111 | {blog.title || }
112 |
113 |
118 |
119 |
120 | {blog.avatar ? (
121 |
129 | ) : (
130 |
131 | )}
132 |
133 |
134 |
139 | {blog.author}
140 |
141 |
142 | {blog.date}
143 |
144 |
145 |
146 |
147 |
148 |
149 | {(value) => {
150 | if (value.userData.user) {
151 | return (
152 |
156 | );
157 | } else {
158 | return "";
159 | }
160 | }}
161 |
162 |
163 |
164 |
165 |
166 |
171 |
172 |
173 |
174 |
179 |
180 |
181 |
182 | );
183 | };
184 |
185 | export default Blog;
186 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/Blog/Blog.styles.scss:
--------------------------------------------------------------------------------
1 | // body {
2 | // background-color: #eef0f1;
3 | // }
4 | // .md-content {
5 | // p {
6 | // font-size: unset;
7 | // }
8 | // h1{
9 | // font-size: 2.25rem;
10 | // line-height: 1.25;
11 | // }
12 | // // h1,
13 | // // h2,
14 | // // h3,
15 | // // h4,
16 | // // h5,
17 | // // h6 {
18 | // // // font-size: unset;
19 | // // }
20 | // }
21 | .user-icon {
22 | width: 100%;
23 | height: 100%;
24 | }
25 | a {
26 | text-decoration: underline;
27 | }
--------------------------------------------------------------------------------
/frontend/src/components/pages/Blog/BlogRender.jsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import {
3 | Text,
4 | Code,
5 | Divider,
6 | Link,
7 | List,
8 | Checkbox,
9 | ListItem,
10 | Heading,
11 | Image,
12 | } from "@chakra-ui/core";
13 |
14 | function getCoreProps(props) {
15 | return props["data-sourcepos"]
16 | ? { "data-sourcepos": props["data-sourcepos"] }
17 | : {};
18 | }
19 |
20 | export const defaults = {
21 | paragraph: (props) => {
22 | const { children } = props;
23 | return (
24 |
25 | {children}
26 |
27 | );
28 | },
29 | emphasis: (props) => {
30 | const { children } = props;
31 | return (
32 |
33 | {children}
34 |
35 | );
36 | },
37 | blockquote: (props) => {
38 | const { children } = props;
39 | return (
40 |
41 | {children}
42 |
43 | );
44 | },
45 | code: (props) => {
46 | const { language, value } = props;
47 | const className = language && `language-${language}`;
48 | return (
49 |
50 |
51 | {value}
52 |
53 |
54 | );
55 | },
56 | delete: (props) => {
57 | const { children } = props;
58 | return {children};
59 | },
60 | thematicBreak: Divider,
61 | link: (props) => {
62 | const { children } = props;
63 | return (
64 |
80 | {children}
81 |
82 | );
83 | },
84 | img: Image,
85 | linkReference: Link,
86 | imageReference: Image,
87 | text: (props) => {
88 | const { children } = props;
89 | return {children};
90 | },
91 | list: (props) => {
92 | const { start, ordered, children, depth } = props;
93 | const attrs = getCoreProps(props);
94 | if (start !== null && start !== 1 && start !== undefined) {
95 | attrs.start = start.toString();
96 | }
97 | let styleType = "disc";
98 | if (ordered) styleType = "decimal";
99 | if (depth === 1) styleType = "circle";
100 | return (
101 |
108 | {children}
109 |
110 | );
111 | },
112 | listItem: (props) => {
113 | const { children, checked } = props;
114 | let checkbox = null;
115 | if (checked !== null && checked !== undefined) {
116 | checkbox = (
117 |
118 | {children}
119 |
120 | );
121 | }
122 | return (
123 |
127 | {checkbox || children}
128 |
129 | );
130 | },
131 | definition: () => null,
132 | heading: (props) => {
133 | const { level, children } = props;
134 | const sizes = ["xl", "lg", "md", "sm", "xs"];
135 |
136 | return (
137 |
145 | {children}
146 |
147 | );
148 | },
149 | inlineCode: (props) => {
150 | const { children } = props;
151 | return {children}
;
152 | },
153 | };
154 |
155 | function ChakraUIRenderer(theme = defaults) {
156 | return {
157 | paragraph: theme.paragraph,
158 | emphasis: theme.emphasis,
159 | blockquote: theme.blockquote,
160 | code: theme.code,
161 | delete: theme.delete,
162 | thematicBreak: theme.thematicBreak,
163 | link: theme.link,
164 | img: theme.img,
165 | linkReference: theme.linkReference,
166 | imageReference: theme.imageReference,
167 | text: theme.text,
168 | list: theme.list,
169 | listItem: theme.listItem,
170 | definition: theme.definition,
171 | heading: theme.heading,
172 | inlineCode: theme.inlineCode,
173 | };
174 | }
175 |
176 | export default ChakraUIRenderer;
177 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/BlogCard/BlogCard.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useEffect, useState, useContext } from "react";
2 | import { Heading, Box, Text, Flex, Grid, Image } from "@chakra-ui/core";
3 | import theme from "../../../themes/theme";
4 | import { Link, Redirect } from "react-router-dom";
5 | import { HiOutlineUserCircle } from "react-icons/hi";
6 | import "./BlogCard.styles.scss";
7 | import Skeleton from "react-loading-skeleton";
8 | import userContext from "../../../context/userContext";
9 |
10 | const BlogCard = (props) => {
11 | const { userData } = useContext(userContext);
12 | const [image, setImageURL] = useState("");
13 | const [imgSrc, setImgSrc] = useState("");
14 |
15 | const fallbackImageURL = "http://unsplash.it/600/600";
16 | let imageExist = true;
17 |
18 | const handleImageLoad = () => {
19 | setImageURL("loaded");
20 | };
21 |
22 | return (
23 |
24 |
34 |
35 |
36 |
37 | {!image && props.data.bannerURL && (
38 |
39 | )}
40 |
48 |
49 |
50 |
51 |
58 | {props.data.title}
59 |
60 |
61 |
62 |
63 | {props.data.avatar ? (
64 |
70 |
71 |
72 | ) : (
73 |
74 | )}
75 |
76 |
77 |
82 | {props.data.author} • {props.data.date}
83 |
84 |
85 |
86 |
87 |
88 |
89 |
90 |
91 |
92 | );
93 | };
94 |
95 | export default BlogCard;
96 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/BlogCard/BlogCard.styles.scss:
--------------------------------------------------------------------------------
1 | .blog-banner:hover{
2 | transform: scale(1.1);
3 | }
4 | .blog-banner{
5 | transition: transform 0.3s ease-in-out;
6 | }
7 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/Blogs/Blogs.component.jsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import useAxios from "axios-hooks";
3 | import { Heading, Flex, Grid, Box, Spinner } from "@chakra-ui/core";
4 | import BlogCard from "./../BlogCard/BlogCard.component";
5 | import theme from "../../../themes/theme";
6 | import "./Blogs.styles.scss";
7 | import SEO from "./../../common/SEO/SEO.component";
8 |
9 | const Blogs = () => {
10 | // const [blogsList, setBlogsList] = useState();
11 | const [{ data, loading, error }, refetch] = useAxios(
12 | process.env.REACT_APP_BACKEND + "/blogs"
13 | );
14 | if (loading)
15 | return (
16 |
17 |
18 |
19 | );
20 | if (error) return Error!
;
21 |
22 | return (
23 |
24 |
25 |
33 |
39 |
40 |
46 | Trending Blogs
47 |
48 |
49 |
50 | {data.blogs ? (
51 |
52 | {data.blogs &&
53 | data.blogs
54 | .map((b) => )
55 | .reverse()}
56 |
57 | ) : (
58 | No Blogs found.
59 | )}
60 |
61 |
62 |
63 |
69 |
70 |
71 | );
72 | };
73 |
74 | export default Blogs;
75 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/Blogs/Blogs.styles.scss:
--------------------------------------------------------------------------------
1 | body{
2 | background-color:#fafafa;
3 | }
--------------------------------------------------------------------------------
/frontend/src/components/pages/EditPage/EditPage.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useState, useEffect, useContext } from "react";
2 | import MDEditor from "@uiw/react-md-editor";
3 | import { Flex, Box, Input, Button, Heading, useToast } from "@chakra-ui/core";
4 | import theme from "../../../themes/theme";
5 | import userContext from "../../../context/userContext";
6 | import { useHistory } from "react-router-dom";
7 | import axios from "axios";
8 |
9 | const EditPage = (props) => {
10 | const { blogInfo } = props.location.state.data;
11 | const { userData } = useContext(userContext);
12 | const [value, setValue] = useState();
13 | const [bannerURL, setBannerURL] = useState("");
14 | const [blogTitle, setBlogTitle] = useState("");
15 | const toast = useToast();
16 | const history = useHistory();
17 |
18 | useEffect(() => {
19 | setBannerURL(blogInfo.bannerURL);
20 | setBlogTitle(blogInfo.title);
21 | setValue(blogInfo.content);
22 | }, []);
23 |
24 | const handleSubmit = (e) => {
25 | e.preventDefault();
26 | const { user, token } = userData;
27 | if (!user) {
28 | toast({
29 | title: "Error!",
30 | status: "error",
31 | description: "You need to be logged in to Edit that blog.",
32 | isClosable: true,
33 | });
34 | history.push("/user/login");
35 | } else {
36 | const editedBlogData = {
37 | title: blogTitle,
38 | bannerURL: bannerURL,
39 | content: value,
40 | };
41 | const reqURL = `/blogs/${blogInfo.author}/${blogInfo._id}`;
42 | if (userData.user.id) {
43 | axios.interceptors.request.use((req) => {
44 | req.headers["x-auth-token"] = token;
45 | return req;
46 | });
47 | axios
48 | .patch(process.env.REACT_APP_BACKEND + reqURL, editedBlogData)
49 | .then((res) => {
50 | toast({
51 | title: "Successfully Edited!",
52 | description: "Your blog was updated.",
53 | status: "success",
54 | isClosable: true,
55 | });
56 | console.log(res);
57 | history.push("/blogs");
58 | })
59 | .catch((err) => {
60 | toast({
61 | title: "Error!",
62 | description: "Blog edit failed.",
63 | status: "error",
64 | isClosable: true,
65 | });
66 | history.push("/blogs");
67 | });
68 | } else {
69 | toast({
70 | title: "Some error occured.",
71 | description: "Please log in and try again.",
72 | status: "error",
73 | isClosable: true,
74 | });
75 | history.push("/user/login");
76 | }
77 | }
78 | };
79 |
80 | const handleBannerURL = (e) => {
81 | setBannerURL(e.target.value);
82 | };
83 |
84 | const handleBlogTitle = (e) => {
85 | setBlogTitle(e.target.value);
86 | };
87 |
88 | return (
89 |
90 |
99 |
132 |
133 |
134 | );
135 | };
136 |
137 | export default EditPage;
138 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/Home/Home.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useContext } from "react";
2 | import { Button, Flex } from "@chakra-ui/core";
3 | import { Box } from "@chakra-ui/core";
4 | import { Heading, Text } from "@chakra-ui/core";
5 | import "./Home.styles.scss";
6 | import theme from "../../../themes/theme";
7 | import { Link } from "react-router-dom";
8 | import home from "../../../assets/Images/bg.jpg";
9 | import userContext from "../../../context/userContext";
10 | import SEO from "./../../common/SEO/SEO.component";
11 | import { AiFillCalculator } from "react-icons/ai";
12 |
13 | const Home = (props) => {
14 | const { userData } = useContext(userContext);
15 |
16 | return (
17 |
18 |
22 |
31 |
32 |
39 |
49 | Express your story.
50 |
51 |
52 | {userData.user ? (
53 | ""
54 | ) : (
55 |
56 |
71 |
72 | )}
73 |
74 |
83 |
84 |
85 |
94 |
99 | Proudly made in India
100 |
101 |
102 |
108 | About this project.
109 |
110 |
116 | Source Code.
117 |
118 |
119 |
120 |
121 |
122 |
123 |
124 | );
125 | };
126 |
127 | export default Home;
128 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/Home/Home.styles.scss:
--------------------------------------------------------------------------------
1 | // *{
2 | // // border: 1px solid black !important;
3 | // }
4 | .landing-text {
5 | background-clip: text;
6 | -webkit-text-fill-color: transparent;
7 | padding-left: var(--padding);
8 | padding-right: var(--padding);
9 | background-image: linear-gradient(90deg, #7928ca, #ff0080);
10 | }
11 | .banner {
12 | background-size: 100%;
13 | background-repeat: no-repeat;
14 | // height: calc(100vh - 4.2rem);
15 | background-image: url('../../../assets/Images/bgp3.png');
16 | }
17 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/NewBlog/NewBlog.component.jsx:
--------------------------------------------------------------------------------
1 | import React, { useContext, useEffect } from "react";
2 | import MDEditor from "@uiw/react-md-editor";
3 | import { useState } from "react";
4 | import {
5 | Box,
6 | Flex,
7 | Text,
8 | Heading,
9 | Button,
10 | Input,
11 | useToast,
12 | } from "@chakra-ui/core";
13 | import theme from "../../../themes/theme";
14 | import userContext from "../../../context/userContext";
15 | import axios from "axios";
16 | import { useHistory, Redirect } from "react-router-dom";
17 | import SEO from "./../../common/SEO/SEO.component";
18 |
19 | const NewBlog = () => {
20 | const [value, setValue] = useState();
21 | const [bannerURL, setBannerURL] = useState("");
22 | const [blogTitle, setBlogTitle] = useState("");
23 | const { userData } = useContext(userContext);
24 | const history = useHistory();
25 | const toast = useToast();
26 | let isSubmitting = false;
27 | const handleSubmit = (e) => {
28 | e.preventDefault();
29 | const { user, token } = userData;
30 | if (!user) {
31 | toast({
32 | title: "Error!",
33 | status: "error",
34 | description: "You need to be logged in to publish a blog.",
35 | isClosable: true,
36 | });
37 | history.push("/user/login");
38 | } else {
39 | const blogData = {
40 | title: blogTitle,
41 | content: value,
42 | author: userData.user.username,
43 | date: new Date().toDateString(),
44 | bannerURL: bannerURL,
45 | authorID: userData.user.id,
46 | avatar: userData.user.avatar,
47 | };
48 | const options = {
49 | headers: {
50 | "x-auth-token": userData.token,
51 | },
52 | };
53 | if (userData.user.id) {
54 | axios
55 | .post(process.env.REACT_APP_BACKEND + "/blogs/new", blogData, options)
56 | .then(() => {
57 | history.push("/blogs");
58 | toast({
59 | title: "Blog Published!",
60 | description: "Please wait for approval :)",
61 | isClosable: true,
62 | status: "success",
63 | });
64 | isSubmitting = true;
65 | })
66 | .catch((err) => {
67 | toast({
68 | title: "Error!",
69 | description:
70 | "Some error occured while publishing your blog. Please try again.",
71 | isClosable: true,
72 | status: "error",
73 | });
74 | history.push("/blogs");
75 | });
76 | } else {
77 | toast({
78 | title: "Error!",
79 | status: "error",
80 | description: "You need to be logged in to publish a blog.",
81 | isClosable: true,
82 | });
83 | history.push("/user/login");
84 | }
85 | }
86 | };
87 | const handleBannerURL = (e) => {
88 | setBannerURL(e.target.value);
89 | };
90 |
91 | const handleBlogTitle = (e) => {
92 | setBlogTitle(e.target.value);
93 | };
94 |
95 | return (
96 |
97 |
98 |
107 |
156 |
157 |
158 | );
159 | };
160 |
161 | export default NewBlog;
162 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/NewBlog/NewBlog.styles.scss:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/src/components/pages/NewBlog/NewBlog.styles.scss
--------------------------------------------------------------------------------
/frontend/src/components/pages/User/User.component.jsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 |
3 | const User = () => {
4 | return Under construction!
;
5 | };
6 |
7 | export default User;
8 |
--------------------------------------------------------------------------------
/frontend/src/components/pages/User/User.styles.scss:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/OmkarK45/blog-app/b8a02406845455db140cdaaec585fda5cdc149ca/frontend/src/components/pages/User/User.styles.scss
--------------------------------------------------------------------------------
/frontend/src/context/userContext.js:
--------------------------------------------------------------------------------
1 | import { createContext } from "react";
2 |
3 | export default createContext(null);
4 |
--------------------------------------------------------------------------------
/frontend/src/index.css:
--------------------------------------------------------------------------------
1 | /* @import url("https://fonts.googleapis.com/css2?family=Kumbh+Sans:wght@400;700&display=swap"); */
2 | @import url("https://fonts.googleapis.com/css2?family=Inter:wght@300;400;700&display=swap");
3 | * {
4 | margin: 0;
5 | padding: 0;
6 | box-sizing: border-box;
7 | border: 1px solid black;
8 | }
9 | :root {
10 | --headingFont: "Inter", "Kumbh Sans", sans-serif;
11 | --bodyFont: "Inter", "Kumbh Sans", "Segoe UI", sans-serif;
12 | --mono: monospace;
13 | }
14 |
15 | body {
16 | margin: 0;
17 |
18 | font-family: "Inter", "Kumbh Sans", "Segoe UI", -apple-system,
19 | BlinkMacSystemFont, "Roboto", "Oxygen", "Ubuntu", "Cantarell", "Fira Sans",
20 | "Droid Sans", "Helvetica Neue", sans-serif;
21 | -webkit-font-smoothing: antialiased;
22 | -moz-osx-font-smoothing: grayscale;
23 | }
24 |
25 | code {
26 | font-family: source-code-pro, Menlo, Monaco, Consolas, "Courier New",
27 | monospace;
28 | }
29 | li {
30 | list-style: none;
31 | }
32 | a {
33 | text-decoration: none;
34 | }
35 | button {
36 | transition: transform 0.2s ease-in-out;
37 | }
38 | button:hover {
39 | transform: translateY(-2px);
40 | }
41 |
--------------------------------------------------------------------------------
/frontend/src/index.js:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import ReactDOM from "react-dom";
3 | import "./index.css";
4 | import App from "./App";
5 | import * as serviceWorker from "./serviceWorker";
6 | import { ThemeProvider, CSSReset, ColorModeProvider } from "@chakra-ui/core";
7 |
8 | ReactDOM.render(
9 |
10 |
11 |
12 | ,
13 | document.getElementById("root")
14 | );
15 |
16 | serviceWorker.unregister();
17 |
--------------------------------------------------------------------------------
/frontend/src/serviceWorker.js:
--------------------------------------------------------------------------------
1 | // This optional code is used to register a service worker.
2 | // register() is not called by default.
3 |
4 | // This lets the app load faster on subsequent visits in production, and gives
5 | // it offline capabilities. However, it also means that developers (and users)
6 | // will only see deployed updates on subsequent visits to a page, after all the
7 | // existing tabs open on the page have been closed, since previously cached
8 | // resources are updated in the background.
9 |
10 | // To learn more about the benefits of this model and instructions on how to
11 | // opt-in, read https://bit.ly/CRA-PWA
12 |
13 | const isLocalhost = Boolean(
14 | window.location.hostname === "localhost" ||
15 | // [::1] is the IPv6 localhost address.
16 | window.location.hostname === "[::1]" ||
17 | // 127.0.0.0/8 are considered localhost for IPv4.
18 | window.location.hostname.match(
19 | /^127(?:\.(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}$/
20 | )
21 | );
22 |
23 | export function register(config) {
24 | if (process.env.NODE_ENV === "production" && "serviceWorker" in navigator) {
25 | // The URL constructor is available in all browsers that support SW.
26 | const publicUrl = new URL(process.env.PUBLIC_URL, window.location.href);
27 | if (publicUrl.origin !== window.location.origin) {
28 | // Our service worker won't work if PUBLIC_URL is on a different origin
29 | // from what our page is served on. This might happen if a CDN is used to
30 | // serve assets; see https://github.com/facebook/create-react-app/issues/2374
31 | return;
32 | }
33 |
34 | window.addEventListener("load", () => {
35 | const swUrl = `${process.env.PUBLIC_URL}/service-worker.js`;
36 |
37 | if (isLocalhost) {
38 | // This is running on localhost. Let's check if a service worker still exists or not.
39 | checkValidServiceWorker(swUrl, config);
40 |
41 | // Add some additional logging to localhost, pointing developers to the
42 | // service worker/PWA documentation.
43 | navigator.serviceWorker.ready.then(() => {
44 | console.log(
45 | "This web app is being served cache-first by a service " +
46 | "worker. To learn more, visit https://bit.ly/CRA-PWA"
47 | );
48 | });
49 | } else {
50 | // Is not localhost. Just register service worker
51 | registerValidSW(swUrl, config);
52 | }
53 | });
54 | }
55 | }
56 |
57 | function registerValidSW(swUrl, config) {
58 | navigator.serviceWorker
59 | .register(swUrl)
60 | .then((registration) => {
61 | registration.onupdatefound = () => {
62 | const installingWorker = registration.installing;
63 | if (installingWorker == null) {
64 | return;
65 | }
66 | installingWorker.onstatechange = () => {
67 | if (installingWorker.state === "installed") {
68 | if (navigator.serviceWorker.controller) {
69 | // At this point, the updated precached content has been fetched,
70 | // but the previous service worker will still serve the older
71 | // content until all client tabs are closed.
72 | console.log(
73 | "New content is available and will be used when all " +
74 | "tabs for this page are closed. See https://bit.ly/CRA-PWA."
75 | );
76 |
77 | // Execute callback
78 | if (config && config.onUpdate) {
79 | config.onUpdate(registration);
80 | }
81 | } else {
82 | // At this point, everything has been precached.
83 | // It's the perfect time to display a
84 | // "Content is cached for offline use." message.
85 | console.log("Content is cached for offline use.");
86 |
87 | // Execute callback
88 | if (config && config.onSuccess) {
89 | config.onSuccess(registration);
90 | }
91 | }
92 | }
93 | };
94 | };
95 | })
96 | .catch((error) => {
97 | console.error("Error during service worker registration:", error);
98 | });
99 | }
100 |
101 | function checkValidServiceWorker(swUrl, config) {
102 | // Check if the service worker can be found. If it can't reload the page.
103 | fetch(swUrl, {
104 | headers: { "Service-Worker": "script" },
105 | })
106 | .then((response) => {
107 | // Ensure service worker exists, and that we really are getting a JS file.
108 | const contentType = response.headers.get("content-type");
109 | if (
110 | response.status === 404 ||
111 | (contentType != null && contentType.indexOf("javascript") === -1)
112 | ) {
113 | // No service worker found. Probably a different app. Reload the page.
114 | navigator.serviceWorker.ready.then((registration) => {
115 | registration.unregister().then(() => {
116 | window.location.reload();
117 | });
118 | });
119 | } else {
120 | // Service worker found. Proceed as normal.
121 | registerValidSW(swUrl, config);
122 | }
123 | })
124 | .catch(() => {
125 | console.log(
126 | "No internet connection found. App is running in offline mode."
127 | );
128 | });
129 | }
130 |
131 | export function unregister() {
132 | if ("serviceWorker" in navigator) {
133 | navigator.serviceWorker.ready
134 | .then((registration) => {
135 | registration.unregister();
136 | })
137 | .catch((error) => {
138 | console.error(error.message);
139 | });
140 | }
141 | }
142 |
--------------------------------------------------------------------------------
/frontend/src/services/authService.js:
--------------------------------------------------------------------------------
1 | import axios from 'axios'
2 |
3 | export async function submitService(creds){
4 | axios.post('/user/login',creds)
5 | .then(res=>{
6 | if(res.data.token){
7 | localStorage.setItem('x-auth-token', res.data.token)
8 | }else{
9 | localStorage.setItem('x-auth-token', null)
10 | }
11 | return res
12 | })
13 | .then(res=>{
14 | return res.data.user
15 | })
16 | .catch(error=>{
17 | console.log(error);
18 | })
19 | }
20 |
21 | export async function registerService(creds){
22 | axios.post('/user/register', creds)
23 | .then(res=>{
24 | console.log(res);
25 | })
26 | .catch(error=>{
27 | console.log(error);
28 | })
29 | }
--------------------------------------------------------------------------------
/frontend/src/setupTests.js:
--------------------------------------------------------------------------------
1 | // jest-dom adds custom jest matchers for asserting on DOM nodes.
2 | // allows you to do things like:
3 | // expect(element).toHaveTextContent(/react/i)
4 | // learn more: https://github.com/testing-library/jest-dom
5 | import "@testing-library/jest-dom/extend-expect";
6 |
--------------------------------------------------------------------------------
/frontend/src/themes/theme.js:
--------------------------------------------------------------------------------
1 | export default {
2 | breakpoints: ["30em", "48em", "62em", "80em"],
3 | fonts: {
4 | heading: '"Inter","Kumbh Sans", sans-serif',
5 | body: '"Inter","Kumbh Sans",system-ui, sans-serif',
6 | mono: "Menlo, monospace",
7 | },
8 | colors: {
9 | gradient : 'linear-gradient(90deg, #7928ca, #ff0080)',
10 | primary: "#27ae60",
11 | danger: "#c0392b",
12 | info: "#8e44ad",
13 | dark: "#230C33",
14 | purple: "#592E83",
15 | accent: "#3b49df",
16 | accentDark: "#323EBE",
17 | background: "#fafafa",
18 | white: "#fff",
19 | black: "#000",
20 | },
21 | fontWeights: {
22 | bold: "700",
23 | light: "300",
24 | regular: "400",
25 | },
26 | fontSizes: {
27 | xs: "0.75rem",
28 | sm: "0.875rem",
29 | md: "1rem",
30 | lg: "1.125rem",
31 | xl: "1.25rem",
32 | "2xl": "1.5rem",
33 | "3xl": "1.875rem",
34 | "4xl": "2.25rem",
35 | "5xl": "3rem",
36 | "6xl": "4rem",
37 | "7xl": "7rem",
38 | },
39 | };
40 |
--------------------------------------------------------------------------------