fix: add sending form data

This commit is contained in:
2025-06-11 13:40:10 +03:00
parent 3fd44fba14
commit 53f214ba28
11 changed files with 121 additions and 27 deletions

View File

@@ -1,13 +1,8 @@
import nodemailer from 'nodemailer'; import nodemailer from 'nodemailer';
import { TBaseForm } from '@shared/api/api.types';
import { CORE } from '@shared/config/core';
type TFormData = { async function sendMail(data: TBaseForm) {
name?: string;
phone: string;
message?: string;
form: string;
};
async function sendMail(data: TFormData) {
const { name, phone, message, form } = data; const { name, phone, message, form } = data;
const formattedBody = ` const formattedBody = `
@@ -25,14 +20,14 @@ async function sendMail(data: TFormData) {
const transporter = nodemailer.createTransport({ const transporter = nodemailer.createTransport({
service: 'yandex', service: 'yandex',
auth: { auth: {
user: process.env.MAIL_USER, user: CORE.MAIL_USER,
pass: process.env.MAIL_PASS, pass: CORE.MAIL_PASS,
}, },
}); });
return await transporter.sendMail({ return await transporter.sendMail({
from: process.env.MAIL_FROM, from: CORE.MAIL_FROM,
to: process.env.MAIL_TO, to: CORE.MAIL_TO,
subject: 'Заявка с сайта FireExams', subject: 'Заявка с сайта FireExams',
html: formattedBody, html: formattedBody,
}); });
@@ -42,14 +37,24 @@ export async function POST(request: Request) {
try { try {
const payload = await request.json(); const payload = await request.json();
if (payload.secure === process.env.MAIL_SECURE_KEY) { if (payload.secure !== CORE.MAIL_SECURE_KEY) {
await Promise.reject('Request failure!'); await Promise.reject('Request failure!');
} }
const sendResult = await sendMail({ ...payload }); const sendResult = await sendMail({ ...payload });
const data = { message: 'Form accepted' };
const headers = new Headers({
'Content-Type': 'application/json',
});
const options = {
status: 200,
statusText: 'OK',
headers: headers,
};
if (sendResult?.messageId) { if (sendResult?.messageId) {
return new Response('Success!', { status: 200 }); return new Response(JSON.stringify(data), options);
} else { } else {
await Promise.reject('Sending request failure!'); await Promise.reject('Sending request failure!');
} }

View File

@@ -0,0 +1,35 @@
import { API_ROUTES } from '@shared/config/routes';
import { TBaseForm } from '@shared/api/api.types';
import { CORE } from '@shared/config/core';
type TRequest = TBaseForm;
const sendFormFn = async ({ ...props }: TRequest) => {
try {
const response = await fetch('/api' + API_ROUTES.SEND_FORM, {
method: 'POST',
credentials: 'include',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
...props,
secure: CORE.MAIL_SECURE_KEY,
}),
});
if (response.ok) {
return response;
} else {
if (response.status === 400) throw new Error(`400 Bad request`);
if (response.status === 401) throw new Error(`401 Unauthorized`);
if (response.status === 500) throw new Error('500 Internal server error');
throw new Error(`${response.status} - Network response failure`);
}
} catch (e) {
console.error(e);
}
};
export { sendFormFn };

View File

@@ -0,0 +1,6 @@
export type TBaseForm = {
form: string;
name?: string;
phone: string;
message?: string;
};

View File

@@ -0,0 +1,7 @@
export const CORE = {
MAIL_USER: process.env.MAIL_USER,
MAIL_PASS: process.env.MAIL_PASS,
MAIL_FROM: process.env.MAIL_FROM,
MAIL_TO: process.env.MAIL_TO,
MAIL_SECURE_KEY: process.env.NEXT_PUBLIC_MAIL_SECURE_KEY,
} as const;

View File

@@ -0,0 +1,4 @@
export const API_ROUTES = {
SEND_FORM: '/sendform',
HEARTBEAT: '/heartbeat',
} as const;

View File

@@ -0,0 +1,10 @@
//import s from './phone-input.module.scss';
import { Input } from '@shared/ui';
export default function PhoneInput() {
return (
<>
<Input />
</>
);
}

View File

@@ -9,6 +9,7 @@ import { zodResolver } from '@hookform/resolvers/zod';
import { z } from 'zod'; import { z } from 'zod';
import bgForm from '@public/images/bg-form.jpg'; import bgForm from '@public/images/bg-form.jpg';
import { sendFormFn } from '@shared/api/api.service';
const FormSchema = z.object({ const FormSchema = z.object({
name: z.string().min(3), name: z.string().min(3),
@@ -27,7 +28,6 @@ export default function ContactsForm() {
control, control,
reset, reset,
formState: { errors }, formState: { errors },
clearErrors,
} = useForm<TForm>({ } = useForm<TForm>({
mode: 'onSubmit', mode: 'onSubmit',
reValidateMode: 'onBlur', reValidateMode: 'onBlur',
@@ -36,8 +36,13 @@ export default function ContactsForm() {
}); });
const onSubmit = async (data: TForm) => { const onSubmit = async (data: TForm) => {
const payload = {
...data,
form: 'contacts-form',
};
try { try {
console.log('Form', data); await sendFormFn(payload);
toast.success('Заявка на консультацию принята'); toast.success('Заявка на консультацию принята');
reset(defaultValues); reset(defaultValues);
} catch (e) { } catch (e) {

View File

@@ -8,6 +8,7 @@ import { zodResolver } from '@hookform/resolvers/zod';
import { z } from 'zod'; import { z } from 'zod';
import man from '@public/images/footer-man.png'; import man from '@public/images/footer-man.png';
import { sendFormFn } from '@shared/api/api.service';
const FormSchema = z.object({ const FormSchema = z.object({
name: z.string().min(3), name: z.string().min(3),
@@ -28,7 +29,6 @@ export default function FooterForm() {
control, control,
reset, reset,
formState: { errors }, formState: { errors },
clearErrors,
} = useForm<TForm>({ } = useForm<TForm>({
mode: 'onSubmit', mode: 'onSubmit',
reValidateMode: 'onBlur', reValidateMode: 'onBlur',
@@ -37,8 +37,13 @@ export default function FooterForm() {
}); });
const onSubmit = async (data: TForm) => { const onSubmit = async (data: TForm) => {
const payload = {
...data,
form: 'footer-form',
};
try { try {
console.log('Form', data); await sendFormFn(payload);
toast.success('Заявка на консультацию принята'); toast.success('Заявка на консультацию принята');
reset(defaultValues); reset(defaultValues);
} catch (e) { } catch (e) {

View File

@@ -9,6 +9,7 @@ import bgForm from '@public/images/bg-form.jpg';
import { z } from 'zod'; import { z } from 'zod';
import { Controller, useForm } from 'react-hook-form'; import { Controller, useForm } from 'react-hook-form';
import { zodResolver } from '@hookform/resolvers/zod'; import { zodResolver } from '@hookform/resolvers/zod';
import { sendFormFn } from '@shared/api/api.service';
const FormSchema = z.object({ const FormSchema = z.object({
name: z.string().min(3), name: z.string().min(3),
@@ -27,7 +28,6 @@ export default function LicenseForm() {
control, control,
reset, reset,
formState: { errors }, formState: { errors },
clearErrors,
} = useForm<TForm>({ } = useForm<TForm>({
mode: 'onSubmit', mode: 'onSubmit',
reValidateMode: 'onBlur', reValidateMode: 'onBlur',
@@ -36,8 +36,13 @@ export default function LicenseForm() {
}); });
const onSubmit = async (data: TForm) => { const onSubmit = async (data: TForm) => {
const payload = {
...data,
form: 'license-form',
};
try { try {
console.log('Form', data); await sendFormFn(payload);
toast.success('Заявка на консультацию принята'); toast.success('Заявка на консультацию принята');
reset(defaultValues); reset(defaultValues);
} catch (e) { } catch (e) {

View File

@@ -6,6 +6,7 @@ import { Controller, useForm } from 'react-hook-form';
import { zodResolver } from '@hookform/resolvers/zod'; import { zodResolver } from '@hookform/resolvers/zod';
import { z } from 'zod'; import { z } from 'zod';
import toast from 'react-hot-toast'; import toast from 'react-hot-toast';
import { sendFormFn } from '@shared/api/api.service';
const FormSchema = z.object({ const FormSchema = z.object({
name: z.string().min(3), name: z.string().min(3),
@@ -24,7 +25,6 @@ export default function OfferForm() {
control, control,
reset, reset,
formState: { errors }, formState: { errors },
clearErrors,
} = useForm<TForm>({ } = useForm<TForm>({
mode: 'onSubmit', mode: 'onSubmit',
reValidateMode: 'onBlur', reValidateMode: 'onBlur',
@@ -33,8 +33,13 @@ export default function OfferForm() {
}); });
const onSubmit = async (data: TForm) => { const onSubmit = async (data: TForm) => {
const payload = {
...data,
form: 'offer-form',
};
try { try {
console.log('Form', data); await sendFormFn(payload);
toast.success('Заявка на консультацию принята'); toast.success('Заявка на консультацию принята');
reset(defaultValues); reset(defaultValues);
} catch (e) { } catch (e) {

View File

@@ -10,6 +10,7 @@ import toast from 'react-hot-toast';
import bgForm from '@public/images/bg-form.jpg'; import bgForm from '@public/images/bg-form.jpg';
import { useState } from 'react'; import { useState } from 'react';
import { sendFormFn } from '@shared/api/api.service';
const FormSchema = z.object({ const FormSchema = z.object({
name: z.string().min(3), name: z.string().min(3),
@@ -28,7 +29,6 @@ export default function OfferRequest() {
control, control,
reset, reset,
formState: { errors }, formState: { errors },
clearErrors,
} = useForm<TForm>({ } = useForm<TForm>({
mode: 'onSubmit', mode: 'onSubmit',
reValidateMode: 'onBlur', reValidateMode: 'onBlur',
@@ -39,8 +39,13 @@ export default function OfferRequest() {
const [inputPhone, setInputPhone] = useState(''); const [inputPhone, setInputPhone] = useState('');
const onSubmitForm = async (data: TForm) => { const onSubmitForm = async (data: TForm) => {
const payload = {
...data,
form: 'offer-request-form',
};
try { try {
console.log('Form', data); await sendFormFn(payload);
toast.success('Заявка на консультацию принята'); toast.success('Заявка на консультацию принята');
reset(defaultValues); reset(defaultValues);
} catch (e) { } catch (e) {
@@ -51,11 +56,13 @@ export default function OfferRequest() {
}; };
const onSubmitPhone = async (phone: string) => { const onSubmitPhone = async (phone: string) => {
const data = { const payload = {
phone, phone: phone,
form: 'offer-request-form',
}; };
try { try {
console.log('Form', data); await sendFormFn(payload);
toast.success('Заявка на консультацию принята'); toast.success('Заявка на консультацию принята');
setInputPhone(''); setInputPhone('');
} catch (e) { } catch (e) {