Split remaining controllers into separate files. Added iOS homescreen icon. Removed additional logging from weather module.
1
.gitignore
vendored
@ -1,4 +1,5 @@
|
|||||||
node_modules
|
node_modules
|
||||||
data
|
data
|
||||||
public
|
public
|
||||||
|
!client/public
|
||||||
build.sh
|
build.sh
|
@ -1,3 +1,6 @@
|
|||||||
|
### v1.7.4 (TBA)
|
||||||
|
- Added iOS "Add to homescreen" icon ([#131](https://github.com/pawelmalak/flame/issues/131))
|
||||||
|
|
||||||
### v1.7.3 (2021-10-28)
|
### v1.7.3 (2021-10-28)
|
||||||
- Fixed bug with custom CSS not updating
|
- Fixed bug with custom CSS not updating
|
||||||
|
|
||||||
|
BIN
client/public/icons/apple-touch-icon-114x114.png
Normal file
After Width: | Height: | Size: 9.4 KiB |
BIN
client/public/icons/apple-touch-icon-120x120.png
Normal file
After Width: | Height: | Size: 7.4 KiB |
BIN
client/public/icons/apple-touch-icon-144x144.png
Normal file
After Width: | Height: | Size: 7.1 KiB |
BIN
client/public/icons/apple-touch-icon-152x152.png
Normal file
After Width: | Height: | Size: 11 KiB |
BIN
client/public/icons/apple-touch-icon-180x180.png
Normal file
After Width: | Height: | Size: 20 KiB |
BIN
client/public/icons/apple-touch-icon-57x57.png
Normal file
After Width: | Height: | Size: 2.5 KiB |
BIN
client/public/icons/apple-touch-icon-72x72.png
Normal file
After Width: | Height: | Size: 3.2 KiB |
BIN
client/public/icons/apple-touch-icon-76x76.png
Normal file
After Width: | Height: | Size: 4.0 KiB |
BIN
client/public/icons/apple-touch-icon.png
Normal file
After Width: | Height: | Size: 2.5 KiB |
Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 15 KiB |
@ -2,7 +2,51 @@
|
|||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8" />
|
<meta charset="utf-8" />
|
||||||
<link rel="icon" href="%PUBLIC_URL%/favicon.ico" />
|
<link rel="icon" href="%PUBLIC_URL%/icons/favicon.ico" />
|
||||||
|
<link
|
||||||
|
rel="apple-touch-icon"
|
||||||
|
href="%PUBLIC_URL%/icons/apple-touch-icon.png"
|
||||||
|
/>
|
||||||
|
<link
|
||||||
|
rel="apple-touch-icon"
|
||||||
|
sizes="57x57"
|
||||||
|
href="%PUBLIC_URL%/icons/apple-touch-icon-57x57.png"
|
||||||
|
/>
|
||||||
|
<link
|
||||||
|
rel="apple-touch-icon"
|
||||||
|
sizes="72x72"
|
||||||
|
href="%PUBLIC_URL%/icons/apple-touch-icon-72x72.png"
|
||||||
|
/>
|
||||||
|
<link
|
||||||
|
rel="apple-touch-icon"
|
||||||
|
sizes="76x76"
|
||||||
|
href="%PUBLIC_URL%/icons/apple-touch-icon-76x76.png"
|
||||||
|
/>
|
||||||
|
<link
|
||||||
|
rel="apple-touch-icon"
|
||||||
|
sizes="114x114"
|
||||||
|
href="%PUBLIC_URL%/icons/apple-touch-icon-114x114.png"
|
||||||
|
/>
|
||||||
|
<link
|
||||||
|
rel="apple-touch-icon"
|
||||||
|
sizes="120x120"
|
||||||
|
href="%PUBLIC_URL%/icons/apple-touch-icon-120x120.png"
|
||||||
|
/>
|
||||||
|
<link
|
||||||
|
rel="apple-touch-icon"
|
||||||
|
sizes="144x144"
|
||||||
|
href="%PUBLIC_URL%/icons/apple-touch-icon-144x144.png"
|
||||||
|
/>
|
||||||
|
<link
|
||||||
|
rel="apple-touch-icon"
|
||||||
|
sizes="152x152"
|
||||||
|
href="%PUBLIC_URL%/icons/apple-touch-icon-152x152.png"
|
||||||
|
/>
|
||||||
|
<link
|
||||||
|
rel="apple-touch-icon"
|
||||||
|
sizes="180x180"
|
||||||
|
href="%PUBLIC_URL%/icons/apple-touch-icon-180x180.png"
|
||||||
|
/>
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
<meta
|
<meta
|
||||||
name="description"
|
name="description"
|
||||||
|
28
controllers/categories/createCategory.js
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const Category = require('../../models/Category');
|
||||||
|
const loadConfig = require('../../utils/loadConfig');
|
||||||
|
|
||||||
|
// @desc Create new category
|
||||||
|
// @route POST /api/categories
|
||||||
|
// @access Public
|
||||||
|
const createCategory = asyncWrapper(async (req, res, next) => {
|
||||||
|
const { pinCategoriesByDefault: pinCategories } = await loadConfig();
|
||||||
|
|
||||||
|
let category;
|
||||||
|
|
||||||
|
if (pinCategories) {
|
||||||
|
category = await Category.create({
|
||||||
|
...req.body,
|
||||||
|
isPinned: true,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
category = await Category.create(req.body);
|
||||||
|
}
|
||||||
|
|
||||||
|
res.status(201).json({
|
||||||
|
success: true,
|
||||||
|
data: category,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = createCategory;
|
45
controllers/categories/deleteCategory.js
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const ErrorResponse = require('../../utils/ErrorResponse');
|
||||||
|
const Category = require('../../models/Category');
|
||||||
|
const Bookmark = require('../../models/Bookmark');
|
||||||
|
|
||||||
|
// @desc Delete category
|
||||||
|
// @route DELETE /api/categories/:id
|
||||||
|
// @access Public
|
||||||
|
const deleteCategory = asyncWrapper(async (req, res, next) => {
|
||||||
|
const category = await Category.findOne({
|
||||||
|
where: { id: req.params.id },
|
||||||
|
include: [
|
||||||
|
{
|
||||||
|
model: Bookmark,
|
||||||
|
as: 'bookmarks',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!category) {
|
||||||
|
return next(
|
||||||
|
new ErrorResponse(
|
||||||
|
`Category with id of ${req.params.id} was not found`,
|
||||||
|
404
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
category.bookmarks.forEach(async (bookmark) => {
|
||||||
|
await Bookmark.destroy({
|
||||||
|
where: { id: bookmark.id },
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
await Category.destroy({
|
||||||
|
where: { id: req.params.id },
|
||||||
|
});
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
data: {},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = deleteCategory;
|
43
controllers/categories/getAllCategories.js
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const Category = require('../../models/Category');
|
||||||
|
const Bookmark = require('../../models/Bookmark');
|
||||||
|
const { Sequelize } = require('sequelize');
|
||||||
|
const loadConfig = require('../../utils/loadConfig');
|
||||||
|
|
||||||
|
// @desc Get all categories
|
||||||
|
// @route GET /api/categories
|
||||||
|
// @access Public
|
||||||
|
const getAllCategories = asyncWrapper(async (req, res, next) => {
|
||||||
|
const { useOrdering: orderType } = await loadConfig();
|
||||||
|
|
||||||
|
let categories;
|
||||||
|
|
||||||
|
if (orderType == 'name') {
|
||||||
|
categories = await Category.findAll({
|
||||||
|
include: [
|
||||||
|
{
|
||||||
|
model: Bookmark,
|
||||||
|
as: 'bookmarks',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
order: [[Sequelize.fn('lower', Sequelize.col('Category.name')), 'ASC']],
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
categories = await Category.findAll({
|
||||||
|
include: [
|
||||||
|
{
|
||||||
|
model: Bookmark,
|
||||||
|
as: 'bookmarks',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
order: [[orderType, 'ASC']],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
data: categories,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = getAllCategories;
|
35
controllers/categories/getSingleCategory.js
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const ErrorResponse = require('../../utils/ErrorResponse');
|
||||||
|
const Category = require('../../models/Category');
|
||||||
|
const Bookmark = require('../../models/Bookmark');
|
||||||
|
|
||||||
|
// @desc Get single category
|
||||||
|
// @route GET /api/categories/:id
|
||||||
|
// @access Public
|
||||||
|
const getSingleCategory = asyncWrapper(async (req, res, next) => {
|
||||||
|
const category = await Category.findOne({
|
||||||
|
where: { id: req.params.id },
|
||||||
|
include: [
|
||||||
|
{
|
||||||
|
model: Bookmark,
|
||||||
|
as: 'bookmarks',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!category) {
|
||||||
|
return next(
|
||||||
|
new ErrorResponse(
|
||||||
|
`Category with id of ${req.params.id} was not found`,
|
||||||
|
404
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
data: category,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = getSingleCategory;
|
8
controllers/categories/index.js
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
module.exports = {
|
||||||
|
createCategory: require('./createCategory'),
|
||||||
|
getAllCategories: require('./getAllCategories'),
|
||||||
|
getSingleCategory: require('./getSingleCategory'),
|
||||||
|
updateCategory: require('./updateCategory'),
|
||||||
|
deleteCategory: require('./deleteCategory'),
|
||||||
|
reorderCategories: require('./reorderCategories'),
|
||||||
|
};
|
22
controllers/categories/reorderCategories.js
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const Category = require('../../models/Category');
|
||||||
|
// @desc Reorder categories
|
||||||
|
// @route PUT /api/categories/0/reorder
|
||||||
|
// @access Public
|
||||||
|
const reorderCategories = asyncWrapper(async (req, res, next) => {
|
||||||
|
req.body.categories.forEach(async ({ id, orderId }) => {
|
||||||
|
await Category.update(
|
||||||
|
{ orderId },
|
||||||
|
{
|
||||||
|
where: { id },
|
||||||
|
}
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
data: {},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = reorderCategories;
|
30
controllers/categories/updateCategory.js
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const ErrorResponse = require('../../utils/ErrorResponse');
|
||||||
|
const Category = require('../../models/Category');
|
||||||
|
|
||||||
|
// @desc Update category
|
||||||
|
// @route PUT /api/categories/:id
|
||||||
|
// @access Public
|
||||||
|
const updateCategory = asyncWrapper(async (req, res, next) => {
|
||||||
|
let category = await Category.findOne({
|
||||||
|
where: { id: req.params.id },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!category) {
|
||||||
|
return next(
|
||||||
|
new ErrorResponse(
|
||||||
|
`Category with id of ${req.params.id} was not found`,
|
||||||
|
404
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
category = await category.update({ ...req.body });
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
data: category,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = updateCategory;
|
@ -1,178 +0,0 @@
|
|||||||
const asyncWrapper = require('../middleware/asyncWrapper');
|
|
||||||
const ErrorResponse = require('../utils/ErrorResponse');
|
|
||||||
const Category = require('../models/Category');
|
|
||||||
const Bookmark = require('../models/Bookmark');
|
|
||||||
const Config = require('../models/Config');
|
|
||||||
const { Sequelize } = require('sequelize');
|
|
||||||
const loadConfig = require('../utils/loadConfig');
|
|
||||||
|
|
||||||
// @desc Create new category
|
|
||||||
// @route POST /api/categories
|
|
||||||
// @access Public
|
|
||||||
exports.createCategory = asyncWrapper(async (req, res, next) => {
|
|
||||||
const { pinCategoriesByDefault: pinCategories } = await loadConfig();
|
|
||||||
|
|
||||||
let category;
|
|
||||||
|
|
||||||
if (pinCategories) {
|
|
||||||
category = await Category.create({
|
|
||||||
...req.body,
|
|
||||||
isPinned: true,
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
category = await Category.create(req.body);
|
|
||||||
}
|
|
||||||
|
|
||||||
res.status(201).json({
|
|
||||||
success: true,
|
|
||||||
data: category,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// @desc Get all categories
|
|
||||||
// @route GET /api/categories
|
|
||||||
// @access Public
|
|
||||||
exports.getCategories = asyncWrapper(async (req, res, next) => {
|
|
||||||
const { useOrdering: orderType } = await loadConfig();
|
|
||||||
|
|
||||||
let categories;
|
|
||||||
|
|
||||||
if (orderType == 'name') {
|
|
||||||
categories = await Category.findAll({
|
|
||||||
include: [
|
|
||||||
{
|
|
||||||
model: Bookmark,
|
|
||||||
as: 'bookmarks',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
order: [[Sequelize.fn('lower', Sequelize.col('Category.name')), 'ASC']],
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
categories = await Category.findAll({
|
|
||||||
include: [
|
|
||||||
{
|
|
||||||
model: Bookmark,
|
|
||||||
as: 'bookmarks',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
order: [[orderType, 'ASC']],
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
res.status(200).json({
|
|
||||||
success: true,
|
|
||||||
data: categories,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// @desc Get single category
|
|
||||||
// @route GET /api/categories/:id
|
|
||||||
// @access Public
|
|
||||||
exports.getCategory = asyncWrapper(async (req, res, next) => {
|
|
||||||
const category = await Category.findOne({
|
|
||||||
where: { id: req.params.id },
|
|
||||||
include: [
|
|
||||||
{
|
|
||||||
model: Bookmark,
|
|
||||||
as: 'bookmarks',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!category) {
|
|
||||||
return next(
|
|
||||||
new ErrorResponse(
|
|
||||||
`Category with id of ${req.params.id} was not found`,
|
|
||||||
404
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
res.status(200).json({
|
|
||||||
success: true,
|
|
||||||
data: category,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// @desc Update category
|
|
||||||
// @route PUT /api/categories/:id
|
|
||||||
// @access Public
|
|
||||||
exports.updateCategory = asyncWrapper(async (req, res, next) => {
|
|
||||||
let category = await Category.findOne({
|
|
||||||
where: { id: req.params.id },
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!category) {
|
|
||||||
return next(
|
|
||||||
new ErrorResponse(
|
|
||||||
`Category with id of ${req.params.id} was not found`,
|
|
||||||
404
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
category = await category.update({ ...req.body });
|
|
||||||
|
|
||||||
res.status(200).json({
|
|
||||||
success: true,
|
|
||||||
data: category,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// @desc Delete category
|
|
||||||
// @route DELETE /api/categories/:id
|
|
||||||
// @access Public
|
|
||||||
exports.deleteCategory = asyncWrapper(async (req, res, next) => {
|
|
||||||
const category = await Category.findOne({
|
|
||||||
where: { id: req.params.id },
|
|
||||||
include: [
|
|
||||||
{
|
|
||||||
model: Bookmark,
|
|
||||||
as: 'bookmarks',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!category) {
|
|
||||||
return next(
|
|
||||||
new ErrorResponse(
|
|
||||||
`Category with id of ${req.params.id} was not found`,
|
|
||||||
404
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
category.bookmarks.forEach(async (bookmark) => {
|
|
||||||
await Bookmark.destroy({
|
|
||||||
where: { id: bookmark.id },
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
await Category.destroy({
|
|
||||||
where: { id: req.params.id },
|
|
||||||
});
|
|
||||||
|
|
||||||
res.status(200).json({
|
|
||||||
success: true,
|
|
||||||
data: {},
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// @desc Reorder categories
|
|
||||||
// @route PUT /api/categories/0/reorder
|
|
||||||
// @access Public
|
|
||||||
exports.reorderCategories = asyncWrapper(async (req, res, next) => {
|
|
||||||
req.body.categories.forEach(async ({ id, orderId }) => {
|
|
||||||
await Category.update(
|
|
||||||
{ orderId },
|
|
||||||
{
|
|
||||||
where: { id },
|
|
||||||
}
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
res.status(200).json({
|
|
||||||
success: true,
|
|
||||||
data: {},
|
|
||||||
});
|
|
||||||
});
|
|
21
controllers/queries/addQuery.js
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const File = require('../../utils/File');
|
||||||
|
|
||||||
|
// @desc Add custom search query
|
||||||
|
// @route POST /api/queries
|
||||||
|
// @access Public
|
||||||
|
const addQuery = asyncWrapper(async (req, res, next) => {
|
||||||
|
const file = new File('data/customQueries.json');
|
||||||
|
let content = JSON.parse(file.read());
|
||||||
|
|
||||||
|
// Add new query
|
||||||
|
content.queries.push(req.body);
|
||||||
|
file.write(content, true);
|
||||||
|
|
||||||
|
res.status(201).json({
|
||||||
|
success: true,
|
||||||
|
data: req.body,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = addQuery;
|
22
controllers/queries/deleteQuery.js
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const File = require('../../utils/File');
|
||||||
|
|
||||||
|
// @desc Delete query
|
||||||
|
// @route DELETE /api/queries/:prefix
|
||||||
|
// @access Public
|
||||||
|
const deleteQuery = asyncWrapper(async (req, res, next) => {
|
||||||
|
const file = new File('data/customQueries.json');
|
||||||
|
let content = JSON.parse(file.read());
|
||||||
|
|
||||||
|
content.queries = content.queries.filter(
|
||||||
|
(q) => q.prefix != req.params.prefix
|
||||||
|
);
|
||||||
|
file.write(content, true);
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
data: content.queries,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = deleteQuery;
|
17
controllers/queries/getQueries.js
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const File = require('../../utils/File');
|
||||||
|
|
||||||
|
// @desc Get custom queries file
|
||||||
|
// @route GET /api/queries
|
||||||
|
// @access Public
|
||||||
|
const getQueries = asyncWrapper(async (req, res, next) => {
|
||||||
|
const file = new File('data/customQueries.json');
|
||||||
|
const content = JSON.parse(file.read());
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
data: content.queries,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = getQueries;
|
@ -1,81 +1,6 @@
|
|||||||
const asyncWrapper = require('../../middleware/asyncWrapper');
|
module.exports = {
|
||||||
const File = require('../../utils/File');
|
addQuery: require('./addQuery'),
|
||||||
const { join } = require('path');
|
getQueries: require('./getQueries'),
|
||||||
|
updateQuery: require('./updateQuery'),
|
||||||
const QUERIES_PATH = join(__dirname, '../../data/customQueries.json');
|
deleteQuery: require('./deleteQuery'),
|
||||||
|
};
|
||||||
// @desc Add custom search query
|
|
||||||
// @route POST /api/queries
|
|
||||||
// @access Public
|
|
||||||
exports.addQuery = asyncWrapper(async (req, res, next) => {
|
|
||||||
const file = new File(QUERIES_PATH);
|
|
||||||
let content = JSON.parse(file.read());
|
|
||||||
|
|
||||||
// Add new query
|
|
||||||
content.queries.push(req.body);
|
|
||||||
file.write(content, true);
|
|
||||||
|
|
||||||
res.status(201).json({
|
|
||||||
success: true,
|
|
||||||
data: req.body,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// @desc Get custom queries file
|
|
||||||
// @route GET /api/queries
|
|
||||||
// @access Public
|
|
||||||
exports.getQueries = asyncWrapper(async (req, res, next) => {
|
|
||||||
const file = new File(QUERIES_PATH);
|
|
||||||
const content = JSON.parse(file.read());
|
|
||||||
|
|
||||||
res.status(200).json({
|
|
||||||
success: true,
|
|
||||||
data: content.queries,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// @desc Update query
|
|
||||||
// @route PUT /api/queries/:prefix
|
|
||||||
// @access Public
|
|
||||||
exports.updateQuery = asyncWrapper(async (req, res, next) => {
|
|
||||||
const file = new File(QUERIES_PATH);
|
|
||||||
let content = JSON.parse(file.read());
|
|
||||||
|
|
||||||
let queryIdx = content.queries.findIndex(
|
|
||||||
(q) => q.prefix == req.params.prefix
|
|
||||||
);
|
|
||||||
|
|
||||||
// query found
|
|
||||||
if (queryIdx > -1) {
|
|
||||||
content.queries = [
|
|
||||||
...content.queries.slice(0, queryIdx),
|
|
||||||
req.body,
|
|
||||||
...content.queries.slice(queryIdx + 1),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
file.write(content, true);
|
|
||||||
|
|
||||||
res.status(200).json({
|
|
||||||
success: true,
|
|
||||||
data: content.queries,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// @desc Delete query
|
|
||||||
// @route DELETE /api/queries/:prefix
|
|
||||||
// @access Public
|
|
||||||
exports.deleteQuery = asyncWrapper(async (req, res, next) => {
|
|
||||||
const file = new File(QUERIES_PATH);
|
|
||||||
let content = JSON.parse(file.read());
|
|
||||||
|
|
||||||
content.queries = content.queries.filter(
|
|
||||||
(q) => q.prefix != req.params.prefix
|
|
||||||
);
|
|
||||||
file.write(content, true);
|
|
||||||
|
|
||||||
res.status(200).json({
|
|
||||||
success: true,
|
|
||||||
data: content.queries,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
32
controllers/queries/updateQuery.js
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const File = require('../../utils/File');
|
||||||
|
|
||||||
|
// @desc Update query
|
||||||
|
// @route PUT /api/queries/:prefix
|
||||||
|
// @access Public
|
||||||
|
const updateQuery = asyncWrapper(async (req, res, next) => {
|
||||||
|
const file = new File('data/customQueries.json');
|
||||||
|
let content = JSON.parse(file.read());
|
||||||
|
|
||||||
|
let queryIdx = content.queries.findIndex(
|
||||||
|
(q) => q.prefix == req.params.prefix
|
||||||
|
);
|
||||||
|
|
||||||
|
// query found
|
||||||
|
if (queryIdx > -1) {
|
||||||
|
content.queries = [
|
||||||
|
...content.queries.slice(0, queryIdx),
|
||||||
|
req.body,
|
||||||
|
...content.queries.slice(queryIdx + 1),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
file.write(content, true);
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
data: content.queries,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = updateQuery;
|
@ -1,31 +0,0 @@
|
|||||||
const asyncWrapper = require('../middleware/asyncWrapper');
|
|
||||||
const ErrorResponse = require('../utils/ErrorResponse');
|
|
||||||
const Weather = require('../models/Weather');
|
|
||||||
const getExternalWeather = require('../utils/getExternalWeather');
|
|
||||||
|
|
||||||
// @desc Get latest weather status
|
|
||||||
// @route GET /api/weather
|
|
||||||
// @access Public
|
|
||||||
exports.getWeather = asyncWrapper(async (req, res, next) => {
|
|
||||||
const weather = await Weather.findAll({
|
|
||||||
order: [['createdAt', 'DESC']],
|
|
||||||
limit: 1,
|
|
||||||
});
|
|
||||||
|
|
||||||
res.status(200).json({
|
|
||||||
success: true,
|
|
||||||
data: weather,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// @desc Update weather
|
|
||||||
// @route GET /api/weather/update
|
|
||||||
// @access Public
|
|
||||||
exports.updateWeather = asyncWrapper(async (req, res, next) => {
|
|
||||||
const weather = await getExternalWeather();
|
|
||||||
|
|
||||||
res.status(200).json({
|
|
||||||
success: true,
|
|
||||||
data: weather,
|
|
||||||
});
|
|
||||||
});
|
|
19
controllers/weather/getWather.js
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const Weather = require('../../models/Weather');
|
||||||
|
|
||||||
|
// @desc Get latest weather status
|
||||||
|
// @route GET /api/weather
|
||||||
|
// @access Public
|
||||||
|
const getWeather = asyncWrapper(async (req, res, next) => {
|
||||||
|
const weather = await Weather.findAll({
|
||||||
|
order: [['createdAt', 'DESC']],
|
||||||
|
limit: 1,
|
||||||
|
});
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
data: weather,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = getWeather;
|
4
controllers/weather/index.js
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
module.exports = {
|
||||||
|
getWeather: require('./getWather'),
|
||||||
|
updateWeather: require('./updateWeather'),
|
||||||
|
};
|
16
controllers/weather/updateWeather.js
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
const asyncWrapper = require('../../middleware/asyncWrapper');
|
||||||
|
const getExternalWeather = require('../../utils/getExternalWeather');
|
||||||
|
|
||||||
|
// @desc Update weather
|
||||||
|
// @route GET /api/weather/update
|
||||||
|
// @access Public
|
||||||
|
const updateWeather = asyncWrapper(async (req, res, next) => {
|
||||||
|
const weather = await getExternalWeather();
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
data: weather,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = updateWeather;
|
@ -3,26 +3,21 @@ const router = express.Router();
|
|||||||
|
|
||||||
const {
|
const {
|
||||||
createCategory,
|
createCategory,
|
||||||
getCategories,
|
getAllCategories,
|
||||||
getCategory,
|
getSingleCategory,
|
||||||
updateCategory,
|
updateCategory,
|
||||||
deleteCategory,
|
deleteCategory,
|
||||||
reorderCategories
|
reorderCategories,
|
||||||
} = require('../controllers/category');
|
} = require('../controllers/categories');
|
||||||
|
|
||||||
router
|
router.route('/').post(createCategory).get(getAllCategories);
|
||||||
.route('/')
|
|
||||||
.post(createCategory)
|
|
||||||
.get(getCategories);
|
|
||||||
|
|
||||||
router
|
router
|
||||||
.route('/:id')
|
.route('/:id')
|
||||||
.get(getCategory)
|
.get(getSingleCategory)
|
||||||
.put(updateCategory)
|
.put(updateCategory)
|
||||||
.delete(deleteCategory);
|
.delete(deleteCategory);
|
||||||
|
|
||||||
router
|
router.route('/0/reorder').put(reorderCategories);
|
||||||
.route('/0/reorder')
|
|
||||||
.put(reorderCategories);
|
|
||||||
|
|
||||||
module.exports = router;
|
module.exports = router;
|
@ -2,23 +2,28 @@ const { Op } = require('sequelize');
|
|||||||
const Weather = require('../models/Weather');
|
const Weather = require('../models/Weather');
|
||||||
const Logger = require('./Logger');
|
const Logger = require('./Logger');
|
||||||
const logger = new Logger();
|
const logger = new Logger();
|
||||||
|
const loadConfig = require('./loadConfig');
|
||||||
|
|
||||||
const clearWeatherData = async () => {
|
const clearWeatherData = async () => {
|
||||||
|
const { WEATHER_API_KEY: secret } = await loadConfig();
|
||||||
|
|
||||||
const weather = await Weather.findOne({
|
const weather = await Weather.findOne({
|
||||||
order: [[ 'createdAt', 'DESC' ]]
|
order: [['createdAt', 'DESC']],
|
||||||
});
|
});
|
||||||
|
|
||||||
if (weather) {
|
if (weather) {
|
||||||
await Weather.destroy({
|
await Weather.destroy({
|
||||||
where: {
|
where: {
|
||||||
id: {
|
id: {
|
||||||
[Op.lt]: weather.id
|
[Op.lt]: weather.id,
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.log('Old weather data was deleted');
|
if (secret) {
|
||||||
}
|
logger.log('Old weather data was deleted');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = clearWeatherData;
|
module.exports = clearWeatherData;
|