From f8de5e9c0f7482ee11b772dafd33f7dcfd488bc7 Mon Sep 17 00:00:00 2001 From: karlosvas <126987511+karlosvas@users.noreply.github.com> Date: Sun, 12 Jan 2025 10:51:33 +0100 Subject: [PATCH] Github actions to create dist in server --- .github/workflows/deploy.yml | 82 ++++++++ .gitignore | 1 - client/.gitignore | 1 + client/vercel.json | 7 - package-lock.json | 18 ++ package.json | 5 +- server/.gitignore | 1 + server/index.ts | 2 +- server/lib/resend/resend.ts | 2 +- server/package-lock.json | 25 --- server/package.json | 4 +- server/public/index.js | 114 +++++++++++ server/public/lib/firebase/firebase-config.js | 41 ++++ server/public/lib/firebase/firebase-db.js | 12 ++ server/public/lib/mailchimp/mailchimp.js | 69 +++++++ server/public/lib/resend/resend.js | 58 ++++++ server/public/middelware/token-logs.js | 27 +++ server/public/playwright.config.js | 41 ++++ server/public/routes/comments.js | 182 ++++++++++++++++++ server/public/routes/firebase.js | 23 +++ server/public/routes/mailchimp.js | 181 +++++++++++++++++ server/public/routes/publications.js | 119 ++++++++++++ server/public/routes/resend.js | 25 +++ server/public/routes/routes.js | 33 ++++ server/public/src/mongodb/models.js | 29 +++ server/public/src/mongodb/mongodb.js | 15 ++ server/public/tests/comments.spec.js | 103 ++++++++++ server/public/tests/index.spec.js | 8 + server/public/tests/mailchimp.spec.js | 141 ++++++++++++++ server/public/tests/mandril.spec.js | 35 ++++ server/public/tests/publications.spec.js | 62 ++++++ server/public/utilities/axios-utils.js | 3 + server/public/utilities/delete-logic.js | 13 ++ server/public/utilities/errorHandle.js | 10 + server/routes/routes.ts | 7 +- server/tsconfig.json | 4 +- server/vercel.json | 16 -- vercel.json | 38 ++++ 38 files changed, 1500 insertions(+), 57 deletions(-) create mode 100644 .github/workflows/deploy.yml create mode 100644 client/.gitignore delete mode 100644 client/vercel.json create mode 100644 server/.gitignore create mode 100644 server/public/index.js create mode 100644 server/public/lib/firebase/firebase-config.js create mode 100644 server/public/lib/firebase/firebase-db.js create mode 100644 server/public/lib/mailchimp/mailchimp.js create mode 100644 server/public/lib/resend/resend.js create mode 100644 server/public/middelware/token-logs.js create mode 100644 server/public/playwright.config.js create mode 100644 server/public/routes/comments.js create mode 100644 server/public/routes/firebase.js create mode 100644 server/public/routes/mailchimp.js create mode 100644 server/public/routes/publications.js create mode 100644 server/public/routes/resend.js create mode 100644 server/public/routes/routes.js create mode 100644 server/public/src/mongodb/models.js create mode 100644 server/public/src/mongodb/mongodb.js create mode 100644 server/public/tests/comments.spec.js create mode 100644 server/public/tests/index.spec.js create mode 100644 server/public/tests/mailchimp.spec.js create mode 100644 server/public/tests/mandril.spec.js create mode 100644 server/public/tests/publications.spec.js create mode 100644 server/public/utilities/axios-utils.js create mode 100644 server/public/utilities/delete-logic.js create mode 100644 server/public/utilities/errorHandle.js delete mode 100644 server/vercel.json create mode 100644 vercel.json diff --git a/.github/workflows/deploy.yml b/.github/workflows/deploy.yml new file mode 100644 index 00000000..37a64b12 --- /dev/null +++ b/.github/workflows/deploy.yml @@ -0,0 +1,82 @@ +name: Deploy to Vercel + +on: + push: + branches: + - main + +jobs: + deploy-frontend: + runs-on: ubuntu-latest + + steps: + - name: Checkout repository + uses: actions/checkout@v3 + + - name: Set up Node.js + uses: actions/setup-node@v3 + with: + node-version: "21.2.0" + + - name: Install Vercel CLI + run: npm install -g vercel@latest + + - name: List files in client directory + run: ls -la ./client + + - name: Link and pull Vercel project for frontend + run: | + vercel link --yes --project ${{ secrets.FRONTEND_PROJECT_ID }} --scope ${{ secrets.VERCEL_TEAM }} --token=${{ secrets.VERCEL_TOKEN }} + vercel pull --yes --environment=preview --token=${{ secrets.VERCEL_TOKEN }} + working-directory: ./client + + - name: Install frontend dependencies + run: npm install + working-directory: ./client + + - name: Build frontend + run: npm run build + working-directory: ./client + + - name: Deploy frontend to Vercel + run: vercel --token ${{ secrets.VERCEL_TOKEN }} + working-directory: ./client + + deploy-backend: + runs-on: ubuntu-latest + + steps: + - name: Checkout repository + uses: actions/checkout@v3 + + - name: Set up Node.js + uses: actions/setup-node@v3 + with: + node-version: "21.2.0" + + - name: Install Vercel CLI + run: npm install -g vercel@latest + + - name: List files in server directory + run: ls -la ./server + + - name: Link and pull Vercel project for backend + run: | + vercel link --yes --project ${{ secrets.BACKEND_PROJECT_ID }} --scope ${{ secrets.VERCEL_TEAM }} --token=${{ secrets.VERCEL_TOKEN }} + vercel pull --yes --environment=preview --token=${{ secrets.VERCEL_TOKEN }} + working-directory: ./server + + - name: Install backend dependencies + run: npm install + working-directory: ./server + + - name: Build backend + run: npm run build + working-directory: ./server + + - name: List files in client directory + run: ls -la ./server + + - name: Deploy backend to Vercel + run: vercel --token ${{ secrets.VERCEL_TOKEN }} + working-directory: ./server diff --git a/.gitignore b/.gitignore index 907aa4c4..7d1bee05 100644 --- a/.gitignore +++ b/.gitignore @@ -12,7 +12,6 @@ dev dist-ssr client/dist *.local -dist # Editor directories and files .vscode/* diff --git a/client/.gitignore b/client/.gitignore new file mode 100644 index 00000000..e985853e --- /dev/null +++ b/client/.gitignore @@ -0,0 +1 @@ +.vercel diff --git a/client/vercel.json b/client/vercel.json deleted file mode 100644 index 3cc3e4ea..00000000 --- a/client/vercel.json +++ /dev/null @@ -1,7 +0,0 @@ -{ - "rewrites": [ - { "source": "/robots.txt", "destination": "/public/robots.txt" }, - { "source": "/sitemap.xml", "destination": "/public/sitemap.xml" }, - { "source": "/(.*)", "destination": "/" } - ] -} diff --git a/package-lock.json b/package-lock.json index fc10297f..beea970e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -7,6 +7,9 @@ "": { "name": "thefluentspanishhouse", "version": "1.0.0", + "dependencies": { + "prettier": "^3.4.2" + }, "devDependencies": { "concurrently": "^9.1.2" } @@ -182,6 +185,21 @@ "dev": true, "license": "MIT" }, + "node_modules/prettier": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.4.2.tgz", + "integrity": "sha512-e9MewbtFo+Fevyuxn/4rrcDAaq0IYxPGLvObpQjiZBMAzB9IGmzlnG9RZy3FFas+eBMu2vA0CszMeduow5dIuQ==", + "license": "MIT", + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" + } + }, "node_modules/require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", diff --git a/package.json b/package.json index ef4e20bf..c31faec1 100644 --- a/package.json +++ b/package.json @@ -21,5 +21,8 @@ "type": "GNU", "url": "https://www.gnu.org/licenses/gpl-3.0.html" } - ] + ], + "dependencies": { + "prettier": "^3.4.2" + } } diff --git a/server/.gitignore b/server/.gitignore new file mode 100644 index 00000000..e985853e --- /dev/null +++ b/server/.gitignore @@ -0,0 +1 @@ +.vercel diff --git a/server/index.ts b/server/index.ts index 0bce527a..0fed44c7 100644 --- a/server/index.ts +++ b/server/index.ts @@ -116,6 +116,6 @@ async function inicializeApp() { } } -inicializeApp().catch((error) => { +export default inicializeApp().catch((error) => { console.error("Error starting the server:", error); }); diff --git a/server/lib/resend/resend.ts b/server/lib/resend/resend.ts index 4074eba3..f95f1537 100644 --- a/server/lib/resend/resend.ts +++ b/server/lib/resend/resend.ts @@ -1,7 +1,7 @@ import dotenv from "dotenv"; dotenv.config(); //////////////////////////////////////////////////////////////// -import { Message, NoteType, SubscriberType } from "types/types"; +import { NoteType, SubscriberType } from "types/types"; import { Resend } from "resend"; // Configurar el cliente de Mandrill diff --git a/server/package-lock.json b/server/package-lock.json index a9ece140..d7202adc 100644 --- a/server/package-lock.json +++ b/server/package-lock.json @@ -28,7 +28,6 @@ "@types/express": "^4.17.21", "@types/jsonwebtoken": "^9.0.6", "@types/mailchimp__mailchimp_marketing": "^3.0.20", - "@types/mailchimp__mailchimp_transactional": "^1.0.10", "dotenv": "^16.4.5", "ts-node": "^10.9.2", "typescript": "^5.5.4" @@ -1220,16 +1219,6 @@ "dev": true, "license": "MIT" }, - "node_modules/@types/mailchimp__mailchimp_transactional": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/@types/mailchimp__mailchimp_transactional/-/mailchimp__mailchimp_transactional-1.0.10.tgz", - "integrity": "sha512-W8EcDqZspM4Mb6shET603kce4C11U29C3s0o8ZchWWcMyaz4MPa2nTyAVUSwnY0atuPii6Jpg3Gs6MuwupT3xw==", - "dev": true, - "license": "MIT", - "dependencies": { - "axios": "^1.6.7" - } - }, "node_modules/@types/mime": { "version": "1.3.5", "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.5.tgz", @@ -2848,20 +2837,6 @@ "node": ">= 0.8" } }, - "node_modules/fsevents": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", - "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", - "hasInstallScript": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": "^8.16.0 || ^10.6.0 || >=11.0.0" - } - }, "node_modules/function-bind": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", diff --git a/server/package.json b/server/package.json index 8df370d0..72fac11e 100644 --- a/server/package.json +++ b/server/package.json @@ -4,8 +4,8 @@ "type": "module", "main": "index.ts", "scripts": { - "start": "npm run build && node dist/index.js", - "dev": "node --loader ts-node/esm src/index.js", + "start": "node public/index.js", + "dev": "node --loader ts-node/esm index.js", "test": "npm run build && playwright test", "build": "tsc" }, diff --git a/server/public/index.js b/server/public/index.js new file mode 100644 index 00000000..cce11553 --- /dev/null +++ b/server/public/index.js @@ -0,0 +1,114 @@ +import dotenv from "dotenv"; +dotenv.config(); +/////////////////////////////////////////// +import { connectDB } from "./src/mongodb/mongodb.js"; +import express from "express"; +import cors from "cors"; +import { router } from "./routes/routes.js"; +import net from "net"; +// Extendemos el límite para que pueda almacenar imagenes en base64 +async function inicializeApp() { + const app = express(); + // Origenes permitidos + const allowedOrigins = ["https://thefluentspanishhouse.com", "http://localhost:5173", process.env.URL_WEB_TEST]; + // Configuración global de CORS + app.use(cors({ + origin: function (origin, callback) { + if (!origin || allowedOrigins.indexOf(origin) !== -1) { + callback(null, true); + } + else { + callback(new Error("Not allowed by CORS")); + } + }, + methods: ["GET", "POST", "PUT", "DELETE", "OPTIONS"], + allowedHeaders: ["Content-Type", "Authorization"], + credentials: true, + })); + app.use(express.json({ limit: "10mb" })); + app.use(express.urlencoded({ limit: "10mb", extended: true })); + app.use(express.json()); + app.use(express.text()); + // Conexión a la base de datos + try { + await connectDB(); + } + catch (error) { + throw new Error("Error connecting to the database"); + } + // Enpoint de bienvenida + app.get("/", (req, res) => { + res.send("Welcome to thefluentespnaishouse server"); + }); + // Rutas de la aplicación + app.use(router); + // Obtener la URL de los preview para hacer testing + app.get("/api/test", async (_req, res) => { + try { + const previewUrl = process.env.VERCEL_URL; + if (!previewUrl) + throw new Error("VERCEL_URL no está definida"); + res.send(previewUrl); + } + catch (error) { + res.status(500).json({ error: "Error en el servidor para obtener URL de preview" }); + } + }); + // Si es development y preview asignamos el puerto disponible a partir de 8080 + if (process.env.NODE_ENV !== "production") { + const PORT_BACKEND = 8080; + // Creamos una funcion flecha que devuelbe una promesa + const checkPort = (port) => { + return new Promise((resolve, reject) => { + // Creamos un servidor con net nativo de (NodeJS) + const server = net.createServer(); + // Intentar escuchar en el puerto + server.listen(port); + // Verificamos si el puerto esta en uso si lo esta devuleve false, si ocurre un error lo rechaza + server.once("error", (err) => { + if (err.code === "EADDRINUSE") { + resolve(false); + } + else { + reject(err); + } + }); + // Si el puerto esta libre lo cerramos y resolvemos la promesa + server.once("listening", () => { + server.close(); + resolve(true); + }); + }); + }; + const startServer = async (port) => { + // Encontramos un puerto libre de [8080, 8090] + if (port > 8090) { + console.log("No ports available"); + return; + } + // Encontra un puerto libre y se lo asigna al servidor, si no encuentra uno libre lo asigna al siguiente + const isPortFree = await checkPort(port); + if (isPortFree) { + app.listen(port, () => { + console.log(`Server running: http://localhost:${port}`); + }); + } + else { + console.log(`Port ${port} is in use, please choose another port.`); + startServer(port + 1); + } + }; + startServer(PORT_BACKEND).catch((error) => { + console.error("Error starting the server:", error); + }); + } + else { + const PORT_BACKEND = 8080; + app.listen(PORT_BACKEND, () => { + console.log(`Server runing: http://localhost:${PORT_BACKEND}`); + }); + } +} +export default inicializeApp().catch((error) => { + console.error("Error starting the server:", error); +}); diff --git a/server/public/lib/firebase/firebase-config.js b/server/public/lib/firebase/firebase-config.js new file mode 100644 index 00000000..6bc1c7fa --- /dev/null +++ b/server/public/lib/firebase/firebase-config.js @@ -0,0 +1,41 @@ +import dotenv from "dotenv"; +dotenv.config(); +//////////////////////////////////// +import admin from "firebase-admin"; +// Importa el archivo de credenciales de Firebase Admin +const serviceAccount = { + type: process.env.FIREBASE_TYPE, + project_id: process.env.FIREBASE_PROJECT_ID, + private_key_id: process.env.FIREBASE_PRIVATE_KEY_ID, + private_key: (process.env.FIREBASE_PRIVATE_KEY || "").replace(/\\n/g, "\n"), + client_email: process.env.FIREBASE_CLIENT_EMAIL, + client_id: process.env.FIREBASE_CLIENT_ID, + auth_uri: process.env.FIREBASE_AUTH_URI, + token_uri: process.env.FIREBASE_TOKEN_URI, + auth_provider_x509_cert_url: process.env.FIREBASE_AUTH_PROVIDER_X509_CERT_URL, + client_x509_cert_url: process.env.FIREBASE_CLIENT_X509_CERT_URL, +}; +// Verifica si la aplicación ya ha sido inicializada +if (!admin.apps.length) { + try { + admin.initializeApp({ + credential: admin.credential.cert(serviceAccount), + databaseURL: process.env.FIREBASE_DB, + }); + console.log("Firebase Admin initialized successfully"); + // Verifica si se puede conectar a Firebase Auth + admin + .auth() + .listUsers(1) + .then(() => { + console.log("Successfully connected to Firebase Auth"); + }) + .catch((error) => { + console.error("Error connecting to Firebase Auth:", error); + }); + } + catch (error) { + console.error("Error initializing Firebase Admin:", error); + } +} +export default admin; diff --git a/server/public/lib/firebase/firebase-db.js b/server/public/lib/firebase/firebase-db.js new file mode 100644 index 00000000..220817c4 --- /dev/null +++ b/server/public/lib/firebase/firebase-db.js @@ -0,0 +1,12 @@ +import admin from "firebase-admin"; +// Función para obtener UID por correo electrónico +export const getUidByEmail = async (email) => { + try { + const userRecord = await admin.auth().getUserByEmail(email); + return userRecord.uid; + } + catch (error) { + console.error("Error fetching user data:", error); + return null; + } +}; diff --git a/server/public/lib/mailchimp/mailchimp.js b/server/public/lib/mailchimp/mailchimp.js new file mode 100644 index 00000000..8f520078 --- /dev/null +++ b/server/public/lib/mailchimp/mailchimp.js @@ -0,0 +1,69 @@ +import dotenv from "dotenv"; +dotenv.config(); +///////////////////////////////////////////////////////////// +import mailchimp from "@mailchimp/mailchimp_marketing"; +import axios from "axios"; +// Id de la lista publica de contactos en Mailchimp +const listId = process.env.MAILCHIMP_LIST_ID; +// Id del grupo de intereses en Mailchimp +const groupId = process.env.MAILCHIMP_GROUP_ID; +// Api key de Mailchimp y prefijo del servidor +const mailchimpKey = process.env.MAILCHIMP_API_KEY; +const serverPrefix = mailchimpKey.split("-").pop(); +// Configuracion del cliente de la API de Mailchimp marketing +mailchimp.setConfig({ + apiKey: mailchimpKey, + server: serverPrefix, +}); +// Convierte los errores de Mailchimp a objeto de javascript +function mailchimpErrors(error) { + return JSON.parse(error.response.text); +} +async function addInterestToCategory(interestCategoryId, interestName) { + const url = `https://${serverPrefix}.api.mailchimp.com/3.0/lists/${listId}/interest-categories/${interestCategoryId}/interests`; + const data = { + name: interestName, + }; + try { + // El user puede ser cualquier string, no importa + const response = await axios.post(url, data, { + auth: { + username: "admin", + password: mailchimpKey, + }, + }); + return response.data; + } + catch (error) { + if (axios.isAxiosError(error) && error.response) { + throw { status: error.response.status, message: error.response.data }; + } + else { + throw { status: 500, message: "Unexpected error occurred" }; + } + } +} +async function deleteInterestCategory(interestCategoryId) { + const url = `https://${serverPrefix}.api.mailchimp.com/3.0/lists/${listId}/interest-categories/${groupId}/interests/${interestCategoryId}`; + try { + // El user puede ser cualquier string, no importa + const response = await axios.delete(url, { + auth: { + username: "admin", + password: mailchimpKey, + }, + }); + console.log(response.data); + return response; + } + catch (error) { + console.log(error); + if (axios.isAxiosError(error) && error.response) { + throw { status: error.response.status, message: error.response.data }; + } + else { + throw { status: 500, message: "Unexpected error occurred" }; + } + } +} +export { listId, mailchimpErrors, mailchimp, addInterestToCategory, deleteInterestCategory, groupId }; diff --git a/server/public/lib/resend/resend.js b/server/public/lib/resend/resend.js new file mode 100644 index 00000000..42dd7649 --- /dev/null +++ b/server/public/lib/resend/resend.js @@ -0,0 +1,58 @@ +import dotenv from "dotenv"; +dotenv.config(); +import { Resend } from "resend"; +// Configurar el cliente de Mandrill +const resend = new Resend(process.env.RESEND_API_KEY); +const admin = process.env.ADMIN_GMAIL; +// Enviar nota de contact us a el administrador +export async function submitNote(note) { + try { + return await resend.emails.send({ + from: "no-reply@thefluentspanishhouse.com", + to: [admin], + subject: note.subject, + html: ` +

The user ${note.username} with email ${note.email_user} sent you this message:


${note.note}

`, + headers: { + "Reply-To": note.email_user, + }, + }); + } + catch (error) { + console.error("Error sending email", error); + throw error; + } +} +// // Enviar email de nuevo estuidiante a el administrador +export async function submitEmailStudent(newstudent) { + try { + return await resend.emails.send({ + from: "no-reply@thefluentspanishhouse.com", + to: [admin], + subject: `New student on TheFluentSpanishHouse ${newstudent.name} ${newstudent.lastname}`, + html: `

The user ${newstudent.name} ${newstudent.lastname} wants to be a new student:


+

He wants to sign up for ${newstudent.class}

`, + }); + } + catch (error) { + console.error("Error sending email", error); + throw error; + } +} +// Enviar email de nuevo comentario a el administrador +export async function submitEmailComment(note, originUrl) { + try { + return await resend.emails.send({ + from: `no-reply@thefluentspanishhouse.com`, + to: [admin], + subject: note.subject, + html: `

The user ${note.username} ${note.email_user} says:


+

${note.note}


+

From the post: ${originUrl}


User: ${note.email_user}`, + }); + } + catch (error) { + console.error("Error sending email", error); + throw error; + } +} diff --git a/server/public/middelware/token-logs.js b/server/public/middelware/token-logs.js new file mode 100644 index 00000000..2a8b3d8e --- /dev/null +++ b/server/public/middelware/token-logs.js @@ -0,0 +1,27 @@ +import admin from "../lib/firebase/firebase-config.js"; +export async function log(req, res, next) { + // Registro de la solicitud + console.log(`[${new Date().toISOString()}] ${req.method} ${req.url}`); + next(); +} +export async function verifyIdToken(req, res, next) { + // Validación de autenticación + const authHeader = req.headers.authorization; + if (!authHeader || !authHeader.startsWith("Bearer ")) + return res.status(401).json({ message: "Unauthorized" }); + // Extrae el token de Firebase del encabezado de autorización, eliminadno el prefijo "Bearer " + const token = authHeader.split(" ")[1]; + // Validación de token autorizado proporcionado por el cliente + if (token === process.env.DEFAULT_TOKEN) + return next(); + try { + // Verifica el token de Firebase usando firebase-admin + const decodedToken = await admin.auth().verifyIdToken(token); + req.user = decodedToken; // Añade el usuario decodificado a la solicitud + next(); + } + catch (error) { + console.log("Error verifying Firebase ID token:", error); + return res.status(401).json({ message: "Unauthorized", error }); + } +} diff --git a/server/public/playwright.config.js b/server/public/playwright.config.js new file mode 100644 index 00000000..56101ffd --- /dev/null +++ b/server/public/playwright.config.js @@ -0,0 +1,41 @@ +import { defineConfig, devices } from "@playwright/test"; +import dotenv from "dotenv"; +dotenv.config(); +export default defineConfig({ + // Look for test files in the "tests" directory, relative to this configuration file. + testDir: "dist/tests", + // Run all tests in parallel. + fullyParallel: true, + // Fail the build on CI if you accidentally left test.only in the source code. + forbidOnly: !!process.env.CI, + // Retry on CI only. + retries: process.env.CI ? 2 : 0, + // Opt out of parallel tests on CI. + workers: process.env.CI ? 1 : undefined, + // Reporter to use + reporter: "html", + use: { + // Base URL to use in actions like `await page.goto('/')`. + baseURL: "http://127.0.0.1:8080", + // Collect trace when retrying the failed test. + trace: "on-first-retry", + // Add extra HTTP headers + extraHTTPHeaders: { + Authorization: `Bearer ${process.env.DEFAULT_TOKEN}`, + Aplication: "aplication-json", + }, + }, + // Configure projects for major browsers. + projects: [ + { + name: "chromium", + use: { ...devices["Desktop Chrome"] }, + }, + ], + // Run your local dev server before starting the tests. + webServer: { + command: "npm run start", + url: "http://127.0.0.1:8080", + reuseExistingServer: !process.env.CI, + }, +}); diff --git a/server/public/routes/comments.js b/server/public/routes/comments.js new file mode 100644 index 00000000..9edc2bd1 --- /dev/null +++ b/server/public/routes/comments.js @@ -0,0 +1,182 @@ +import { Router } from "express"; +import { modelComment } from "../src/mongodb/models.js"; +import { isValidObjectId, Types } from "mongoose"; +import { submitEmailComment } from "../lib/resend/resend.js"; +import { deleteCommentAndChildren } from "../utilities/delete-logic.js"; +import { handleServerError } from "../utilities/errorHandle.js"; +import { verifyIdToken, log } from "../middelware/token-logs.js"; +const router = Router(); +// <--------------- GET ---------------> +// Obtener todos los comentarios +router.get("/all", log, verifyIdToken, async (req, res) => { + // El id es el id del comentario padre (parent_id) + try { + // Buscar el comentario padre y obtener sus hijos + const comments = await modelComment.find().select("data"); + res.status(200).json(comments); + } + catch (error) { + res.status(500).json({ error: "Error al obtener los comentarios" }); + } +}); +// Obtener los hijos de un comentario +router.get("/children/:id", log, verifyIdToken, async (req, res) => { + // El id es el id del comentario padre (parent_id) + const { id } = req.params; + try { + // Buscar el comentario padre y obtener sus hijos + const comments = await modelComment.findById(id).populate("answers"); + if (!comments) + throw new Error("Comments not found"); + res.status(200).json(comments.answers); + } + catch (error) { + res.status(500).json({ error: "Error al obtener los comentarios" }); + } +}); +// Carga de comentarios al entrar en una publicación +router.get("/:id", log, verifyIdToken, async (req, res) => { + const { id } = req.params; + try { + const parentComments = await modelComment.find({ pattern_id: id }); + res.status(200).json(parentComments); + } + catch (error) { + res.status(500).json({ error: "Error al obtener los comentarios" }); + } +}); +// <--------------- POST ---------------> +// Agregar comentarios +router.post("/new", log, verifyIdToken, async (req, res) => { + const { newCommentData, originUrl } = req.body; + try { + // Crear un nuevo ObjectId para el nuevo comentario + newCommentData._id = new Types.ObjectId(); + // Crear un nuevo comentario hijo + const newComment = new modelComment(newCommentData); + newComment.save(); + const note = { + subject: `${newComment.owner.displayName} has sent a comment`, + username: newComment.owner.displayName, + note: newComment.data, + email_user: newComment.owner.email, + }; + // Avisamos al administrador de la web del nuevo comentario + await submitEmailComment(note, originUrl); + res.status(201).json(newComment); + } + catch (error) { + console.error("Error al añadir el comentario:", error); + res.status(500).json({ error: "Error al añadir el comentario" }); + } +}); +// Agregar comentarios hijos y hacer referencia al comentario padre +router.post("/children/:id", log, verifyIdToken, async (req, res) => { + const { id } = req.params; + const { newCommentData, originUrl } = req.body; + try { + // Crear un nuevo ObjectId para el nuevo comentario + newCommentData._id = new Types.ObjectId(); + // Crear un nuevo comentario hijo + const newComment = new modelComment(newCommentData); + newComment.save(); + // Avisamos al administrador de la web del nuevo comentario + const note = { + subject: `${newComment.owner.displayName} has sent a comment`, + username: newComment.owner.displayName, + note: newComment.data, + email_user: newComment.owner.email, + }; + await submitEmailComment(note, originUrl); + // Agregar el nuevo comentario al array de comentarios del comentario padre + const parentComment = await modelComment.findById(id); + if (!parentComment) { + res.status(404).send("Parent comment not found"); + throw new Error("Parent comment not found"); + } + parentComment.answers && parentComment.answers.push(newComment._id); + await parentComment.save(); + res.status(201).send(newComment); + } + catch (error) { + console.error("Error adding comment:", error); + res.status(500).send("Error adding comment:"); + } +}); +// <--------------- PUT ---------------> +// Actualizar likes de comentarios +router.put("/likes", log, verifyIdToken, async (req, res) => { + const { uid_user_firebase, _id, likes, originUrl } = req.body; + if (likes === undefined || likes === null || !uid_user_firebase || !_id) + return res.status(400).json({ error: "Los campos son requeridos" }); + try { + const comment = await modelComment.findById(_id); + if (!comment) + return res.status(404).json({ error: "Comentario no encontrado" }); + if (comment.likedBy && comment.likedBy.includes(uid_user_firebase)) { + const index = comment.likedBy.indexOf(uid_user_firebase); + comment.likedBy.splice(index, 1); + comment.likes -= 1; + } + else { + comment.likedBy && comment.likedBy.push(uid_user_firebase); + comment.likes += 1; + // Implemetaacion futura añadir correo al administrador por likes + // const note: NoteType = { + // subject: `${comment.owner.displayName} has received a like`, + // username: comment.owner.displayName, + // note: comment.data, + // email_user: comment.owner.email, + // }; + // console.log("note", note); + // Avisamos al administrador de la web del nuevo like + // const res = await submitEmailComment(note, originUrl); + console.log("res", res); + } + const updatedComment = await comment.save(); + res.status(200).json(updatedComment); + } + catch (error) { + handleServerError(res, error); + } +}); +// Editar comentarios +router.put("/edit/:id", log, verifyIdToken, async (req, res) => { + const { id } = req.params; + const { textEdit } = req.body; + if (!textEdit) + return res.status(400).json({ message: "Missing content" }); + try { + const comment = await modelComment.findById(id); + if (!comment) + return res.status(404).json({ message: "Comment not found" }); + comment.data = textEdit; + await comment.save(); + res.status(200).json(comment); + } + catch (error) { + handleServerError(res, error); + } +}); +// <--------------- DELETE ---------------> +// Eliminar comentarios +router.delete("/del/:id", log, verifyIdToken, async (req, res) => { + const { id } = req.params; + if (!isValidObjectId(id)) + return res.status(400).json({ message: "Invalid publication ID" }); + try { + const fatherComment = await modelComment.findById(id); + if (!fatherComment) + return res.status(404).json({ message: "Comment not found" }); + // Elimina los hijos del comentario + if (fatherComment.answers && fatherComment.answers.length > 0) + await deleteCommentAndChildren(fatherComment.answers); + // Elimina el comentario principal + await modelComment.findByIdAndDelete(id); + res.status(204).send(); + } + catch (error) { + handleServerError(res, error); + } +}); +export { router }; diff --git a/server/public/routes/firebase.js b/server/public/routes/firebase.js new file mode 100644 index 00000000..1a331d27 --- /dev/null +++ b/server/public/routes/firebase.js @@ -0,0 +1,23 @@ +import { Router } from "express"; +import { log, verifyIdToken } from "../middelware/token-logs.js"; +import { getUidByEmail } from "../lib/firebase/firebase-db.js"; +const router = Router(); +router.get("/user/:email", log, verifyIdToken, async (req, res) => { + // Obtener el UID de un usuario por su correo electrónico + const { email } = req.params; + if (!email) + return res.status(400).json({ message: "Missing required fields" }); + getUidByEmail(email) + .then((uid) => { + // Devolbemos el UID del usuario si existe + if (uid) + res.status(200).json({ uid }); + else + res.status(404).json({ message: "User not found" }); + }) + .catch((error) => { + console.error(error); + res.status(500).json({ message: "Unexpected error" }); + }); +}); +export { router }; diff --git a/server/public/routes/mailchimp.js b/server/public/routes/mailchimp.js new file mode 100644 index 00000000..1954cf02 --- /dev/null +++ b/server/public/routes/mailchimp.js @@ -0,0 +1,181 @@ +import { Router } from "express"; +import { listId, mailchimpErrors, mailchimp, addInterestToCategory, deleteInterestCategory, groupId, } from "../lib/mailchimp/mailchimp.js"; +import { log, verifyIdToken } from "../middelware/token-logs.js"; +import crypto from "crypto"; +import { isErrorAxios } from "../utilities/axios-utils.js"; +const router = Router(); +// <--------------- GET ---------------> +// Obtener todos los miembros de una lista +router.get("/getall/member", log, verifyIdToken, async (_req, res) => { + mailchimp.lists + .getListMembersInfo(listId) + .then((response) => { + res.status(200).json(response); + }) + .catch((error) => { + res.status(500).json(mailchimpErrors(error)); + }); +}); +// Obtener un miembro de la lista por email +router.get("/getone/member/:email", log, verifyIdToken, async (req, res) => { + const { email } = req.params; + const subscriberHash = crypto.createHash("md5").update(email).digest("hex"); + mailchimp.lists + .getListMember(listId, subscriberHash) + .then((response) => { + res.status(200).json(response); + }) + .catch((error) => { + const parsedError = mailchimpErrors(error); + res.status(parsedError.status).json(parsedError); + }); +}); +// Obtener todos los grupos de categorias +router.get("/groupscategory", log, verifyIdToken, async (req, res) => { + mailchimp.lists + .getListInterestCategories(listId) + .then((response) => { + res.status(200).json({ response }); + }) + .catch((error) => { + res.status(500).json(mailchimpErrors(error)); + }); +}); +// Obtener el grupo de los intereses especificos +router.get("/get/interests", log, verifyIdToken, async (req, res) => { + if (!groupId) + return res.status(400).send("groupID is required"); + mailchimp.lists + .listInterestCategoryInterests(listId, groupId) + .then((response) => { + res.status(200).json(response); + }) + .catch((error) => { + res.status(500).json(mailchimpErrors(error)); + }); +}); +// <--------------- POST ---------------> +// Añadir un miembro a la lista +router.post("/add/member", log, verifyIdToken, async (req, res) => { + const member = req.body; + mailchimp.lists + .addListMember(listId, member) + .then((response) => { + res.status(201).json(response); + }) + .catch((error) => { + res.status(500).json(mailchimpErrors(error)); + }); +}); +// Añadir varios miembros a la lista +router.post("/add/batchcontact", log, verifyIdToken, async (req, res) => { + const members = req.body; + // Usuario y etiquetas para añadir al usuario + mailchimp.lists + .batchListMembers(listId, { + members, + update_existing: true, + }) + .then((response) => { + res.status(201).json(response); + }) + .catch((error) => { + res.status(500).json(mailchimpErrors(error)); + }); +}); +// Añadir el nuevo interés +router.post("/add/interests", log, verifyIdToken, async (req, res) => { + const { name } = req.body; + if (!groupId || !name) + res.status(400).send("Category ID and interest name are required"); + try { + const response = await addInterestToCategory(groupId, name); + res.status(201).json(response); + } + catch (error) { + if (isErrorAxios(error)) { + res.status(error.status).json(error.message); + } + else { + res.status(500).json(error); + } + } +}); +// <--------------- PUT ---------------> +// Actualizar el estado de un miembro de la lista +router.put("/updatecontact/status/:email", log, verifyIdToken, async (req, res) => { + const { email } = req.params; + const { status } = req.body; + const subscriberHash = crypto.createHash("md5").update(email.toLowerCase()).digest("hex"); + mailchimp.lists + .updateListMember(listId, subscriberHash, { + status: status, + }) + .then((response) => res.status(200).json(response)) + .catch((error) => { + const parserError = mailchimpErrors(error); + res.status(parserError.status).json(parserError); + }); +}); +// Añadir el tag de un miembro de la lista +router.put("/updatecontact/tag/:email", log, verifyIdToken, async (req, res) => { + const { email } = req.params; + const { tag } = req.body; + const subscriberHash = crypto.createHash("md5").update(email).digest("hex"); + await mailchimp.lists + .updateListMemberTags(listId, subscriberHash, { + tags: [{ name: tag, status: "active" }], + }) + .then((response) => res.status(200).json(response)) + .catch((error) => { + const parserError = mailchimpErrors(error); + res.status(parserError.status).json(parserError); + }); +}); +// <--------------- DEL ---------------> +// Eliminar un miembro de la lista +router.delete("/del/member/:email", log, verifyIdToken, async (req, res) => { + // Obtenemos el email del usuario a eliminar desde la URL + const { email } = req.params; + const subscriberHash = crypto.createHash("md5").update(email).digest("hex"); + // Eliminamos al usuario de la lista, si no existe, se envía un error, si existe devuelbe null + await mailchimp.lists + .deleteListMember(listId, subscriberHash) + .then(() => res.status(204).end()) + .catch((error) => { + const parserError = mailchimpErrors(error); + res.status(parserError.status).json(parserError); + }); +}); +// Eliminar un tag de un miembro de la lista +router.delete("/del/tag/:email", log, verifyIdToken, async (req, res) => { + // Obtenemos el email del usuario a eliminar desde la URL + const { email } = req.params; + const { tag } = req.body; + const subscriberHash = crypto.createHash("md5").update(email.toLowerCase()).digest("hex"); + // Eliminamos el tag de la lista, si no existe, se envía un error, si existe devuelve null + await mailchimp.lists + .updateListMemberTags(listId, subscriberHash, { + tags: [{ name: tag, status: "inactive" }], + }) + .then(() => res.status(204).end()) + .catch((error) => { + const parserError = mailchimpErrors(error); + res.status(parserError.status).json(parserError); + }); +}); +// Eliminar un interes de una categoria +router.delete("/del/interests/:id", log, verifyIdToken, async (req, res) => { + const { id } = req.params; + try { + await deleteInterestCategory(id); + res.status(204).end(); + } + catch (error) { + if (isErrorAxios(error)) + res.status(error.status).json(error.message); + else + res.status(500).json(error); + } +}); +export { router }; diff --git a/server/public/routes/publications.js b/server/public/routes/publications.js new file mode 100644 index 00000000..7398ccf0 --- /dev/null +++ b/server/public/routes/publications.js @@ -0,0 +1,119 @@ +import { Router } from "express"; +import { modelComment, modelPublication } from "../src/mongodb/models.js"; +import { isValidObjectId, Types } from "mongoose"; +import { handleServerError } from "../utilities/errorHandle.js"; +import { log, verifyIdToken } from "../middelware/token-logs.js"; +const router = Router(); +// <--------------- GET ---------------> +// Obtener la última publicación +router.get("/last", log, verifyIdToken, async (_req, res) => { + try { + const lastPublication = await modelPublication.findOne().sort({ currentPage: -1 }).select("currentPage").exec(); + if (!lastPublication) + res.status(404).json({ message: "No posts available" }); + res.status(200).json(lastPublication); + } + catch (error) { + console.error("Error retrieving publication:", error); + handleServerError(res, error); + } +}); +// Obtener publicaciones segun la página +router.get("/page/:page", log, verifyIdToken, async (req, res) => { + const page = parseInt(req.params.page, 10); + if (isNaN(page)) + res.status(400).json({ message: "Invalid page number" }); + try { + const publications = await modelPublication.find({ currentPage: page }); + if (!publications) + res.status(404).json({ message: "Publication not found" }); + res.status(200).json(publications); + } + catch (error) { + console.error("Error retrieving publication:", error); + handleServerError(res, error); + } +}); +// Encontrar publicación por id del publication para entrar en la publicación selecionada +router.get("/:id", log, verifyIdToken, async (req, res) => { + const { id } = req.params; + try { + if (!isValidObjectId(id)) + res.status(400).json({ message: "Invalid publication ID" }); + const publication = await modelPublication.findById(id); + if (!publication) + res.status(404).json({ message: "Publication not found" }); + res.status(200).json(publication); + } + catch (error) { + console.error("Error retrieving publication:", error); + handleServerError(res, error); + } +}); +// <--------------- POST ---------------> +// Añadir nuevas publicaciones +router.post("/new", log, verifyIdToken, async (req, res) => { + const newPublication = req.body; + try { + // Validaciones + if (!newPublication.title || !newPublication.subtitle || !newPublication.content) + return res.status(400).json({ message: "Missing required fields" }); + if (newPublication.base64_img && !/^data:image\/[a-zA-Z]+;base64,/.test(newPublication.base64_img)) + return res.status(400).json({ message: "Invalid image format" }); + // Crear una nueva tarjeta de blog + const newCardBlog = new modelPublication({ + _id: new Types.ObjectId(), + title: newPublication.title, + subtitle: newPublication.subtitle, + content: newPublication.content, + base64_img: newPublication.base64_img, + currentPage: newPublication.currentPage, + }); + // Guardar el nuevo documento en la base de datos + await newCardBlog.save(); + res.status(201).json(newCardBlog); + } + catch (error) { + console.error("Error adding card blog:", error); + handleServerError(res, error); + } +}); +// <--------------- PUT ---------------> +// Editar publicaciones +router.put("/edit/:id", log, verifyIdToken, async (req, res) => { + const { id } = req.params; + const updatedFields = req.body; + try { + const publication = await modelPublication.findById(id); + if (!publication) + return res.status(404).json({ message: "Publication not found" }); + publication.title = updatedFields.title; + publication.subtitle = updatedFields.subtitle; + publication.content = updatedFields.content; + await publication.save(); + res.status(200).json(publication); + } + catch (error) { + console.error("Error updating publication:", error); + handleServerError(res, error); + } +}); +// <--------------- DEL ---------------> +// Eliminar publicaciones +router.delete("/del/:id", log, verifyIdToken, async (req, res) => { + const { id } = req.params; + if (!isValidObjectId(id)) + return res.status(400).json({ message: "Invalid publication ID" }); + try { + const result = await modelPublication.findByIdAndDelete(id); + await modelComment.deleteMany({ pattern_id: id }); + if (!result) + return res.status(404).json({ message: "Publication not found" }); + res.status(204).send(); + } + catch (error) { + console.error("Error deleting publication:", error); + handleServerError(res, error); + } +}); +export { router }; diff --git a/server/public/routes/resend.js b/server/public/routes/resend.js new file mode 100644 index 00000000..e47ac25e --- /dev/null +++ b/server/public/routes/resend.js @@ -0,0 +1,25 @@ +import { Router } from "express"; +import { submitEmailStudent, submitNote } from "../lib/resend/resend.js"; +import { log, verifyIdToken } from "../middelware/token-logs.js"; +const router = Router(); +// Enviamos un email con una nota de contact us +router.post("/note", log, verifyIdToken, async (req, res) => { + const newNote = req.body; + if (!newNote.email_user || !newNote.username || !newNote.subject || !newNote.note) + return res.status(400).send({ message: "Missing required fields" }); + submitNote(newNote) + .then((response) => res.status(201).json({ message: "Email sent successfully", response })) + .catch((error) => res.status(500).send({ message: "Error sending email", error })); +}); +// Avisamos de que se a suscrito un nuevo estudiante a una clase +router.post("/newstudent", log, verifyIdToken, async (req, res) => { + const newSubcriber = req.body; + console.log(newSubcriber); + if (!newSubcriber.email || !newSubcriber.name || !newSubcriber.lastname || !newSubcriber.class) + return res.status(400).send({ message: "Missing required fields" }); + console.log("Enviando email"); + submitEmailStudent(newSubcriber) + .then((response) => res.status(201).json({ message: "Email sent successfully", response })) + .catch((error) => res.status(500).send({ message: "Error sending email", error })); +}); +export { router }; diff --git a/server/public/routes/routes.js b/server/public/routes/routes.js new file mode 100644 index 00000000..557d47e8 --- /dev/null +++ b/server/public/routes/routes.js @@ -0,0 +1,33 @@ +import { Router } from "express"; +import { readdirSync } from "fs"; +import path from "path"; +import { fileURLToPath, pathToFileURL } from "url"; +const __dirname = path.dirname(fileURLToPath(import.meta.url)); +const PATH_ROUTER = __dirname; +const cleanExtension = (file) => file.split(".")[0]; +const router = Router(); +readdirSync(PATH_ROUTER).forEach((file) => { + const fileClean = cleanExtension(file); + const fileExtension = path.extname(file); + if (fileClean !== "routes" && + (fileExtension === ".js" || fileExtension === ".ts") && + !file.endsWith(".js.map") && + !file.endsWith(".ts.map")) { + const modulePath = path.join(PATH_ROUTER, `${fileClean}${fileExtension}`); + const moduleURL = pathToFileURL(modulePath).href; + import(moduleURL) + .then((module) => { + if (module.router) { + console.log(`Route ${fileClean} loaded successfully`); + router.use(`/${fileClean}`, module.router); + } + else { + console.error(`Error: No router found in module ${fileClean}`); + } + }) + .catch((err) => { + console.error(`Error loading route ${fileClean}:`, err); + }); + } +}); +export { router }; diff --git a/server/public/src/mongodb/models.js b/server/public/src/mongodb/models.js new file mode 100644 index 00000000..76ec099b --- /dev/null +++ b/server/public/src/mongodb/models.js @@ -0,0 +1,29 @@ +import { Schema, model } from "mongoose"; +const commentSchema = new Schema({ + _id: { type: Schema.Types.ObjectId, required: true }, + pattern_id: { type: String, required: true }, + owner: { + uid: { type: String, required: true }, + email: { type: String, required: true }, + displayName: { type: String, required: true }, + photoURL: { type: String, required: false }, + }, + data: { type: String, required: true }, + likes: { type: Number, required: true }, + likedBy: { type: [String], required: true, default: [] }, + answers: { + type: [{ type: Schema.Types.ObjectId, ref: "Comment" }], + required: true, + default: [], + }, +}); +export const modelComment = model("Comment", commentSchema, "comments"); +const publicationBlogSchema = new Schema({ + _id: { type: Schema.Types.ObjectId, required: true }, + title: { type: String, required: true }, + subtitle: { type: String, required: true }, + content: { type: String, required: true }, + base64_img: { type: String, required: false }, + currentPage: { type: Number, required: true }, +}); +export const modelPublication = model("PublicationBlog", publicationBlogSchema, "publications"); diff --git a/server/public/src/mongodb/mongodb.js b/server/public/src/mongodb/mongodb.js new file mode 100644 index 00000000..43e6ea6d --- /dev/null +++ b/server/public/src/mongodb/mongodb.js @@ -0,0 +1,15 @@ +import mongoose from "mongoose"; +// Conexion a la base de datos de MongoDBAtlas +export const connectDB = async () => { + if (mongoose.connection.readyState === 0) { + try { + const url = process.env.MONGO_DB_FLUENT; + if (!url) + throw new Error("Connection string not found"); + await mongoose.connect(url); + } + catch (error) { + console.error("Error al conectarse a la base de datos: ", error); + } + } +}; diff --git a/server/public/tests/comments.spec.js b/server/public/tests/comments.spec.js new file mode 100644 index 00000000..3632cef5 --- /dev/null +++ b/server/public/tests/comments.spec.js @@ -0,0 +1,103 @@ +import { test, expect } from "@playwright/test"; +import dotenv from "dotenv"; +import { Types } from "mongoose"; +dotenv.config(); +//####################### GET ####################### +test("/comments/all obtener todos los comentarios", async ({ page }) => { + const responseAll = await page.goto("/comments/all"); + expect(responseAll?.status()).toBe(200); + const allComments = await responseAll?.json(); + expect(allComments).toBeDefined(); +}); +test.describe.serial("Comment tests", () => { + let createdCommentId = ""; + let originUrl = "https://thefluentspanishhouse.com/publication/"; + const requestBody = { + uid_user_firebase: "user_firebase_123", + _id: "", + likes: 0, + originUrl, + }; + //####################### POST ####################### + test("/comments/new agregar nuevo comentario", async ({ page }) => { + const responseLast = await page.goto(`/publications/last`); + const response = await responseLast?.json(); + if (!response) { + console.log("No hay publicaciones disponibles saltando test de comentarios"); + expect(responseLast).not.toBeNull(); + return; + } + originUrl += response._id; + requestBody.originUrl = originUrl; + const newCommentData = { + _id: new Types.ObjectId(), + pattern_id: "66d4e739b705f46a0a395947", + owner: { + uid: "user123", + displayName: "John Doe", + email: "john.doe@example.com", + photoURL: "http://example.com/photo.jpg", + }, + data: "Este es un nuevo comentario", + likes: 0, + likedBy: [], + answers: [], + }; + // Cremos un comentario padre + const responseNewComment = await page.request.post("/comments/new", { + data: { newCommentData, originUrl }, + }); + expect(responseNewComment.status()).toBe(201); + // Almaecenamos la id del comentario creado + const fatherComment = await responseNewComment.json(); + createdCommentId = fatherComment._id; + requestBody._id = fatherComment._id; + }); + //####################### PUT ####################### + test("/comments/edit/:id editar nuevo comentario", async ({ page }) => { + // Cremos un comentario padre + const responsePost = await page.request.put(`/comments/edit/${createdCommentId}`, { + data: { textEdit: "Este es un comentario editado" }, + }); + expect(responsePost.status()).toBe(200); + }); + test("/comments/:id obtener por id", async ({ page }) => { + // Si existe algún comentario se obtiene el primer comentario + const responseID = await page.goto(`/comments/${createdCommentId}`); + expect(responseID?.status()).toBe(200); + const data = await responseID?.json(); + expect(data).toBeDefined(); + }); + test("comments/children/:id obtener por id los hijos", async ({ page }) => { + // Si existe algun comentario se obtiene el primer comentario + const responseChildren = await page.goto(`/comments/children/${createdCommentId}`); + expect(responseChildren?.status()).toBe(200); + const data = await responseChildren?.json(); + expect(data).toBeDefined(); + }); + test("/likes agregar like", async ({ page }) => { + const responseLike = await page.request.put(`/comments/likes`, { + data: requestBody, + }); + expect(responseLike.status()).toBe(200); + const updatedComment = await responseLike.json(); + expect(updatedComment.likes).toBe(1); + }); + test("/likes eliminar like", async ({ page }) => { + const newRequestBody = { ...requestBody }; + newRequestBody.likes = 1; + const responseDislike = await page.request.put(`/comments/likes`, { + data: newRequestBody, + }); + expect(responseDislike.status()).toBe(200); + const newUpdatedComment = await responseDislike.json(); + expect(newUpdatedComment.likes).toBe(0); + }); + //####################### DELETE ####################### + test("/comments/del/:id eliminar comentario", async ({ page }) => { + if (!createdCommentId) + throw new Error("No comment ID available to delete"); + const responseDel = await page.request.delete(`/comments/del/${createdCommentId}`); + expect(responseDel.status()).toBe(204); + }); +}); diff --git a/server/public/tests/index.spec.js b/server/public/tests/index.spec.js new file mode 100644 index 00000000..7a5cc460 --- /dev/null +++ b/server/public/tests/index.spec.js @@ -0,0 +1,8 @@ +import { test, expect } from "@playwright/test"; +test("La ruta raíz devuelve el mensaje esperado", async ({ page }) => { + // Navegar a la página principal + await page.goto("/"); + // Verificar que el contenido de la página sea el esperado + const content = await page.textContent("body"); + expect(content).toBe("Welcome to thefluentespnaishouse server"); +}); diff --git a/server/public/tests/mailchimp.spec.js b/server/public/tests/mailchimp.spec.js new file mode 100644 index 00000000..3def81a9 --- /dev/null +++ b/server/public/tests/mailchimp.spec.js @@ -0,0 +1,141 @@ +import { test, expect } from "@playwright/test"; +//####################### GET ####################### +test("/getall/member obtener todos los miembros de una lista", async ({ page }) => { + const responseMembersList = await page.goto(`mailchimp/getall/member`); + expect(responseMembersList?.status()).toBe(200); + const data = await responseMembersList?.json(); + expect(data).toBeDefined(); +}); +test("/groupscategory obtener todos los grupos de categorias", async ({ page }) => { + const responseMember = await page.goto(`/mailchimp/groupscategory`); + expect(responseMember?.status()).toBe(200); + const data = await responseMember?.json(); + expect(data).toBeDefined(); +}); +test.describe.serial("Mailchimp tests", () => { + const testEmail = "testing@gmail.com"; + // Intereses del grupo actual (Pensado para un unico grupo) + let group; + test("/get/interests obtener los intereses de un grupo de categorias", async ({ page }) => { + const responseMember = await page.goto(`/mailchimp/get/interests`); + expect(responseMember?.status()).toBe(200); + const data = await responseMember?.json(); + expect(data).toBeDefined(); + group = data; + }); + //####################### POST ####################### + const member = { + email_address: `${testEmail}`, + status: "transactional", + email_type: "html", + merge_fields: { + FNAME: "testing", + LNAME: "testinglastname", + }, + interests: {}, + tags: [], + status_if_new: "transactional", + }; + test("/add/member añadir un miembro", async ({ page }) => { + let responseNewMember = await page.request.post(`mailchimp/add/member`, { + data: member, + }); + const data = await responseNewMember?.json(); + if (data.title === "Member Exists") { + // Si el miembro ya existe, no se puede añadir + expect(data.title).toContain(`Member Exists`); + } + else { + expect(responseNewMember?.status()).toBe(201); + expect(data).toBeDefined(); + } + }); + //####################### GET ####################### + test("/getone/member/:email obtener un miembro", async ({ page }) => { + const responseMember = await page.goto(`/mailchimp/getone/member/${testEmail}`); + expect(responseMember?.status()).toBe(200); + const data = await responseMember?.json(); + expect(data).toBeDefined(); + }); + //####################### POST ####################### + test("/add/batchcontact añadir varios miembros a la lista (opcional)", async ({ page }) => { + const copyMember1 = { ...member }; + copyMember1.email_address = "testing1@gmail.com"; + const copyMember2 = { ...member }; + copyMember2.email_address = "testing2@gmail.com"; + const members = [copyMember1, copyMember2]; + const responseNewMembers = await page.request.post(`/mailchimp/add/batchcontact`, { + data: members, + }); + expect(responseNewMembers?.status()).toBe(201); + const data = await responseNewMembers?.json(); + expect(data).toBeDefined(); + }); + // Id del interes creado en el test + let idInterest = ""; + test("/add/interests añadir intereses", async ({ page }) => { + const name = "testing"; + const responseNewInterests = await page.request.post(`mailchimp/add/interests`, { + data: { name }, + }); + const data = await responseNewInterests?.json(); + if (data.detail && data.detail == `Cannot add "${name}" because it already exists on the list.`) { + // Si el interés ya existe, no se puede añadir asique lo buscamos + const responseAllInterests = await page.request.get(`/mailchimp/get/interests`); + const dataInterest = await responseAllInterests?.json(); + const element = dataInterest.interests.find((element) => element.name == name); + if (element) + idInterest = element.id; + expect(responseAllInterests?.status()).toBe(200); + } + else { + expect(responseNewInterests?.status()).toBe(201); + const data = await responseNewInterests?.json(); + idInterest = data.id; + } + }); + //####################### PUT ####################### + test("/updatecontact/status/:email editar estado de un usuario por email", async ({ page }) => { + const status = "transactional"; + const responseStatus = await page.request.put(`/mailchimp/updatecontact/status/${testEmail}`, { + data: { status }, + }); + expect(responseStatus?.status()).toBe(200); + const data = await responseStatus?.json(); + expect(data).toBeDefined(); + }); + test("/updatecontact/tag/:email editar tag de un usuario por email", async ({ page }) => { + const tag = "GROUP_CLASS"; + const responseTag = await page.request.put(`/mailchimp/updatecontact/tag/${testEmail}`, { + data: { tag }, + }); + expect(responseTag?.status()).toBe(200); + const data = await responseTag?.json(); + expect(data).toBeDefined(); + }); + //####################### DELETE ####################### + test("/del/interests/:id eliminar intereses de una categoria", async ({ page }) => { + // Verificamos si existe el interes de testing y obtenemos su id + if (idInterest == "") + console.warn("Dont exist the interest testing"); + else { + const responseInterests = await page.request.delete(`/mailchimp/del/interests/${idInterest}`); + expect(responseInterests?.status()).toBe(204); + } + }); + test("/del/tag/:email eliminar tag de un miembro", async ({ page }) => { + const tag = "GROUP_CLASS"; + const responseDelTag = await page.request.delete(`/mailchimp/del/tag/${testEmail}`, { + data: { tag }, + }); + expect(responseDelTag?.status()).toBe(204); + }); + test("/del/user/:email eliminar usuarios", async ({ page }) => { + const responseUser = await page.request.delete(`/mailchimp/del/member/${testEmail}`); + expect(responseUser?.status()).toBe(204); + const responseUser1 = await page.request.delete(`/mailchimp/del/member/testing1@gmail.com`); + expect(responseUser1?.status()).toBe(204); + const responseUser2 = await page.request.delete(`/mailchimp/del/member/testing2@gmail.com`); + expect(responseUser2?.status()).toBe(204); + }); +}); diff --git a/server/public/tests/mandril.spec.js b/server/public/tests/mandril.spec.js new file mode 100644 index 00000000..84982408 --- /dev/null +++ b/server/public/tests/mandril.spec.js @@ -0,0 +1,35 @@ +import { test, expect } from "@playwright/test"; +//####################### POST ####################### +test("/note enviar email de la nota de contact", async ({ page }) => { + let newNote = { + email_user: "carlosrvasquezsanchez@gmail.com", + username: "Test", + subject: "Test", + note: "Test", + }; + // Enviamos un email de la nota de contacto a admin + const responseNewComment = await page.request.post("/mandrill/note", { + data: newNote, + }); + expect(responseNewComment.status()).toBe(201); + const data = await responseNewComment.json(); + expect(data).toBeDefined(); +}); +test("/newstudent enviar email de neuvo estudiante", async ({ page }) => { + let newSubcriber = { + email: "carlosrvasquezsanchez@gmail.com", + name: "Test", + lastname: "Test", + class: "FREE_CLASS", + consentEmails: true, + acceptTerms: true, + acceptPrivacy: true, + }; + // Enviamos un email de la nota de nuevo estudiante a admin + const responseNewComment = await page.request.post("/mandrill/newstudent", { + data: newSubcriber, + }); + expect(responseNewComment.status()).toBe(201); + const data = await responseNewComment.json(); + expect(data).toBeDefined(); +}); diff --git a/server/public/tests/publications.spec.js b/server/public/tests/publications.spec.js new file mode 100644 index 00000000..b0e9f4b4 --- /dev/null +++ b/server/public/tests/publications.spec.js @@ -0,0 +1,62 @@ +import { test, expect } from "@playwright/test"; +import { Types } from "mongoose"; +//####################### GET ####################### +test("/last obtener la última publicacion", async ({ page }) => { + const responseLast = await page.goto(`/publications/last`); + expect(responseLast?.status()).toBe(200); + const data = await responseLast?.json(); + expect(data).toBeDefined(); +}); +test("/page/:page obtener publicaciones por página", async ({ page }) => { + const responseLast = await page.goto(`/publications/page/1`); + expect(responseLast?.status()).toBe(200); + const data = await responseLast?.json(); + expect(data).toBeDefined(); +}); +test.describe.serial("Comment tests", () => { + //####################### POST ####################### + let id_publication; + const newPublication = { + _id: new Types.ObjectId(), + title: "Test", + subtitle: "Test", + content: "Test", + base64_img: "", + currentPage: 1, + }; + test("/new añadir publicaciones", async ({ page }) => { + const responseNewPublication = await page.request.post(`/publications/new`, { + data: newPublication, + }); + expect(responseNewPublication?.status()).toBe(201); + const data = await responseNewPublication?.json(); + expect(data).toBeDefined(); + // La .id cambia en el servidor, por lo que se debe cambiar + id_publication = data._id; + }); + //####################### GET ####################### + test("/:id obtener publicaciones por id", async ({ page }) => { + const responseID = await page.goto(`/publications/${id_publication}`); + expect(responseID?.status()).toBe(200); + const data = await responseID?.json(); + expect(data).toBeDefined(); + }); + //####################### PUT ####################### + test("/edit/:id editar publicaciones por id", async ({ page }) => { + const updateFields = { + ...newPublication, + }; + updateFields.title = "Title edited"; + const responseID = await page.request.put(`/publications/edit/${id_publication}`, { + data: updateFields, + }); + expect(responseID?.status()).toBe(200); + const data = await responseID?.json(); + expect(data).toBeDefined(); + }); + //####################### DELETE ####################### + test("/del/:id eliminar publicaciones por id", async ({ page }) => { + const responseID = await page.request.delete(`/publications/del/${id_publication}`); + expect(responseID?.status()).toBe(204); + }); +}); diff --git a/server/public/utilities/axios-utils.js b/server/public/utilities/axios-utils.js new file mode 100644 index 00000000..3ae5055b --- /dev/null +++ b/server/public/utilities/axios-utils.js @@ -0,0 +1,3 @@ +export function isErrorAxios(error) { + return error.status !== undefined && error.message !== undefined; +} diff --git a/server/public/utilities/delete-logic.js b/server/public/utilities/delete-logic.js new file mode 100644 index 00000000..a7a48eca --- /dev/null +++ b/server/public/utilities/delete-logic.js @@ -0,0 +1,13 @@ +import { modelComment } from "../src/mongodb/models.js"; +export const deleteCommentAndChildren = async (commentId) => { + // Obtener el comentario con sus hijos + const comment = await modelComment.findById(commentId).populate("answers"); + if (!comment) + throw new Error("Comment not found"); + // Eliminar recursivamente todos los hijos + if (comment.answers) + for (const childCommentId of comment.answers) + await deleteCommentAndChildren(childCommentId); + // Eliminar el comentario padre + await modelComment.findByIdAndDelete(commentId); +}; diff --git a/server/public/utilities/errorHandle.js b/server/public/utilities/errorHandle.js new file mode 100644 index 00000000..0cdd732f --- /dev/null +++ b/server/public/utilities/errorHandle.js @@ -0,0 +1,10 @@ +const handleHTTPErrorLog = (res, error) => { + res.status(500).send({ error: error.message }); +}; +export function handleServerError(res, error) { + res.status(500).json({ + message: "Server error", + error: error instanceof Error ? error.message : String(error), + }); +} +export { handleHTTPErrorLog }; diff --git a/server/routes/routes.ts b/server/routes/routes.ts index 428d2921..cdca8644 100644 --- a/server/routes/routes.ts +++ b/server/routes/routes.ts @@ -12,7 +12,12 @@ readdirSync(PATH_ROUTER).forEach((file) => { const fileClean = cleanExtension(file); const fileExtension = path.extname(file); - if (fileClean !== "routes" && (fileExtension === ".js" || fileExtension === ".ts") && !file.endsWith(".map")) { + if ( + fileClean !== "routes" && + (fileExtension === ".js" || fileExtension === ".ts") && + !file.endsWith(".js.map") && + !file.endsWith(".ts.map") + ) { const modulePath = path.join(PATH_ROUTER, `${fileClean}${fileExtension}`); const moduleURL = pathToFileURL(modulePath).href; diff --git a/server/tsconfig.json b/server/tsconfig.json index 61df3bf8..d76e1367 100644 --- a/server/tsconfig.json +++ b/server/tsconfig.json @@ -6,7 +6,7 @@ "rootDir": "./" /* Specify the root folder within your source files. */, "moduleResolution": "node" /* Specify how TypeScript looks up a file from a given module specifier. */, "baseUrl": "./" /* Specify the base directory to resolve non-relative module names. */, - "outDir": "dist" /* Specify an output folder for all emitted files. */, + "outDir": "public" /* Specify an output folder for all emitted files. */, "paths": { "types/*": ["types/*"] } /* Specify a set of entries that re-map imports to additional lookup locations. */, @@ -20,5 +20,5 @@ "esm": true } /* Enable all strict type-checking options. */, "include": ["./**/*.ts", "types"], - "exclude": ["node_modules", "dist"] + "exclude": ["node_modules", "public"] } diff --git a/server/vercel.json b/server/vercel.json deleted file mode 100644 index 957d5ead..00000000 --- a/server/vercel.json +++ /dev/null @@ -1,16 +0,0 @@ -{ - "version": 2, - "builds": [ - { - "src": "package.json", - "use": "@vercel/node", - "config": { "distDir": "dist" } - } - ], - "routes": [ - { - "src": "/(.*)", - "dest": "/dist/index.js" - } - ] -} diff --git a/vercel.json b/vercel.json new file mode 100644 index 00000000..ebcfa3c4 --- /dev/null +++ b/vercel.json @@ -0,0 +1,38 @@ +{ + "version": 2, + "projects": [ + { + "name": "thefluenspanishh-client", + "root": "client", + "builds": [ + { + "src": "client/package.json", + "use": "@vercel/static-build", + "config": { + "distDir": "client/dist" + } + } + ], + "routes": [{ "src": "/(.*)", "dest": "client/dist/$1" }], + "rewrites": [ + { "source": "/robots.txt", "destination": "/public/robots.txt" }, + { "source": "/sitemap.xml", "destination": "/public/sitemap.xml" }, + { "source": "/(.*)", "destination": "/client/$1" } + ] + }, + { + "name": "thefluenspanishhouse-server", + "root": "server", + "builds": [ + { + "src": "server/public/index.js", + "use": "@vercel/node", + "config": { + "outputDirectory": "server/public" + } + } + ], + "routes": [{ "src": "/(.*)", "dest": "server/public/index.js" }] + } + ] +}