Compare commits
25 Commits
dogsitters
...
02e50bb2f9
| Author | SHA1 | Date | |
|---|---|---|---|
| 02e50bb2f9 | |||
| fadc62c8f0 | |||
| 4759f6f7ee | |||
| 14f2164a82 | |||
| 14ef1f9bad | |||
| dc99318ff0 | |||
| d2fc5f4d5c | |||
| 938bd48fff | |||
| 96f819dc91 | |||
| 25eee8adf5 | |||
| d2b2a29d3d | |||
| 1cf71261d1 | |||
| 88552eb04f | |||
| ab92c99321 | |||
| 02963de893 | |||
| 48550416d9 | |||
| 878c5ffd68 | |||
|
|
6e37fe93f7 | ||
|
|
72a2667549 | ||
|
|
39db7b4d26 | ||
| ff25c0ecb9 | |||
|
|
f1a93bffb5 | ||
| aa231d4f43 | |||
|
|
f254d57db4 | ||
| 106f835934 |
4
package-lock.json
generated
4
package-lock.json
generated
@@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "multi-stub",
|
"name": "multi-stub",
|
||||||
"version": "1.2.0",
|
"version": "1.2.1",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "multi-stub",
|
"name": "multi-stub",
|
||||||
"version": "1.2.0",
|
"version": "1.2.1",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"ai": "^4.1.13",
|
"ai": "^4.1.13",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "multi-stub",
|
"name": "multi-stub",
|
||||||
"version": "1.2.0",
|
"version": "1.2.1",
|
||||||
"description": "",
|
"description": "",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
|||||||
@@ -1,74 +0,0 @@
|
|||||||
const { Router } = require("express");
|
|
||||||
const hash = require("pbkdf2-password")();
|
|
||||||
const { promisify } = require("node:util");
|
|
||||||
const jwt = require('jsonwebtoken')
|
|
||||||
|
|
||||||
const { getAnswer } = require("../../utils/common");
|
|
||||||
|
|
||||||
const { AuthModel } = require("./model/todo/auth");
|
|
||||||
const { TOKEN_KEY } = require('./const')
|
|
||||||
const { UserModel } = require("./model/todo/user");
|
|
||||||
|
|
||||||
const { requiredValidate } = require('./utils')
|
|
||||||
|
|
||||||
const router = Router();
|
|
||||||
|
|
||||||
router.post(
|
|
||||||
"/signup",
|
|
||||||
requiredValidate("login", "password", "email"),
|
|
||||||
async (req, res, next) => {
|
|
||||||
const { login, password, email } = req.body
|
|
||||||
|
|
||||||
const user = await AuthModel.findOne({ login });
|
|
||||||
|
|
||||||
if (user) {
|
|
||||||
throw new Error("Пользователь с таким логином уже существует");
|
|
||||||
}
|
|
||||||
|
|
||||||
hash({ password }, async function (err, pass, salt, hash) {
|
|
||||||
if (err) return next(err);
|
|
||||||
|
|
||||||
const user = await UserModel.create({ login, email });
|
|
||||||
await AuthModel.create({ login, hash, salt, userId: user.id });
|
|
||||||
|
|
||||||
res.json(getAnswer(null, { ok: true }))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
function authenticate(login, pass, cb) {
|
|
||||||
AuthModel.findOne({ login }).populate('userId').exec().then((user) => {
|
|
||||||
if (!user) return cb(null, null)
|
|
||||||
|
|
||||||
hash({ password: pass, salt: user.salt }, function (err, pass, salt, hash) {
|
|
||||||
if (err) return cb(err)
|
|
||||||
if (hash === user.hash) return cb(null, user)
|
|
||||||
cb(null, null)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
const auth = promisify(authenticate)
|
|
||||||
|
|
||||||
router.post('/signin', requiredValidate('login', 'password'), async (req, res) => {
|
|
||||||
const { login, password } = req.body
|
|
||||||
|
|
||||||
const user = await auth(login, password)
|
|
||||||
|
|
||||||
if (!user) {
|
|
||||||
throw new Error("Неверный логин или пароль")
|
|
||||||
}
|
|
||||||
|
|
||||||
const accessToken = jwt.sign({
|
|
||||||
...JSON.parse(JSON.stringify(user.userId)),
|
|
||||||
}, TOKEN_KEY, {
|
|
||||||
expiresIn: '12h'
|
|
||||||
})
|
|
||||||
|
|
||||||
res.json(getAnswer(null, {
|
|
||||||
user: user.userId,
|
|
||||||
token: accessToken,
|
|
||||||
}))
|
|
||||||
})
|
|
||||||
|
|
||||||
module.exports = router
|
|
||||||
@@ -1,3 +1,2 @@
|
|||||||
exports.DSF_AUTH_PASSWD_MODEL_NAME = 'DSF_AUTH_PASSWD'
|
|
||||||
exports.DSF_AUTH_USER_MODEL_NAME = 'DSF_AUTH_USER'
|
exports.DSF_AUTH_USER_MODEL_NAME = 'DSF_AUTH_USER'
|
||||||
exports.DSF_INTERACTION_MODEL_NAME = 'DSF_INTERACTION'
|
exports.DSF_INTERACTION_MODEL_NAME = 'DSF_INTERACTION'
|
||||||
|
|||||||
@@ -8,20 +8,20 @@ router.post("/auth", (request, response) => {
|
|||||||
const { phoneNumber, password } = request.body;
|
const { phoneNumber, password } = request.body;
|
||||||
console.log(phoneNumber, password);
|
console.log(phoneNumber, password);
|
||||||
if (phoneNumber === "89999999999" || phoneNumber === "89559999999") {
|
if (phoneNumber === "89999999999" || phoneNumber === "89559999999") {
|
||||||
response.send(require("../json/auth/success.json"));
|
response.send(require("./json/auth/success.json"));
|
||||||
} else {
|
} else {
|
||||||
response.status(401).send(require("../json/auth/error.json"));
|
response.status(401).send(require("./json/auth/error.json"));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
router.post("/auth/2fa", (request, response) => {
|
router.post("/auth/2fa", (request, response) => {
|
||||||
const { phoneNumber, code } = request.body;
|
const { phoneNumber, code } = request.body;
|
||||||
if (code === "0000" && phoneNumber === "89999999999") {
|
if (code === "0000" && phoneNumber === "89999999999") {
|
||||||
response.send(require("../json/2fa/dogsitter.success.json"));
|
response.send(require("./json/2fa/dogsitter.success.json"));
|
||||||
} else if (code === "0000" && phoneNumber === "89559999999") {
|
} else if (code === "0000" && phoneNumber === "89559999999") {
|
||||||
response.send(require("../json/2fa/owner.success.json"));
|
response.send(require("./json/2fa/owner.success.json"));
|
||||||
} else {
|
} else {
|
||||||
response.status(401).send(require("../json/2fa/error.json"));
|
response.status(401).send(require("./json/2fa/error.json"));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -29,11 +29,11 @@ router.post("/register", (request, response) => {
|
|||||||
const { firstName, secondName, phoneNumber, password, role } = request.body;
|
const { firstName, secondName, phoneNumber, password, role } = request.body;
|
||||||
console.log(phoneNumber, password, role);
|
console.log(phoneNumber, password, role);
|
||||||
if (phoneNumber === "89999999999" || phoneNumber === "89559999999") {
|
if (phoneNumber === "89999999999" || phoneNumber === "89559999999") {
|
||||||
response.status(401).send(require("../json/register/error.json"));
|
response.status(401).send(require("./json/register/error.json"));
|
||||||
} else if (role === "dogsitter") {
|
} else if (role === "dogsitter") {
|
||||||
response.send(require("../json/register/dogsitter.success.json"));
|
response.send(require("./json/register/dogsitter.success.json"));
|
||||||
} else {
|
} else {
|
||||||
response.send(require("../json/register/owner.success.json"));
|
response.send(require("./json/register/owner.success.json"));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -58,12 +58,14 @@ router.get("/auth/session", (request, response) => {
|
|||||||
const decoded = jwt.verify(token, secretKey);
|
const decoded = jwt.verify(token, secretKey);
|
||||||
|
|
||||||
if (decoded.role === "dogsitter") {
|
if (decoded.role === "dogsitter") {
|
||||||
response.send(require("../json/role/dogsitter.success.json"));
|
response.send(require("./json/role/dogsitter.success.json"));
|
||||||
} else {
|
} else {
|
||||||
response.send(require("../json/role/owner.success.json"));
|
response.send(require("./json/role/owner.success.json"));
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.log("token e:", e);
|
console.log("token e:", e);
|
||||||
return response.status(403).json({ error: "Invalid token" });
|
return response.status(403).json({ error: "Invalid token" });
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
module.exports = router
|
||||||
@@ -1,39 +1,69 @@
|
|||||||
[
|
{
|
||||||
{
|
"data": [
|
||||||
"id": 1,
|
{
|
||||||
"phone_number": 89283244141,
|
"id": 1,
|
||||||
"first_name": "Вася",
|
"phone_number": "89999999999",
|
||||||
"second_name": "Пупкин",
|
"first_name": "Вася",
|
||||||
"role": "dogsitter",
|
"second_name": "Пупкин",
|
||||||
"location": "Россия, республика Татарстан, Казань, улица Пушкина, 12",
|
"role": "dogsitter",
|
||||||
"price": 1500,
|
"location": "Россия, республика Татарстан, Казань, Пушкина, 12",
|
||||||
"about_me": "Я люблю собак"
|
"price": "1500",
|
||||||
},
|
"about_me": "Я люблю собак!",
|
||||||
{
|
"rating": 5,
|
||||||
"id": 2,
|
"ratings": [
|
||||||
"phone_number": 89272844541,
|
5,
|
||||||
"first_name": "Ваня",
|
5
|
||||||
"second_name": "Пуськин",
|
],
|
||||||
"role": "dogsitter",
|
"tg": "jullllllie"
|
||||||
"location": "Россия, республика Татарстан, Казань, улица Абсалямова, 19",
|
},
|
||||||
"price": 1000000,
|
{
|
||||||
"about_me": "Я не люблю собак. И вообще я котоман."
|
"id": 2,
|
||||||
},
|
"phone_number": 89272844541,
|
||||||
{
|
"first_name": "Ваня",
|
||||||
"id": 3,
|
"second_name": "Пуськин",
|
||||||
"phone_number": 89872855893,
|
"role": "dogsitter",
|
||||||
"first_name": "Гадий",
|
"location": "Россия, республика Татарстан, Казань, улица Абсалямова, 19",
|
||||||
"second_name": "Петрович",
|
"price": 2000,
|
||||||
"role": "owner"
|
"about_me": "Я не люблю собак. И вообще я котоман.",
|
||||||
},
|
"rating": 4,
|
||||||
{
|
"ratings": [
|
||||||
"id": 4,
|
4,
|
||||||
"phone_number": 89872844591,
|
4
|
||||||
"first_name": "Галкин",
|
],
|
||||||
"second_name": "Максим",
|
"tg": "vanya006"
|
||||||
"role": "dogsitter",
|
},
|
||||||
"location": "Россия, республика Татарстан, Казань, проспект Ямашева, 83",
|
{
|
||||||
"price": 1000000,
|
"id": 3,
|
||||||
"about_me": "Миллион алых роз"
|
"phone_number": 89559999999,
|
||||||
}
|
"first_name": "Гадий",
|
||||||
]
|
"second_name": "Петрович",
|
||||||
|
"role": "owner"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": 4,
|
||||||
|
"phone_number": 89872844591,
|
||||||
|
"first_name": "Галкин",
|
||||||
|
"second_name": "Максим",
|
||||||
|
"role": "dogsitter",
|
||||||
|
"location": "Россия, республика Татарстан, Казань, проспект Ямашева, 83",
|
||||||
|
"price": 1750,
|
||||||
|
"about_me": "Миллион алых роз",
|
||||||
|
"rating": 4.5,
|
||||||
|
"ratings": [
|
||||||
|
4,
|
||||||
|
5
|
||||||
|
],
|
||||||
|
"tg": "maks100500"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"interactions": [
|
||||||
|
{
|
||||||
|
"owner_id": 3,
|
||||||
|
"dogsitter_id": 4
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"owner_id": 1,
|
||||||
|
"dogsitter_id": 2
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
const { Schema, model } = require("mongoose");
|
|
||||||
|
|
||||||
const {
|
|
||||||
DSF_AUTH_PASSWD_MODEL_NAME,
|
|
||||||
DSF_AUTH_USER_MODEL_NAME,
|
|
||||||
} = require("../../const");
|
|
||||||
|
|
||||||
const schema = new Schema({
|
|
||||||
login: {
|
|
||||||
type: String,
|
|
||||||
required: true,
|
|
||||||
unique: true
|
|
||||||
},
|
|
||||||
hash: {
|
|
||||||
type: String,
|
|
||||||
required: true
|
|
||||||
},
|
|
||||||
salt: {
|
|
||||||
type: String,
|
|
||||||
required: true
|
|
||||||
},
|
|
||||||
userId: {
|
|
||||||
type: Schema.Types.ObjectId,
|
|
||||||
ref: DSF_AUTH_USER_MODEL_NAME
|
|
||||||
},
|
|
||||||
created: {
|
|
||||||
type: Date,
|
|
||||||
default: () => new Date().toISOString(),
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
schema.set("toJSON", {
|
|
||||||
virtuals: true,
|
|
||||||
versionKey: false,
|
|
||||||
transform: function (doc, ret) {
|
|
||||||
delete ret._id;
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
schema.virtual("id").get(function () {
|
|
||||||
return this._id.toHexString();
|
|
||||||
});
|
|
||||||
|
|
||||||
exports.AuthModel = model(DSF_AUTH_PASSWD_MODEL_NAME, schema);
|
|
||||||
@@ -8,6 +8,10 @@ router.get("/update-like", (request, response) => {
|
|||||||
response.send(require("./json/gamepage/success.json"));
|
response.send(require("./json/gamepage/success.json"));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
router.get("/add-to-cart", (request, response) => {
|
||||||
|
response.send(require("./json/home-page-data/games-in-cart.json"));
|
||||||
|
});
|
||||||
|
|
||||||
router.get("/categories", (request, response) => {
|
router.get("/categories", (request, response) => {
|
||||||
response.send(require("./json/home-page-data/all-games.json"));
|
response.send(require("./json/home-page-data/all-games.json"));
|
||||||
});
|
});
|
||||||
@@ -16,18 +20,32 @@ router.get("/favourites", (request, response) => {
|
|||||||
response.send(require("./json/home-page-data/all-games.json"));
|
response.send(require("./json/home-page-data/all-games.json"));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// router.get("/shopping-cart", (request, response) => {
|
||||||
|
// response.send(require("./json/shopping-cart/success.json"));
|
||||||
|
// });
|
||||||
|
|
||||||
router.get("/shopping-cart", (request, response) => {
|
router.get("/shopping-cart", (request, response) => {
|
||||||
response.send(require("./json/shopping-cart/success.json"));
|
response.send(require("./json/home-page-data/games-in-cart.json"));
|
||||||
});
|
});
|
||||||
|
|
||||||
router.get("/home", (request, response) => {
|
// Добавляем поддержку разных ответов для /home
|
||||||
response.send(require("./json/home-page-data/success.json"));
|
router.get("/home", (req, res) => {
|
||||||
|
if (stubs.home === "success") {
|
||||||
|
res.send(require("./json/home-page-data/success.json"));
|
||||||
|
} else if (stubs.home === "empty") {
|
||||||
|
res.send({ data: [] }); // Отправляем пустой массив
|
||||||
|
} else {
|
||||||
|
res.status(500).json({ success: false, message: "Server error" });
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
router.get("/all-games", (request, response) => {
|
router.get("/all-games", (request, response) => {
|
||||||
response.send(require("./json/home-page-data/all-games.json"));
|
response.send(require("./json/home-page-data/all-games.json"));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const stubs = {
|
||||||
|
home: "success",
|
||||||
|
};
|
||||||
|
|
||||||
// // Маршрут для обновления лайков
|
// // Маршрут для обновления лайков
|
||||||
// router.post("/update-like", (request, response) => {
|
// router.post("/update-like", (request, response) => {
|
||||||
@@ -42,7 +60,6 @@ router.get("/all-games", (request, response) => {
|
|||||||
// });
|
// });
|
||||||
// });
|
// });
|
||||||
|
|
||||||
|
|
||||||
const fs = require("fs").promises;
|
const fs = require("fs").promises;
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
|
|
||||||
@@ -53,7 +70,7 @@ const commentsFilePath = path.join(__dirname, "./json/gamepage/success.json");
|
|||||||
async function readComments() {
|
async function readComments() {
|
||||||
const data = await fs.readFile(commentsFilePath, "utf-8");
|
const data = await fs.readFile(commentsFilePath, "utf-8");
|
||||||
const parsedData = JSON.parse(data);
|
const parsedData = JSON.parse(data);
|
||||||
console.log("Прочитанные данные:", parsedData); // Логируем полученные данные
|
console.log("Прочитанные данные:", parsedData); // Логируем полученные данные
|
||||||
return parsedData;
|
return parsedData;
|
||||||
}
|
}
|
||||||
// Write to JSON file
|
// Write to JSON file
|
||||||
@@ -92,5 +109,149 @@ router.post("/update-like", async (req, res) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Путь к JSON-файлу с корзиной
|
||||||
|
const cartFilePath = path.join(
|
||||||
|
__dirname,
|
||||||
|
"./json/home-page-data/games-in-cart.json"
|
||||||
|
);
|
||||||
|
|
||||||
|
// Функция для чтения JSON-файла
|
||||||
|
async function readCart() {
|
||||||
|
const data = await fs.readFile(cartFilePath, "utf-8");
|
||||||
|
return JSON.parse(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Функция для записи в JSON-файл
|
||||||
|
async function writeCart(data) {
|
||||||
|
await fs.writeFile(cartFilePath, JSON.stringify(data, null, 2), "utf-8");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Маршрут для добавления/удаления товара в корзине
|
||||||
|
router.post("/add-to-cart", async (req, res) => {
|
||||||
|
const { id, action } = req.body;
|
||||||
|
|
||||||
|
// Проверка наличия id и action
|
||||||
|
if (id === undefined || action === undefined) {
|
||||||
|
return res
|
||||||
|
.status(400)
|
||||||
|
.json({ success: false, message: "Invalid id or action" });
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const cartData = await readCart();
|
||||||
|
let ids = cartData.data.ids;
|
||||||
|
|
||||||
|
if (action === "add") {
|
||||||
|
// Если action "add", добавляем товар, если его нет в корзине
|
||||||
|
if (!ids?.includes(id)) {
|
||||||
|
ids.push(id);
|
||||||
|
}
|
||||||
|
} else if (action === "remove") {
|
||||||
|
// Если action "remove", удаляем товар, если он есть в корзине
|
||||||
|
if (ids?.includes(id)) {
|
||||||
|
ids = ids.filter((item) => item !== id);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Если action невалиден
|
||||||
|
return res
|
||||||
|
.status(400)
|
||||||
|
.json({ success: false, message: "Invalid action" });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Записываем обновленные данные обратно в файл
|
||||||
|
cartData.data.ids = ids;
|
||||||
|
await writeCart(cartData);
|
||||||
|
|
||||||
|
res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
message: "Cart updated successfully",
|
||||||
|
data: cartData.data, // Возвращаем обновленные данные
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error updating cart:", error);
|
||||||
|
res.status(500).json({ success: false, message: "Server error" });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
module.exports = router;
|
module.exports = router;
|
||||||
|
|
||||||
|
const createElement = (key, value, buttonTitle, basePath) => `
|
||||||
|
<label>
|
||||||
|
<input name="${key}" type="radio" ${
|
||||||
|
stubs[key] === value ? "checked" : ""
|
||||||
|
} onclick="fetch('${basePath}/admin/set/${key}/${value}')"/>
|
||||||
|
${buttonTitle || value}
|
||||||
|
</label>
|
||||||
|
`;
|
||||||
|
|
||||||
|
router.get("/admin/home", (request, response) => {
|
||||||
|
const basePath = request.baseUrl; // Получаем базовый путь маршрутизатора
|
||||||
|
response.send(`
|
||||||
|
<div>
|
||||||
|
<fieldset>
|
||||||
|
<legend>Настройка данных для /home</legend>
|
||||||
|
${createElement("home", "success", "Отдать успешный ответ", basePath)}
|
||||||
|
${createElement("home", "empty", "Отдать пустой массив", basePath)}
|
||||||
|
${createElement("home", "error", "Отдать ошибку", basePath)}
|
||||||
|
</fieldset>
|
||||||
|
</div>
|
||||||
|
`);
|
||||||
|
});
|
||||||
|
|
||||||
|
router.get("/admin/game-page", (request, response) => {
|
||||||
|
response.send(`
|
||||||
|
<div>
|
||||||
|
<fieldset>
|
||||||
|
<legend>Настройка данных для /game-page</legend>
|
||||||
|
${createElement(
|
||||||
|
"game-page",
|
||||||
|
"success",
|
||||||
|
"Отдать успешный ответ"
|
||||||
|
)}
|
||||||
|
${createElement("game-page", "empty", "Отдать пустой массив")}
|
||||||
|
${createElement("game-page", "error", "Отдать ошибку")}
|
||||||
|
|
||||||
|
</fieldset>
|
||||||
|
</div>
|
||||||
|
`);
|
||||||
|
});
|
||||||
|
|
||||||
|
router.get("/admin/categories", (request, response) => {
|
||||||
|
response.send(`
|
||||||
|
<div>
|
||||||
|
<fieldset>
|
||||||
|
<legend>Настройка данных для /categories</legend>
|
||||||
|
${createElement(
|
||||||
|
"categories",
|
||||||
|
"success",
|
||||||
|
"Отдать успешный ответ"
|
||||||
|
)}
|
||||||
|
${createElement("categories", "empty", "Отдать пустой массив")}
|
||||||
|
${createElement("categories", "error", "Отдать ошибку")}
|
||||||
|
</fieldset>
|
||||||
|
</div>
|
||||||
|
`);
|
||||||
|
});
|
||||||
|
|
||||||
|
router.get("/admin/favourites", (request, response) => {
|
||||||
|
response.send(`
|
||||||
|
<div>
|
||||||
|
<fieldset>
|
||||||
|
<legend>Настройка данных для /favourites</legend>
|
||||||
|
${createElement(
|
||||||
|
"favourites",
|
||||||
|
"success",
|
||||||
|
"Отдать успешный ответ"
|
||||||
|
)}
|
||||||
|
${createElement("favourites", "empty", "Отдать пустой массив")}
|
||||||
|
${createElement("favourites", "error", "Отдать ошибку")}
|
||||||
|
</fieldset>
|
||||||
|
</div>
|
||||||
|
`);
|
||||||
|
});
|
||||||
|
|
||||||
|
router.get("/admin/set/:key/:value", (request, response) => {
|
||||||
|
const { key, value } = request.params;
|
||||||
|
stubs[key] = value;
|
||||||
|
response.send("Настройки обновлены!");
|
||||||
|
});
|
||||||
@@ -5,28 +5,28 @@
|
|||||||
{
|
{
|
||||||
"username": "Пользователь1",
|
"username": "Пользователь1",
|
||||||
"text": "Текст комментария 1",
|
"text": "Текст комментария 1",
|
||||||
"likes": 11,
|
"likes": 13,
|
||||||
"rating": 8,
|
"rating": 8,
|
||||||
"date": "2025-03-01T10:00:00Z"
|
"date": "2025-03-01T10:00:00Z"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"username": "Пользователь2",
|
"username": "Пользователь2",
|
||||||
"text": "Текст комментария 2",
|
"text": "Текст комментария 2",
|
||||||
"likes": 7,
|
"likes": 10,
|
||||||
"rating": 7,
|
"rating": 7,
|
||||||
"date": "2025-01-01T10:00:00Z"
|
"date": "2025-01-01T10:00:00Z"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"username": "Пользователь3",
|
"username": "Пользователь3",
|
||||||
"text": "Текст комментария 3",
|
"text": "Текст комментария 3",
|
||||||
"likes": 2,
|
"likes": 4,
|
||||||
"rating": 3,
|
"rating": 3,
|
||||||
"date": "2025-02-01T10:00:00Z"
|
"date": "2025-02-01T10:00:00Z"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"username": "Пользователь4",
|
"username": "Пользователь4",
|
||||||
"text": "Текст комментария 4",
|
"text": "Текст комментария 4",
|
||||||
"likes": 15,
|
"likes": 18,
|
||||||
"rating": 2,
|
"rating": 2,
|
||||||
"date": "2025-12-01T10:00:00Z"
|
"date": "2025-12-01T10:00:00Z"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -105,23 +105,27 @@
|
|||||||
{
|
{
|
||||||
"image": "news1",
|
"image": "news1",
|
||||||
"text": "Разработчики Delta Force: Hawk Ops представили крупномасштабный режим Havoc Warfare",
|
"text": "Разработчики Delta Force: Hawk Ops представили крупномасштабный режим Havoc Warfare",
|
||||||
"imgPath": "img_news_1"
|
"imgPath": "img_news_1",
|
||||||
|
"link": "https://gamemag.ru/news/185583/delta-force-hawk-ops-gameplay-showcase-havoc-warfare"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"image": "news2",
|
"image": "news2",
|
||||||
"text": "Первый трейлер Assassin’s Creed Shadows — с темнокожим самураем в феодальной Японии",
|
"text": "Первый трейлер Assassin’s Creed Shadows — с темнокожим самураем в феодальной Японии",
|
||||||
"imgPath": "img_news_2"
|
"imgPath": "img_news_2",
|
||||||
|
"link": "https://stopgame.ru/newsdata/62686/pervyy_trailer_assassin_s_creed_shadows_s_temnokozhim_samuraem_v_feodalnoy_yaponii"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"image": "news3",
|
"image": "news3",
|
||||||
"text": "Призрак Цусимы» вышел на ПК — и уже ставит рекорды для Sony",
|
"text": "Призрак Цусимы» вышел на ПК — и уже ставит рекорды для Sony",
|
||||||
"imgPath": "img_news_3"
|
"imgPath": "img_news_3",
|
||||||
|
"link": "https://stopgame.ru/newsdata/62706/prizrak_cusimy_vyshel_na_pk_i_uzhe_stavit_rekordy_dlya_sony"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"image": "news4",
|
"image": "news4",
|
||||||
"text": "Авторы Skull and Bones расширяют планы на второй сезо",
|
"text": "Авторы Skull and Bones расширяют планы на второй сезон",
|
||||||
"imgPath": "img_news_4"
|
"imgPath": "img_news_4",
|
||||||
|
"link": "https://stopgame.ru/newsdata/62711/avtory_skull_and_bones_rasshiryayut_plany_na_vtoroy_sezon"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user