Новые измнения в коде
This commit is contained in:
parent
beeec4740e
commit
4b9735118b
4 changed files with 251 additions and 21 deletions
113
backend/pb_migrations/1776610409_created_consultations.js
Normal file
113
backend/pb_migrations/1776610409_created_consultations.js
Normal file
|
|
@ -0,0 +1,113 @@
|
||||||
|
/// <reference path="../pb_data/types.d.ts" />
|
||||||
|
migrate((app) => {
|
||||||
|
const collection = new Collection({
|
||||||
|
"createRule": null,
|
||||||
|
"deleteRule": null,
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"autogeneratePattern": "[a-z0-9]{15}",
|
||||||
|
"hidden": false,
|
||||||
|
"id": "text3208210256",
|
||||||
|
"max": 15,
|
||||||
|
"min": 15,
|
||||||
|
"name": "id",
|
||||||
|
"pattern": "^[a-z0-9]+$",
|
||||||
|
"presentable": false,
|
||||||
|
"primaryKey": true,
|
||||||
|
"required": true,
|
||||||
|
"system": true,
|
||||||
|
"type": "text"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"autogeneratePattern": "",
|
||||||
|
"hidden": false,
|
||||||
|
"id": "text1579384326",
|
||||||
|
"max": 0,
|
||||||
|
"min": 0,
|
||||||
|
"name": "name",
|
||||||
|
"pattern": "",
|
||||||
|
"presentable": false,
|
||||||
|
"primaryKey": false,
|
||||||
|
"required": false,
|
||||||
|
"system": false,
|
||||||
|
"type": "text"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"autogeneratePattern": "",
|
||||||
|
"hidden": false,
|
||||||
|
"id": "text1146066909",
|
||||||
|
"max": 0,
|
||||||
|
"min": 0,
|
||||||
|
"name": "phone",
|
||||||
|
"pattern": "",
|
||||||
|
"presentable": false,
|
||||||
|
"primaryKey": false,
|
||||||
|
"required": false,
|
||||||
|
"system": false,
|
||||||
|
"type": "text"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"autogeneratePattern": "",
|
||||||
|
"hidden": false,
|
||||||
|
"id": "text3785202386",
|
||||||
|
"max": 0,
|
||||||
|
"min": 0,
|
||||||
|
"name": "service",
|
||||||
|
"pattern": "",
|
||||||
|
"presentable": false,
|
||||||
|
"primaryKey": false,
|
||||||
|
"required": false,
|
||||||
|
"system": false,
|
||||||
|
"type": "text"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"autogeneratePattern": "",
|
||||||
|
"hidden": false,
|
||||||
|
"id": "text2063623452",
|
||||||
|
"max": 0,
|
||||||
|
"min": 0,
|
||||||
|
"name": "status",
|
||||||
|
"pattern": "",
|
||||||
|
"presentable": false,
|
||||||
|
"primaryKey": false,
|
||||||
|
"required": false,
|
||||||
|
"system": false,
|
||||||
|
"type": "text"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": false,
|
||||||
|
"id": "autodate2990389176",
|
||||||
|
"name": "created",
|
||||||
|
"onCreate": true,
|
||||||
|
"onUpdate": false,
|
||||||
|
"presentable": false,
|
||||||
|
"system": false,
|
||||||
|
"type": "autodate"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": false,
|
||||||
|
"id": "autodate3332085495",
|
||||||
|
"name": "updated",
|
||||||
|
"onCreate": true,
|
||||||
|
"onUpdate": true,
|
||||||
|
"presentable": false,
|
||||||
|
"system": false,
|
||||||
|
"type": "autodate"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"id": "pbc_3441013282",
|
||||||
|
"indexes": [],
|
||||||
|
"listRule": null,
|
||||||
|
"name": "consultations",
|
||||||
|
"system": false,
|
||||||
|
"type": "base",
|
||||||
|
"updateRule": null,
|
||||||
|
"viewRule": null
|
||||||
|
});
|
||||||
|
|
||||||
|
return app.save(collection);
|
||||||
|
}, (app) => {
|
||||||
|
const collection = app.findCollectionByNameOrId("pbc_3441013282");
|
||||||
|
|
||||||
|
return app.delete(collection);
|
||||||
|
})
|
||||||
24
backend/pb_migrations/1776610679_updated_consultations.js
Normal file
24
backend/pb_migrations/1776610679_updated_consultations.js
Normal file
|
|
@ -0,0 +1,24 @@
|
||||||
|
/// <reference path="../pb_data/types.d.ts" />
|
||||||
|
migrate((app) => {
|
||||||
|
const collection = app.findCollectionByNameOrId("pbc_3441013282")
|
||||||
|
|
||||||
|
// update collection data
|
||||||
|
unmarshal({
|
||||||
|
"createRule": "",
|
||||||
|
"deleteRule": "@request.auth.id != \"\"",
|
||||||
|
"updateRule": "@request.auth.id != \"\""
|
||||||
|
}, collection)
|
||||||
|
|
||||||
|
return app.save(collection)
|
||||||
|
}, (app) => {
|
||||||
|
const collection = app.findCollectionByNameOrId("pbc_3441013282")
|
||||||
|
|
||||||
|
// update collection data
|
||||||
|
unmarshal({
|
||||||
|
"createRule": null,
|
||||||
|
"deleteRule": null,
|
||||||
|
"updateRule": null
|
||||||
|
}, collection)
|
||||||
|
|
||||||
|
return app.save(collection)
|
||||||
|
})
|
||||||
|
|
@ -69,6 +69,7 @@ const title = 'Бесплатная консультация';
|
||||||
modal.classList.add('active');
|
modal.classList.add('active');
|
||||||
modal.setAttribute('aria-hidden', 'false');
|
modal.setAttribute('aria-hidden', 'false');
|
||||||
document.body.style.overflow = 'hidden';
|
document.body.style.overflow = 'hidden';
|
||||||
|
startAutoCloseTimer();
|
||||||
}
|
}
|
||||||
|
|
||||||
function closeModal() {
|
function closeModal() {
|
||||||
|
|
@ -76,6 +77,24 @@ const title = 'Бесплатная консультация';
|
||||||
modal.classList.remove('active');
|
modal.classList.remove('active');
|
||||||
modal.setAttribute('aria-hidden', 'true');
|
modal.setAttribute('aria-hidden', 'true');
|
||||||
document.body.style.overflow = '';
|
document.body.style.overflow = '';
|
||||||
|
resetAutoCloseTimer();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Таймер для автозакрытия
|
||||||
|
let autoCloseTimer: ReturnType<typeof setTimeout> | null = null;
|
||||||
|
|
||||||
|
function startAutoCloseTimer() {
|
||||||
|
if (autoCloseTimer) clearTimeout(autoCloseTimer);
|
||||||
|
autoCloseTimer = setTimeout(() => {
|
||||||
|
closeModal();
|
||||||
|
}, 5000);
|
||||||
|
}
|
||||||
|
|
||||||
|
function resetAutoCloseTimer() {
|
||||||
|
if (autoCloseTimer) {
|
||||||
|
clearTimeout(autoCloseTimer);
|
||||||
|
autoCloseTimer = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Открытие по кастомному событию
|
// Открытие по кастомному событию
|
||||||
|
|
@ -119,7 +138,7 @@ const title = 'Бесплатная консультация';
|
||||||
const phone = formData.get('phone');
|
const phone = formData.get('phone');
|
||||||
|
|
||||||
if (!name || !phone) {
|
if (!name || !phone) {
|
||||||
alert('Пожалуйста, заполните имя и телефон');
|
(window as any).toast?.show('Пожалуйста, заполните имя и телефон', 'error', 3000);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -133,18 +152,22 @@ const title = 'Бесплатная консультация';
|
||||||
const result = await response.json();
|
const result = await response.json();
|
||||||
|
|
||||||
if (result.success) {
|
if (result.success) {
|
||||||
alert('Спасибо! Мы свяжемся с вами в течение 15 минут.');
|
(window as any).toast?.show('Спасибо! Мы свяжемся с вами в течение 15 минут.', 'success', 4000);
|
||||||
form.reset();
|
form.reset();
|
||||||
closeModal();
|
closeModal();
|
||||||
} else {
|
} else {
|
||||||
alert('Ошибка: ' + result.error);
|
(window as any).toast?.show(result.error || 'Ошибка отправки', 'error', 4000);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Error:', error);
|
console.error('Error:', error);
|
||||||
alert('Ошибка отправки. Попробуйте позже.');
|
(window as any).toast?.show('Ошибка отправки. Попробуйте позже.', 'error', 4000);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Сброс таймера при активности пользователя
|
||||||
|
modal?.addEventListener('mousemove', resetAutoCloseTimer);
|
||||||
|
modal?.addEventListener('click', resetAutoCloseTimer);
|
||||||
|
|
||||||
// Для Astro View Transitions
|
// Для Astro View Transitions
|
||||||
document.addEventListener('astro:page-load', () => {
|
document.addEventListener('astro:page-load', () => {
|
||||||
// Переинициализация при навигации
|
// Переинициализация при навигации
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,17 @@
|
||||||
import type { APIRoute } from 'astro';
|
import type { APIRoute } from 'astro';
|
||||||
import { pb } from '../../lib/pb';
|
import nodemailer from 'nodemailer';
|
||||||
|
|
||||||
|
const POCKETBASE_URL = import.meta.env.POCKETBASE_URL || 'http://localhost:8090';
|
||||||
|
const SMTP_HOST = import.meta.env.SMTP_HOST || 'localhost';
|
||||||
|
const SMTP_PORT = import.meta.env.SMTP_PORT || '1025';
|
||||||
|
const NOTIFY_EMAIL = import.meta.env.NOTIFY_EMAIL || 'info@avtourist.ru';
|
||||||
|
|
||||||
|
const transporter = nodemailer.createTransport({
|
||||||
|
host: SMTP_HOST,
|
||||||
|
port: parseInt(SMTP_PORT),
|
||||||
|
secure: false,
|
||||||
|
ignoreTLS: true,
|
||||||
|
});
|
||||||
|
|
||||||
const RATE_LIMIT_WINDOW = 60 * 1000;
|
const RATE_LIMIT_WINDOW = 60 * 1000;
|
||||||
const MAX_REQUESTS = 3;
|
const MAX_REQUESTS = 3;
|
||||||
|
|
@ -37,55 +49,113 @@ export const POST: APIRoute = async ({ request }) => {
|
||||||
return new Response(JSON.stringify({
|
return new Response(JSON.stringify({
|
||||||
success: false,
|
success: false,
|
||||||
error: 'Слишком много запросов. Попробуйте позже.'
|
error: 'Слишком много запросов. Попробуйте позже.'
|
||||||
}), { status: 429 });
|
}), { status: 429, headers: { 'Content-Type': 'application/json' } });
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const data = await request.json();
|
const data = await request.json();
|
||||||
|
|
||||||
const { name, phone, service, website } = data;
|
const { name, phone, website } = data;
|
||||||
|
|
||||||
if (website) {
|
if (website) {
|
||||||
return new Response(JSON.stringify({
|
return new Response(JSON.stringify({
|
||||||
success: false,
|
success: false,
|
||||||
error: 'Спам обнаружен'
|
error: 'Спам обнаружен'
|
||||||
}), { status: 400 });
|
}), { status: 400, headers: { 'Content-Type': 'application/json' } });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!name || !phone) {
|
if (!name || !phone) {
|
||||||
return new Response(JSON.stringify({
|
return new Response(JSON.stringify({
|
||||||
success: false,
|
success: false,
|
||||||
error: 'Имя и телефон обязательны'
|
error: 'Имя и телефон обязательны'
|
||||||
}), { status: 400 });
|
}), { status: 400, headers: { 'Content-Type': 'application/json' } });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (name.length < 2 || name.length > 100) {
|
if (name.length < 2 || name.length > 100) {
|
||||||
return new Response(JSON.stringify({
|
return new Response(JSON.stringify({
|
||||||
success: false,
|
success: false,
|
||||||
error: 'Некорректное имя'
|
error: 'Некорректное имя'
|
||||||
}), { status: 400 });
|
}), { status: 400, headers: { 'Content-Type': 'application/json' } });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!validatePhone(phone)) {
|
if (!validatePhone(phone)) {
|
||||||
return new Response(JSON.stringify({
|
return new Response(JSON.stringify({
|
||||||
success: false,
|
success: false,
|
||||||
error: 'Некорректный номер телефона'
|
error: 'Некорректный номер телефона'
|
||||||
}), { status: 400 });
|
}), { status: 400, headers: { 'Content-Type': 'application/json' } });
|
||||||
}
|
}
|
||||||
|
|
||||||
const record = await pb.collection('consultations').create({
|
const cleanPhone = phone.replace(/\D/g, '');
|
||||||
|
|
||||||
|
const response = await fetch(
|
||||||
|
`${POCKETBASE_URL}/api/collections/consultations/records`,
|
||||||
|
{
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
name: name.trim(),
|
name: name.trim(),
|
||||||
phone: phone.replace(/\D/g, ''),
|
phone: cleanPhone,
|
||||||
service: service || '',
|
|
||||||
status: 'new',
|
status: 'new',
|
||||||
created_at: new Date().toISOString(),
|
}),
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const result = await response.json();
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
return new Response(JSON.stringify({
|
||||||
|
success: false,
|
||||||
|
error: result.message || 'Ошибка при отправке заявки'
|
||||||
|
}), { status: response.status, headers: { 'Content-Type': 'application/json' } });
|
||||||
|
}
|
||||||
|
|
||||||
|
const emailHtml = `
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
</head>
|
||||||
|
<body style="margin: 0; padding: 20px; font-family: Arial, sans-serif; background-color: #f5f7fa;">
|
||||||
|
<div style="max-width: 600px; margin: 0 auto; background: #ffffff; border-radius: 12px; padding: 30px;">
|
||||||
|
<h2 style="color: #1e3050; margin: 0 0 20px;">Новая заявка на консультацию!</h2>
|
||||||
|
<table style="width: 100%; border-collapse: collapse;">
|
||||||
|
<tr>
|
||||||
|
<td style="padding: 10px 0; border-bottom: 1px solid #e0e4e8;"><strong>Имя:</strong></td>
|
||||||
|
<td style="padding: 10px 0; border-bottom: 1px solid #e0e4e8;">${name}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td style="padding: 10px 0; border-bottom: 1px solid #e0e4e8;"><strong>Телефон:</strong></td>
|
||||||
|
<td style="padding: 10px 0; border-bottom: 1px solid #e0e4e8;"><a href="tel:${cleanPhone}">${phone}</a></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td style="padding: 10px 0;"><strong>Дата:</strong></td>
|
||||||
|
<td style="padding: 10px 0;">${new Date().toLocaleString('ru-RU')}</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`.trim();
|
||||||
|
|
||||||
|
try {
|
||||||
|
await transporter.sendMail({
|
||||||
|
from: 'avtourist@surgut.ru',
|
||||||
|
to: NOTIFY_EMAIL,
|
||||||
|
subject: `Новая заявка от ${name}`,
|
||||||
|
html: emailHtml,
|
||||||
});
|
});
|
||||||
|
console.log('Email notification sent');
|
||||||
|
} catch (emailError) {
|
||||||
|
console.error('Email send error:', emailError);
|
||||||
|
}
|
||||||
|
|
||||||
return new Response(JSON.stringify({
|
return new Response(JSON.stringify({
|
||||||
success: true,
|
success: true,
|
||||||
message: 'Заявка отправлена! Мы свяжемся с вами в течение 15 минут.',
|
message: 'Заявка отправлена! Мы свяжемся с вами в течение 15 минут.',
|
||||||
id: record.id
|
id: result.id
|
||||||
}), { status: 201 });
|
}), { status: 201, headers: { 'Content-Type': 'application/json' } });
|
||||||
|
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
console.error('Consultation error:', error);
|
console.error('Consultation error:', error);
|
||||||
|
|
@ -93,6 +163,6 @@ export const POST: APIRoute = async ({ request }) => {
|
||||||
return new Response(JSON.stringify({
|
return new Response(JSON.stringify({
|
||||||
success: false,
|
success: false,
|
||||||
error: error.message || 'Ошибка при отправке заявки'
|
error: error.message || 'Ошибка при отправке заявки'
|
||||||
}), { status: 400 });
|
}), { status: 400, headers: { 'Content-Type': 'application/json' } });
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
Loading…
Add table
Add a link
Reference in a new issue