From 02ff66cb8f66d6b01c4c81173c4787c97eb5a910 Mon Sep 17 00:00:00 2001
From: Wini_Fy
Nếu bạn đang tìm kiếm học bổng để giảm chi phí du học, đây là 5 chương trình bạn không nên bỏ qua. Mỗi chương trình đều có tiêu chí xét tuyển, mức hỗ trợ và thời hạn đăng ký khác nhau.
", + "category": ["Study Abroad"], + "tags": ["StudentVisa", "Scholarship"], + "author": "Admin", + "status": "published", + "publishedAt": "20 March 2025", + "isFeatured": false, + "featuredImage": "/uploads/blog/scholarship-programs-main.jpg", + "galleryImages": [], + "commentsCount": 0 + }, + { + "title": "10 Travel Safety Tips You Should Know Before Flying", + "slug": "10-travel-safety-tips-before-flying", + "excerpt": "Những lưu ý quan trọng để đảm bảo an toàn cho chuyến bay và hành trình của bạn.", + "content": "An toàn luôn là ưu tiên hàng đầu khi đi du lịch. Dưới đây là 10 tips giúp bạn yên tâm hơn trên mọi chuyến đi, từ việc chuẩn bị giấy tờ, bảo hiểm đến cách bảo vệ tài sản cá nhân.
", + "category": ["Travel Tips"], + "tags": ["TravelSafety"], + "author": "Admin", + "status": "published", + "publishedAt": "05 April 2025", + "isFeatured": false, + "featuredImage": "/uploads/blog/travel-safety-main.jpg", + "galleryImages": [ + "/uploads/blog/travel-safety-1.jpg" + ], + "commentsCount": 1 + } + ], + "recentPosts": [ + { + "title": "Ultimate Guide To Getting A Work Visa In Canada", + "slug": "ultimate-guide-work-visa-canada", + "thumbnail": "/uploads/blog/work-visa-canada-main.jpg", + "publishedAt": "11 March 2025" + }, + { + "title": "Top 5 Scholarship Programs For International Students", + "slug": "top-5-scholarship-programs-international-students", + "thumbnail": "/uploads/blog/scholarship-programs-main.jpg", + "publishedAt": "20 March 2025" + }, + { + "title": "10 Travel Safety Tips You Should Know Before Flying", + "slug": "10-travel-safety-tips-before-flying", + "thumbnail": "/uploads/blog/travel-safety-main.jpg", + "publishedAt": "05 April 2025" + } + ], + "comments": [ + { + "postSlug": "ultimate-guide-work-visa-canada", + "authorName": "Frank Flores", + "authorAvatar": "/assets/img/inner-page/news-details/comment-1.png", + "content": "Bài viết rất hữu ích, cảm ơn bạn đã chia sẻ!", + "createdAt": "February 10, 2024", + "status": "approved", + "parentAuthorName": null + }, + { + "postSlug": "ultimate-guide-work-visa-canada", + "authorName": "Courtney Henry", + "authorAvatar": "/assets/img/inner-page/news-details/comment-2.png", + "content": "Mình đã làm theo hướng dẫn và hồ sơ được duyệt nhanh hơn hẳn.", + "createdAt": "February 12, 2024", + "status": "approved", + "parentAuthorName": "Frank Flores" + }, + { + "postSlug": "10-travel-safety-tips-before-flying", + "authorName": "Jenny Wilson", + "authorAvatar": "/assets/img/inner-page/news-details/comment-3.png", + "content": "Những tip này rất thực tế, đặc biệt là phần chuẩn bị bảo hiểm!", + "createdAt": "March 02, 2024", + "status": "approved", + "parentAuthorName": null + } + ] +} diff --git a/models/blog.js b/models/blog.js new file mode 100644 index 0000000..39aea69 --- /dev/null +++ b/models/blog.js @@ -0,0 +1,135 @@ +const mongoose = require('mongoose'); + +const blogSchema = new mongoose.Schema({ + // Basic blog information + title: { + type: String, + required: true, + trim: true + }, + slug: { + type: String, + required: true, + unique: true, + trim: true + }, + excerpt: { + type: String, + required: true, + maxlength: 500 + }, + content: { + type: mongoose.Schema.Types.Mixed, // Có thể là string HTML hoặc JSON EditorJS + required: true + }, + + // Media + featuredImage: { + type: String, + default: '' + }, + galleryImages: [{ + type: String + }], // Mảng URL ảnh cho gallery (details-2, details-3) + + // Author and publishing + author: { + type: String, + default: 'Admin' + }, + publishedAt: { + type: String, // Format: "11 March 2025" + default: function() { + return new Date().toLocaleDateString('en-GB', { + day: 'numeric', + month: 'long', + year: 'numeric' + }); + } + }, + + // Categorization (simple strings, no references) + category: [{ + type: String // ["Visa", "Travel", ...] - Một bài có thể thuộc nhiều category + }], + tags: [{ + type: String // ["WorkVisa", "StudentVisa", ...] + }], + + // Status and features + status: { + type: String, + enum: ['draft', 'published'], + default: 'published' + }, + isFeatured: { + type: Boolean, + default: false + }, + + // Comments count (có thể fake trước) + commentsCount: { + type: Number, + default: 0 + } +}, { + timestamps: true +}); + +// Indexes +blogSchema.index({ slug: 1 }); +blogSchema.index({ status: 1, createdAt: -1 }); +blogSchema.index({ category: 1, status: 1 }); +blogSchema.index({ isFeatured: 1, status: 1 }); +blogSchema.index({ tags: 1, status: 1 }); + +// Remove __v from JSON output +blogSchema.set('toJSON', { + transform: function(doc, ret) { + delete ret.__v; + return ret; + } +}); + +// Pre-save middleware +blogSchema.pre('save', function(next) { + // Auto-generate slug if not provided + if (!this.slug && this.title) { + this.slug = this.title + .toLowerCase() + .replace(/[^a-z0-9\s-]/g, '') + .replace(/\s+/g, '-') + .replace(/-+/g, '-') + .trim('-'); + } + + next(); +}); + +// Static methods +blogSchema.statics.getPublished = function() { + return this.find({ status: 'published' }).sort({ createdAt: -1 }); +}; + +blogSchema.statics.getFeatured = function() { + return this.find({ + status: 'published', + isFeatured: true + }).sort({ createdAt: -1 }); +}; + +blogSchema.statics.getByCategory = function(category) { + return this.find({ + status: 'published', + category: { $in: [category] } // Tìm trong array categories + }).sort({ createdAt: -1 }); +}; + +blogSchema.statics.getByTag = function(tag) { + return this.find({ + status: 'published', + tags: tag + }).sort({ createdAt: -1 }); +}; + +module.exports = mongoose.model('Blog', blogSchema); \ No newline at end of file diff --git a/models/blogCategory.js b/models/blogCategory.js new file mode 100644 index 0000000..7e342f1 --- /dev/null +++ b/models/blogCategory.js @@ -0,0 +1,76 @@ +const mongoose = require('mongoose'); + +const blogCategorySchema = new mongoose.Schema({ + name: { + type: String, + required: true, + trim: true, + unique: true // "Permanent Residency (PR)" + }, + slug: { + type: String, + required: true, + unique: true, + trim: true // "permanent-residency" + }, + postCount: { + type: Number, + default: 0 // "(04)" + }, + description: { + type: String, + default: '' + }, + isActive: { + type: Boolean, + default: true + } +}, { + timestamps: true +}); + +// Indexes +blogCategorySchema.index({ slug: 1 }); +blogCategorySchema.index({ isActive: 1, name: 1 }); + +// Remove __v from JSON output +blogCategorySchema.set('toJSON', { + transform: function(doc, ret) { + delete ret.__v; + return ret; + } +}); + +// Pre-save middleware +blogCategorySchema.pre('save', function(next) { + // Auto-generate slug if not provided + if (!this.slug && this.name) { + this.slug = this.name + .toLowerCase() + .replace(/[^a-z0-9\s-]/g, '') + .replace(/\s+/g, '-') + .replace(/-+/g, '-') + .trim('-'); + } + + next(); +}); + +// Static methods +blogCategorySchema.statics.getActive = function() { + return this.find({ isActive: true }).sort({ name: 1 }); +}; + +// Method to update post count +blogCategorySchema.methods.updatePostCount = async function() { + const Blog = require('./blog'); + const count = await Blog.countDocuments({ + category: { $in: [this.name] }, // Tìm trong array categories + status: 'published' + }); + this.postCount = count; + await this.save(); + return count; +}; + +module.exports = mongoose.model('BlogCategory', blogCategorySchema); \ No newline at end of file diff --git a/models/blogComment.js b/models/blogComment.js new file mode 100644 index 0000000..80e24b1 --- /dev/null +++ b/models/blogComment.js @@ -0,0 +1,84 @@ +const mongoose = require('mongoose'); + +const blogCommentSchema = new mongoose.Schema({ + postId: { + type: mongoose.Schema.Types.ObjectId, + ref: 'Blog', + required: true + }, + authorName: { + type: String, + required: true, + trim: true // "Frank Flores" + }, + authorAvatar: { + type: String, + default: '' // "/assets/img/inner-page/news-details/comment-1.png" + }, + content: { + type: String, + required: true, + trim: true + }, + parentId: { + type: mongoose.Schema.Types.ObjectId, + ref: 'BlogComment', + default: null // Cho threaded comments (reply) + }, + status: { + type: String, + enum: ['pending', 'approved', 'rejected'], + default: 'pending' + } +}, { + timestamps: true // Vẫn giữ timestamps cho admin quản lý +}); + +// Indexes +blogCommentSchema.index({ postId: 1, status: 1, createdAt: -1 }); +blogCommentSchema.index({ parentId: 1 }); +blogCommentSchema.index({ status: 1 }); + +// Remove __v from JSON output +blogCommentSchema.set('toJSON', { + transform: function(doc, ret) { + delete ret.__v; + return ret; + } +}); + +// Static methods +blogCommentSchema.statics.getApprovedByPost = function(postId) { + return this.find({ + postId: postId, + status: 'approved', + parentId: null // Chỉ lấy comments gốc, không lấy replies + }).sort({ createdAt: -1 }); +}; + +blogCommentSchema.statics.getReplies = function(parentId) { + return this.find({ + parentId: parentId, + status: 'approved' + }).sort({ createdAt: 1 }); // Replies sắp xếp theo thời gian tăng dần +}; + +blogCommentSchema.statics.getByStatus = function(status) { + return this.find({ status: status }) + .populate('postId', 'title slug') + .sort({ createdAt: -1 }); +}; + +// Method to approve comment +blogCommentSchema.methods.approve = function() { + this.status = 'approved'; + return this.save(); +}; + +// Method to reject comment +blogCommentSchema.methods.reject = function() { + this.status = 'rejected'; + return this.save(); +}; + +module.exports = mongoose.model('BlogComment', blogCommentSchema); \ No newline at end of file diff --git a/models/blogTag.js b/models/blogTag.js new file mode 100644 index 0000000..8db8d9c --- /dev/null +++ b/models/blogTag.js @@ -0,0 +1,78 @@ +const mongoose = require('mongoose'); + +const blogTagSchema = new mongoose.Schema({ + name: { + type: String, + required: true, + trim: true, + unique: true // "WorkVisa" + }, + slug: { + type: String, + required: true, + unique: true, + trim: true // "work-visa" + }, + postCount: { + type: Number, + default: 0 + }, + isActive: { + type: Boolean, + default: true + } +}, { + timestamps: true +}); + +// Indexes +blogTagSchema.index({ slug: 1 }); +blogTagSchema.index({ isActive: 1, name: 1 }); + +// Remove __v from JSON output +blogTagSchema.set('toJSON', { + transform: function(doc, ret) { + delete ret.__v; + return ret; + } +}); + +// Pre-save middleware +blogTagSchema.pre('save', function(next) { + // Auto-generate slug if not provided + if (!this.slug && this.name) { + this.slug = this.name + .toLowerCase() + .replace(/[^a-z0-9\s-]/g, '') + .replace(/\s+/g, '-') + .replace(/-+/g, '-') + .trim('-'); + } + + next(); +}); + +// Static methods +blogTagSchema.statics.getActive = function() { + return this.find({ isActive: true }).sort({ name: 1 }); +}; + +blogTagSchema.statics.getPopular = function(limit = 10) { + return this.find({ isActive: true }) + .sort({ postCount: -1, name: 1 }) + .limit(limit); +}; + +// Method to update post count +blogTagSchema.methods.updatePostCount = async function() { + const Blog = require('./blog'); + const count = await Blog.countDocuments({ + tags: { $in: [this.name] }, // Tìm trong array tags + status: 'published' + }); + this.postCount = count; + await this.save(); + return count; +}; + +module.exports = mongoose.model('BlogTag', blogTagSchema); \ No newline at end of file diff --git a/models/recentPost.js b/models/recentPost.js new file mode 100644 index 0000000..6c43272 --- /dev/null +++ b/models/recentPost.js @@ -0,0 +1,79 @@ +const mongoose = require('mongoose'); + +// Recent Post model - có thể là view hoặc collection riêng để optimize performance +const recentPostSchema = new mongoose.Schema({ + title: { + type: String, + required: true, + trim: true + }, + slug: { + type: String, + required: true, + trim: true + }, + thumbnail: { + type: String, + default: '' // Ảnh nhỏ ở sidebar + }, + publishedAt: { + type: String, // "March 26, 2025" + required: true + }, + // Reference to original blog post + originalPostId: { + type: mongoose.Schema.Types.ObjectId, + ref: 'Blog', + required: true + } +}, { + timestamps: true +}); + +// Indexes +recentPostSchema.index({ createdAt: -1 }); +recentPostSchema.index({ originalPostId: 1 }); + +// Remove __v from JSON output +recentPostSchema.set('toJSON', { + transform: function(doc, ret) { + delete ret.__v; + return ret; + } +}); + +// Static method to sync with Blog posts +recentPostSchema.statics.syncFromBlogs = async function(limit = 5) { + const Blog = require('./blog'); + + // Get recent published blogs + const recentBlogs = await Blog.find({ status: 'published' }) + .sort({ createdAt: -1 }) + .limit(limit) + .select('title slug featuredImage publishedAt'); + + // Clear existing recent posts + await this.deleteMany({}); + + // Create new recent posts + const recentPosts = recentBlogs.map(blog => ({ + title: blog.title, + slug: blog.slug, + thumbnail: blog.featuredImage, + publishedAt: blog.publishedAt, + originalPostId: blog._id + })); + + if (recentPosts.length > 0) { + await this.insertMany(recentPosts); + } + + return recentPosts; +}; + +// Static method to get recent posts +recentPostSchema.statics.getRecent = function(limit = 5) { + return this.find({}).sort({ createdAt: -1 }).limit(limit); +}; + +module.exports = mongoose.model('RecentPost', recentPostSchema); \ No newline at end of file diff --git a/routes/admin.js b/routes/admin.js index d41e56d..94e870f 100644 --- a/routes/admin.js +++ b/routes/admin.js @@ -24,6 +24,11 @@ const insuranceController = require("../controllers/insuranceController"); const activityController = require("../controllers/activityController"); const bookingSubmissionController = require("../controllers/bookingSubmissionController"); +// Blog controllers +const blogController = require("../controllers/blogController"); +const blogCategoryController = require("../controllers/blogCategoryController"); +const blogTagController = require("../controllers/blogTagController"); + // Dashboard router.get("/dashboard", ensureAuthenticated, dashboardController.getDashboard); @@ -344,4 +349,29 @@ router.get("/test-images", ensureAuthenticated, (req, res) => { }); }); +// Blog routes +// Blog Management Routes +router.get("/blog", ensureAuthenticated, blogController.index); +router.get("/blog/create", ensureAuthenticated, blogController.create); +router.post("/blog/create", ensureAuthenticated, blogController.store); +router.get("/blog/:id/edit", ensureAuthenticated, blogController.edit); +router.post("/blog/:id/edit", ensureAuthenticated, blogController.update); +router.post("/blog/:id/delete", ensureAuthenticated, blogController.destroy); + +// Blog Categories Management +router.get("/blog/categories", ensureAuthenticated, blogCategoryController.index); +router.get("/blog/categories/create", ensureAuthenticated, blogCategoryController.create); +router.post("/blog/categories/create", ensureAuthenticated, blogCategoryController.store); +router.get("/blog/categories/:id/edit", ensureAuthenticated, blogCategoryController.edit); +router.post("/blog/categories/:id/edit", ensureAuthenticated, blogCategoryController.update); +router.post("/blog/categories/:id/delete", ensureAuthenticated, blogCategoryController.destroy); + +// Blog Tags Management +router.get("/blog/tags", ensureAuthenticated, blogTagController.index); +router.get("/blog/tags/create", ensureAuthenticated, blogTagController.create); +router.post("/blog/tags/create", ensureAuthenticated, blogTagController.store); +router.get("/blog/tags/:id/edit", ensureAuthenticated, blogTagController.edit); +router.post("/blog/tags/:id/edit", ensureAuthenticated, blogTagController.update); +router.post("/blog/tags/:id/delete", ensureAuthenticated, blogTagController.destroy); + module.exports = router; diff --git a/routes/index.js b/routes/index.js index beecd84..3e5eea9 100644 --- a/routes/index.js +++ b/routes/index.js @@ -19,6 +19,11 @@ const activityController = require("../controllers/activityController"); const travelController = require("../controllers/travelController"); const bookingSubmissionController = require("../controllers/bookingSubmissionController"); +// Blog controllers +const blogController = require("../controllers/blogController"); +const blogCategoryController = require("../controllers/blogCategoryController"); +const blogTagController = require("../controllers/blogTagController"); + // Trang chủ router.get("/", (req, res) => { res.render("index", { @@ -121,6 +126,27 @@ router.get("/demo/session-booking-api", (req, res) => { res.sendFile(path.join(__dirname, '../views/demo/session-booking-api.html')); }); +// Blog API Routes +router.get("/api/blog", blogController.api); +router.get("/api/blog/featured", blogController.apiFeatured); +router.get("/api/blog/recent", blogController.apiRecent); + +// Blog Categories API (must come before /api/blog/:slug) +router.get("/api/blog/categories", blogCategoryController.api); +router.get("/api/blog/categories/:slug", blogCategoryController.apiShow); + +// Blog Tags API (must come before /api/blog/:slug) +router.get("/api/blog/tags", blogTagController.api); +router.get("/api/blog/tags/popular", blogTagController.apiPopular); +router.get("/api/blog/tags/:slug", blogTagController.apiShow); + +// Blog post specific APIs (must come before /api/blog/:slug) +router.get("/api/blog/:id/categories", blogController.apiCategories); +router.get("/api/blog/:id/tags", blogController.apiTags); + +// Blog detail by slug (must come last among blog routes) +router.get("/api/blog/:slug", blogController.apiShow); + // // API route cho blog detail // router.get('/api/blog-detail', blogDetailController.api); diff --git a/scripts/2025_12_02_114122_aboutus.js b/scripts/2025_12_02_114122_aboutus.js deleted file mode 100644 index 267dc94..0000000 --- a/scripts/2025_12_02_114122_aboutus.js +++ /dev/null @@ -1,87 +0,0 @@ -require("dotenv").config(); -const fs = require("fs").promises; -const path = require("path"); -const connectDB = require("../config/database"); -const AboutUs = require("../models/aboutUs"); - -/** - * Normalize the provided aboutUs.json into the AboutUs model shape. - */ -function transformAboutUs(source) { - const hero = { - banner: source?.hero?.banner || "", - title: source?.hero?.title || "", - breadcrumb: source?.hero?.breadcrumb || "", - }; - - // Introduce section - const introduce = { - header: source?.introduce?.header || {}, - services: Array.isArray(source?.introduce?.services) - ? source.introduce.services - : [], - }; - - // Stats - const stats = Array.isArray(source?.stats) ? source.stats : []; - - // Features: header + items - const features = { - header: source?.features?.header || {}, - items: Array.isArray(source?.features?.items) ? source.features.items : [], - }; - - // Activities - const activities = source?.activities || {}; - - // Newsletter - const newsletter = source?.newsletter || {}; - - // Events: header + items - const events = { - header: source?.events?.header || {}, - items: Array.isArray(source?.events?.items) ? source.events.items : [], - }; - - return { - hero, - introduce, - stats, - features, - activities, - newsletter, - events, - updatedAt: new Date(), - }; -} - -/** - * Migration: aboutus - */ -async function migrate() { - try { - await connectDB(); - - const filePath = path.join(__dirname, "..", "data", "aboutUs.json"); - const raw = await fs.readFile(filePath, "utf8"); - const source = JSON.parse(raw); - - const doc = transformAboutUs(source); - - await AboutUs.create(doc); - - const mongoose = require('mongoose'); - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error('Migration error:', error); - process.exit(1); - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate }; diff --git a/scripts/2025_12_02_114122_asd.js b/scripts/2025_12_02_114122_asd.js deleted file mode 100644 index 59084ad..0000000 --- a/scripts/2025_12_02_114122_asd.js +++ /dev/null @@ -1,32 +0,0 @@ -require('dotenv').config(); -const connectDB = require('../config/database'); - -/** - * Migration: asd - * Created: 11:41:22 2/12/2025 - */ -async function migrate() { - try { - // Kết nối database - await connectDB(); - console.log('Starting migration: asd...'); - - // TODO: Thêm code migration của bạn ở đây - - console.log('Migration asd completed successfully!'); - - const mongoose = require('mongoose'); - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error('Migration error:', error); - process.exit(1); - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate }; diff --git a/scripts/2025_12_02_114127_contact.js b/scripts/2025_12_02_114127_contact.js deleted file mode 100644 index bf0b4d3..0000000 --- a/scripts/2025_12_02_114127_contact.js +++ /dev/null @@ -1,38 +0,0 @@ -require("dotenv").config(); -const fs = require("fs").promises; -const path = require("path"); -const connectDB = require("../config/database"); -const Contact = require("../models/contact"); -const mongoose = require("mongoose"); - -/** - * Migration: contact - * Migrate contact data from contact-data.json - */ -async function migrate() { - try { - await connectDB(); - - // Read contact-data.json file - const contactJsonPath = path.join(__dirname, "../data/contact-data.json"); - const contactData = JSON.parse(await fs.readFile(contactJsonPath, "utf8")); - - // Migrate data using the model's static method - await Contact.migrateFromJson(contactData); - - console.log("Contact migration completed successfully"); - - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error("Migration error:", error); - process.exit(1); - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate }; diff --git a/scripts/2025_12_03_154826_create_safety_table.js b/scripts/2025_12_03_154826_create_safety_table.js deleted file mode 100644 index 4a75823..0000000 --- a/scripts/2025_12_03_154826_create_safety_table.js +++ /dev/null @@ -1,89 +0,0 @@ -require("dotenv").config(); -const fs = require("fs").promises; -const path = require("path"); -const connectDB = require("../config/database"); -const Safety = require("../models/safety"); -const mongoose = require("mongoose"); - -/** - * Chuẩn hóa safety.json đúng theo safetySchema - */ -function transformSafety(data) { - return { - hero: { - banner: data?.hero?.banner || "", - title: data?.hero?.title || "", - }, - - approach: { - badge: data?.approach?.badge || "", - title: data?.approach?.title || "", - description: data?.approach?.description || "", - imgs: { - img1: data?.approach?.imgs?.img1 || "", - img2: data?.approach?.imgs?.img2 || "", - }, - stats: { - count: data?.approach?.stats?.count || "", - label: data?.approach?.stats?.label || "", - avatars: Array.isArray(data?.approach?.stats?.avatars) - ? data.approach.stats.avatars - : [], - }, - features: Array.isArray(data?.approach?.features) - ? data.approach.features - : [], - cards: Array.isArray(data?.approach?.cards) - ? data.approach.cards - : [], - }, - - philosophy: { - title: data?.philosophy?.title || "", - subtitle: data?.philosophy?.subtitle || "", - cards: Array.isArray(data?.philosophy?.cards) - ? data.philosophy.cards - : [], - }, - - security: { - title: data?.security?.title || "", - subtitle: data?.security?.subtitle || "", - cards: Array.isArray(data?.security?.cards) - ? data.security.cards - : [], - }, - - updatedAt: new Date(), - }; -} - -/** - * MIGRATION - */ -async function migrate() { - try { - console.log("Starting migration: create_safety_table..."); - await connectDB(); - - const safetyJsonPath = path.join(__dirname, "../data/safety.json"); - const raw = await fs.readFile(safetyJsonPath, "utf8"); - const source = JSON.parse(raw); - - const doc = transformSafety(source); - - await Safety.create(doc); - - console.log("Migration create_safety_table completed successfully!"); - - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error("Migration error:", error); - process.exit(1); - } -} - -if (require.main === module) migrate(); - -module.exports = { migrate }; diff --git a/scripts/2025_12_04_162258_faq.js b/scripts/2025_12_04_162258_faq.js deleted file mode 100644 index 9640b47..0000000 --- a/scripts/2025_12_04_162258_faq.js +++ /dev/null @@ -1,328 +0,0 @@ -require("dotenv").config(); -const fs = require("fs").promises; -const path = require("path"); -const connectDB = require("../config/database"); -const FAQ = require("../models/faq"); -const mongoose = require("mongoose"); - -/** - * Migration: faq - * Migrate FAQ data from faq-data.json - * Created: 16:22:58 4/12/2025 - */ -async function migrate() { - try { - // Kết nối database - await connectDB(); - console.log("Starting migration: faq..."); - - // Read faq-data.json file - cùng cấp với file này - const faqJsonPath = path.join(__dirname, "../data/faq-data.json"); - - try { - const faqData = JSON.parse(await fs.readFile(faqJsonPath, "utf8")); - console.log("FAQ JSON data loaded successfully"); - - // Đảm bảo có trường name - if (!faqData.name) { - faqData.name = "default"; - } - - // Migrate data using the model's static method - const result = await FAQ.importFromJson(faqData); - - // Tính tổng số FAQ - const totalFaqs = result.faqSections.reduce((total, section) => { - return total + (section.faqs ? section.faqs.length : 0); - }, 0); - - console.log("FAQ migration completed successfully"); - console.log(`Total sections migrated: ${result.faqSections.length}`); - console.log(`Total FAQs migrated: ${totalFaqs}`); - console.log(`FAQ ID: ${result._id}`); - - } catch (fileError) { - console.error("Error reading FAQ JSON file:", fileError.message); - - // Nếu không có file JSON, tạo data mẫu với dữ liệu đầy đủ - console.log("Creating complete FAQ data..."); - - const defaultFaqData = { - name: "default", - hero: { - title: "Go and Grow Camp", - backgroundImage: "yootheme/cache/18/faqs_header_new.jpg", - overlayColor: "rgba(0, 0, 0, 0)", - sectionClass: "uk-section-secondary uk-section-overlap uk-preserve-color uk-light", - titleClass: "uk-heading-large uk-text-center !text-[5vw]", - enableScrollspy: true, - backgroundPosition: "top-center" - }, - sidebarNav: [ - { - id: "general-information", - label: "General Information" - }, - { - id: "camps", - label: "Camps" - }, - { - id: "camp-routine", - label: "Camp Routine" - }, - { - id: "camp-counselors", - label: "Camp Counselors" - }, - { - id: "camp-rules", - label: "Camp Rules" - }, - { - id: "safety", - label: "Safety" - }, - { - id: "accommodation-catering", - label: "Accommodation & Catering" - }, - { - id: "transfers-shuttles", - label: "Transfers & Shuttles" - } - ], - contactBox: { - title: "Let's plan your perfect nature escape", - phone: { - icon: "phone", - text: "+(123)-456-789" - }, - email: { - icon: "email", - text: "hello@ggcamp.org" - } - }, - faqSections: [ - { - id: "general-information", - title: "General Information", - faqs: [ - { - title: "What are FAQ?", - description: "FAQ are the initials for \"Frequently Asked Questions\".\n\nThe FAQ have been compiled by us over a long period of time and are intended to help give a general overview of our camps and clarify questions that arise before booking a camp." - }, - { - title: "General booking process", - description: "Once the booking has been confirmed by us, you will receive an e-mail requesting a deposit. As soon as we have received this, you will receive an e-mail with a payment confirmation.\nPlease have a look at the welcome package, which will reach you by e-mail with the Last Travel Information. This contains information that applies to the camp you have booked.\n\nStep 1: Registration\nStep 2: Receipt of registration confirmation, total invoice and deposit request (e-mail)\nStep 3: Deposit of USD 50 (due within 7 days after booking)\nStep 4: Receiving an email with the latest important travel information, a packing list, addresses and important emergency phone numbers plus remaining payment request about 3-4 weeks before the camp starts." - }, - { - title: "Terms & Conditions", - description: "Our Terms & Conditions can be found in our official documents section." - }, - { - title: "Where can I find a packing guide for Camps?", - description: "Just click here to download our packing list." - }, - { - title: "Where can I find contact information from Camps and addresses?", - description: "Here you can find all the necessary information if you want to drive to our camps or send something. If you want to send something please ALWAYS include the full name of your child on the letter/package and please only send it at the time when your kids are staying in camp as we cannot store it for a longer period of time.\n\nWalsrode/Lüneburger Heide - Germany:\nCamp Adventure, Vethem 58, 29664 Walsrode, Germany\nwalsrode@campadventure.de\n\nRegen/Bavarian Forest - Germany:\nCamp Adventure, Badstrasse 18, 94209 Regen\nregen@campadventure.de\n\nBarcelona - Spain:\nBISC International Sailing Center, c/o Camp Adventure, Parc del Fòrum Sota plaça fotovoltàica, 08930 Sant Adrià de Besòs, Barcelona, Spain\nbarcelona@campadventure.de\n\nBath - England:\nUniversity of Bath, c/o Camp Adventure, Claverton Down, Bath BA2 7AY, England\nengland@campadventure.de\n\nRossall - England:\nRossall School, Broadway, Fleetwood, Lancashire FY7 8JW, England\nengland@campadventure.de" - } - ] - }, - { - id: "camps", - title: "Camps", - faqs: [ - { - title: "Where do kids and camp counselors come from?", - description: "Camp Adventure attaches great importance to internationality. The participants and supervisors in our camps come from many different countries. Last year, for example, we had participants from over 60 different countries and counselors from 25 different nations. Of course, we don't know where they will come from this year. So we are at least as excited as you are.\n\nThrough our office in Hamburg and our branch office in Canada, we reach motivated and committed counselors from all over the world. Canadian and Australian teamers can therefore be found as well as German or Spanish teamers.\n\nDue to the different experiences and cultural backgrounds an indescribably fantastic, international atmosphere is created." - }, - { - title: "Which languages are spoken in camp?", - description: "The main language in all our camps is English. In addition, there is the language of the country in which the camp takes place. As we have our headquarters in Germany, German teamers are always present in all camps in Germany. All announcements and explanations are here therefore always in German and English. Of course, all our teamers with their different nationalities are also available for individual translations." - }, - { - title: "Are there problems if children have low language skills?", - description: "No, because there are usually more participants and team members who speak the same language. We know from experience that children are excellent at communicating nonverbally. They often need a few days to warm up to it, but are then very open to other children as well." - }, - { - title: "Are girls and boys separated?", - description: "Girls and boys are accommodated separately in the dormitories/tents. The program is completely mixed." - }, - { - title: "How big are the camps? How high is the caregiver ratio?", - description: "Capacities range from around 30 participants in smaller language camps to a maximum of about 400 children in our camp Lueneburger Heide. However, the maximum capacity is not reached every week. However, a minimum number of participants must be guaranteed in order to run the camp.\n\nIt is important to us that all children are always grouped in small groups of 5-8, with a counselor as a contact person. This way homesickness doesn't stand a chance and despite the size of the camp in their group family, they experience a strong bond on which they can count on!" - }, - { - title: "Should 12-year-olds go to Junior Camp or Senior Camp?", - description: "This question is not easy to answer and depends on the individual stage of development of your child. Therefore, as parents, we leave you the opportunity to decide for yourself. In the Junior Camp they belong to the older ones and can explore a lot in a playful way. In the Senior Camp they are the younger ones, who have role models through the older ones, whom they can emulate." - } - ] - }, - { - id: "camp-routine", - title: "Camp Routine", - faqs: [ - { - title: "How is the choice of activities/courses in the camps made?", - description: "If your child would like to participate in a paid additional course (e.g. horse riding, language course, Survival etc.), this must be booked in advance when registering. In principle, no extra additional courses have to be booked. A program with a variety of activities is of course available to the participants in all camps. The various activities can be chosen by the participants on site in the respective camps. We present the offers to the participants, so that everyone gets an insight into the different courses. The children can then register in the lists of the respective courses." - }, - { - title: "What is a hike?", - description: "The hike is a 1-3 day walking tour, in which all participants of the Adventure Camp who stay 2 weeks in the camp take part. On this hike the participants will not spend the night in a tent, but either in the open air or under a self-made shelter e.g. from tarpaulins. They will of course be accompanied by their teamers. The hike is a very special experience and a highlight for all participants. For this hike the participants need sturdy shoes and a big backpack." - }, - { - title: "Can I wash my clothes during the camp?", - description: "In principle, participants should bring sufficient clothing and change of clothes for the entire camp period.\n\nOnly in the camps in Lüneburger Heide and Bayerischer Wald a laundry service will be offered for kids staying three weeks or more, which means that a laundry bag (approx. 3 kg) will be washed in the laundry centre of the next village at a price of USD 45. This service can be booked upon registration for three-week camps. Please note that the laundry will be done either after one week or after two weeks." - }, - { - title: "Anti Homesick Adviser", - description: "Dear parents\n\nNow it's almost time: In summer your child travels for the first time with Camp Adventure. Maybe it will be the first time that he travels alone without parents or relatives. As we are getting more and more questions, we have decided to put together a small package for you parents with little tips from experts to make everything as easy as possible for you and your child. Follow our tips and your child will have a fantastic holiday, have many new experiences and make friends from all over the world! All these tips have been developed together with the International Camping Fellowship. And the more you think your child will be a \"homesick candidate\" - or your child even claims to be one - the more you consider the following tips." - } - ] - }, - { - id: "camp-counselors", - title: "Camp Counselors - Our Teamers", - faqs: [ - { - title: "Who are the camp counselors?", - description: "Every year our team is made up of an international mix. The non-profit association Camp Europe e.V. with headquarters in Hamburg and a branch office in Canada takes care of the acquisition of national and international applicants. Since we have about 50% German-speaking children, there are also German carers in every location. But many also come from other countries, such as England, Spain, Canada and Australia, to name just a few." - }, - { - title: "How are the teamers trained?", - description: "All counselors go through an extensive application process. For a successful application, not only an interesting curriculum vitae and a minimum age of 19 years are sufficient! We conduct a personal interview with each individual in which our employees get a first impression of the applicant.\n\nBefore the camp season, everyone, both the first supervisors (teamers) as well as many recomers, complete a one-week training in which they are prepared for their assignment by trained coaches. They must have a first aid certificate, which may not be older than two years, as well as an internationally flawless police clearance certificate. We know how important the teamers are for a great camp and therefore select them very conscientiously." - } - ] - }, - { - id: "camp-rules", - title: "Camp Rules", - faqs: [ - { - title: "Drugs, Alcohol & Camp?", - description: "From our point of view an absolutely unacceptable and indiscutable combination! Due to our cooperation with the association \"Keine Macht den Drogen\" (No power to drugs) and our common opinion that all kinds of drugs do not belong in the hands of children & teenagers, any possession or consumption of drugs is forbidden for teenagers and children in the camp and also outside the camp.\n\nViolations can lead to exclusion or even to criminal charges. The term \"drugs\" also includes cigarettes and alcohol! Through our varied activities, we offer a much better alternative! We would like to make it clear from the outset that we are also against any form of discrimination or \"putting down\". This is - just like violence - immediately prevented by us, in order to offer each young person a relaxed and joyful time in the camp." - }, - { - title: "Should I call my kid or write an old-fashioned letter?", - description: "We ask all parents to write to their child at least once. This is especially useful at the beginning, as it is a particularly upsetting experience for every child and every teenager when most of the participants receive a letter, but they do not.\n\nPlease note that there is NO public \"camp phone\" available for incoming or outgoing calls. If your child doesn't bring her/his own phone, she/he won't be able to call you. In case of any problems, we will of course contact you immediately.\n\nIf your child brings a mobile phone, we will collect it on arrival and store it with the valuables. Your child's Teamer may hand it over during the phone time after lunch. Please keep in mind: no news is good news (the location manager will contact you if it is necessary due to homesickness or illness). We kindly ask you not to call the office in Hamburg to ask about your kid's health and wellbeing, nor if you would like to know why your child hasn't called you yet. Please use our camp email service for such enquiries.\n\nOur recommendation is the following:\nWe recommend not to call your child (even if he or she has a mobile phone with him or her) and not to tell him or her to call you. Telephoning can in our experience promote homesickness very strongly and your child will be cured thereby if completely immersed in camp life! At noon after lunch, if absolutely necessary, your child can pick up his or her mobile phone from the counselors until the start of next program and make phone calls. Instead, you are welcome to bring a pre-stamped and addressed envelope with you. We will then make sure that your child has enough time to write letters. Since letters and postcards often arrive late at the camps, we also offer the e-mail service. You can send your child max. ONE email per day directly to the camp, which we then print out and give to your child. There is no way for them to reply, but your child will be happy to receive a small message from home. You can find the postal and email address in the info package of the booked camp." - }, - { - title: "Are there any prohibited items?", - description: "Yes, there are. Not allowed are pocket knives with lockable blades, all weapons, lighters and matches (danger of fire in the forest!). Drugs of any kind, including alcohol and cigarettes, are also included." - } - ] - }, - { - id: "safety", - title: "Safety", - faqs: [ - { - title: "Electronic equipment and valuables", - description: "We recommend that you do not take an MP3 player, e-book, tablet, etc. or any valuables with you. On the one hand we do not assume any liability and on the other hand there are no possibilities to charge the devices. We are of the opinion that the camp time is a special experience for the participants if they do not have the headphones in their ears all the time or are busy with their mobile phones. Instead they have the chance to deal with other topics and they find time to dedicate themselves to the new people in the camp." - }, - { - title: "How do you provide safety for the kids?", - description: "Before our camp counselors start working with us, we check their police clearance certificates. You must be at least 19 years old to work for us as a teamer. They must also have a \"First Aid Certificate\", which must not be older than two years. In the camps we try to make sure that only adults from our camp or familiar faces are on the campground and that all our carers look after strangers.\n\nWe have many different camp sites. Some of them are fenced in, others are not. There are no armed guards or the like in our camps, as we believe that these conditions create a very insecure feeling. We do not have a high security zone in Germany, Northern Ireland or England, but we keep our eyes open and do everything we can to ensure that all participants have a great time." - }, - { - title: "Insurance in case of illness?", - description: "If your child should fall ill during the camp and medical help is required, he or she will of course be taken to the doctor by our carers and cared for there as well. It is therefore necessary for each participant to take their insurance card with them to the camp. We offer all participants the possibility of taking out liability, casualty & health insurance for travel abroad with us. This covers all costs in case of illness and prevents international children in particular from having to \"advance\" their own cash. You can find more detailed information on insurance in our documents section." - } - ] - }, - { - id: "accommodation-catering", - title: "Accommodation & Catering", - faqs: [ - { - title: "How's the food at the camps?", - description: "Full board for the entire duration of the camp is of course already included in the camp price. In addition, water and fruit are available for the participants around the clock. For us it is a matter of course to provide one variant for vegetarians and one pork-free with each meal. In case of special allergies or intolerances of your children let us know in advance and we will try to find a solution." - }, - { - title: "How is my child accommodated in the camp?", - description: "In our Adventure Camp Bayerischer Wald and our Camp Lueneburger Heide, the Juniors (7-12) and the Seniors (12-16) can choose between tents and huts.\n\nThe tents are equipped with a floor and a wooden platform, up to 7 children can share one tent. The participants can make themselves comfortable with sleeping bag and sleeping mat. The wooden huts are equipped with bunk beds and can accommodate 4-8 children. At the other locations, participants will be accommodated in shared rooms in youth hostels, sports centres or boarding schools of private schools. You will find detailed information about the accommodation on the individual camp pages." - } - ] - }, - { - id: "transfers-shuttles", - title: "Transfers & Shuttles", - faqs: [ - { - title: "Entry regulations/Travel Consent for group flights", - description: "All parents need to fill this out and bring it to camp:\n\nBelow is a summary of the travel requirements for minors from various EU countries traveling with Camp Adventure on group flights. Please note that regulations can change, so it's essential to consult the official resources provided for the most up-to-date information." - }, - { - title: "Which transfers are offered?", - description: "The respective transfer possibilities depend on the period and venue of the camp. Check directly on the respective camp page under \"Arrival & Departure Services\"." - }, - { - title: "Where can I find the exact arrival and departure times?", - description: "Information about the different arrival and departure times can be found on the respective camp page under \"Arrival & Departure Services\"." - }, - { - title: "How do the transfer costs come about?", - description: "When booking a train or air trip, the indicated price includes the arrival and departure as well as the accompaniment by a supervisor." - }, - { - title: "Where can I find the address/driving directions from the camp?", - description: "You will receive the exact address and directions of the camp with the Last Travel Information about 3-4 weeks before the camp starts." - } - ] - } - ], - video: { - url: "https://www.youtube.com/embed/3NtE5wSwYTo?list=PLSOedrxa1c-bxvH6uuz_oZdIfJkov66wB&disablekb=1", - title: "Anti Homesickness Adviser" - } - }; - - const result = await FAQ.importFromJson(defaultFaqData); - console.log("Complete FAQ data created successfully"); - console.log(`FAQ ID: ${result._id}`); - } - - // Kiểm tra lại data đã được lưu - const savedFaq = await FAQ.findOne({ name: "default" }); - if (savedFaq) { - const totalFaqs = savedFaq.faqSections.reduce((total, section) => { - return total + (section.faqs ? section.faqs.length : 0); - }, 0); - - console.log("\n=== Migration Summary ==="); - console.log(`FAQ Name: ${savedFaq.name}`); - console.log(`Hero Title: ${savedFaq.hero.title}`); - console.log(`Sidebar Items: ${savedFaq.sidebarNav.length}`); - console.log(`FAQ Sections: ${savedFaq.faqSections.length}`); - console.log(`Total FAQ Items: ${totalFaqs}`); - console.log(`Created At: ${savedFaq.createdAt}`); - console.log(`Updated At: ${savedFaq.updatedAt}`); - - // Hiển thị chi tiết từng section - console.log("\n=== FAQ Sections Details ==="); - savedFaq.faqSections.forEach((section, index) => { - console.log(`Section ${index + 1}: ${section.title} (${section.faqs.length} FAQs)`); - }); - console.log("=========================\n"); - } - - await mongoose.disconnect(); - console.log("Migration faq completed successfully!"); - process.exit(0); - } catch (error) { - console.error("Migration error:", error); - if (mongoose.connection.readyState !== 0) { - await mongoose.disconnect(); - } - process.exit(1); - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate }; \ No newline at end of file diff --git a/scripts/2025_12_08_135459_aboutus.js b/scripts/2025_12_08_135459_aboutus.js deleted file mode 100644 index 8eff6f1..0000000 --- a/scripts/2025_12_08_135459_aboutus.js +++ /dev/null @@ -1,79 +0,0 @@ -const mongoose = require('mongoose'); -const AboutUs = require('../models/aboutUs'); -const fs = require('fs'); -require('dotenv').config(); - -// Load and clean JSON data -const raw = fs.readFileSync(require('path').join(__dirname, '..', 'data', 'aboutUs.json'), 'utf8'); -let data = JSON.parse(raw || '{}'); - -// Remove _id fields recursively to avoid conflicts -function stripIds(obj) { - if (Array.isArray(obj)) return obj.map(i => stripIds(i)); - if (obj && typeof obj === 'object') { - const out = {}; - for (const k in obj) { - if (k !== '_id') out[k] = stripIds(obj[k]); - } - return out; - } - return obj; -} - -data = stripIds(data); - -// Check for --dry-run flag -const dryRun = process.argv.includes('--dry-run') || process.argv.includes('-n'); - -async function importAboutUs() { - try { - const dbUri = process.env.MONGODB_URI || 'mongodb://localhost:27017/ggcamps'; - console.log('📍 Using DB URI:', dbUri); - - if (dryRun) { - console.log('\n🔍 === DRY RUN MODE ==='); - console.log('Document to be upserted (preview only, no DB changes):\n'); - console.log(JSON.stringify(data, null, 2)); - console.log('\n=== END DRY RUN ===\n'); - console.log('To actually import, run without --dry-run flag'); - process.exit(0); - } - - console.log('🔄 Connecting to database...'); - await mongoose.connect(dbUri); - console.log('✓ Connected to database'); - - // Safe upsert: update existing doc or create new one - console.log('📥 Upserting AboutUs document (safe mode)...'); - const result = await AboutUs.findOneAndUpdate({}, data, { - upsert: true, - new: true, - setDefaultsOnInsert: true - }); - - console.log('✅ Successfully upserted AboutUs data!'); - console.log('📝 Document ID:', result._id.toString()); - console.log('📊 Data structure:'); - console.log(' - Hero:', data.hero ? '✓' : '✗'); - console.log(' - Introduction:', data.introduction ? '✓' : '✗'); - console.log(' - Introduction Services:', data.introduction?.services?.length || 0, 'items'); - console.log(' - Statistics:', data.statistics ? '✓' : '✗'); - console.log(' - Statistics Items:', data.statistics?.items?.length || 0, 'items'); - console.log(' - Accommodation:', data.accommodation ? '✓' : '✗'); - console.log(' - Accommodation Features:', data.accommodation?.features?.length || 0, 'items'); - console.log(' - Activities:', data.activities ? '✓' : '✗'); - console.log(' - Activities Gallery:', data.activities?.gallery?.length || 0, 'items'); - console.log(' - Newsletter:', data.newsletter ? '✓' : '✗'); - console.log(' - Events:', data.events ? '✓' : '✗'); - console.log(' - Events Items:', data.events?.items?.length || 0, 'items'); - - process.exit(0); - } catch (error) { - console.error('❌ Error:', error.message); - console.error(error); - process.exit(1); - } -} - -// Run import -importAboutUs(); diff --git a/scripts/2025_12_09_131838_camp_location.js b/scripts/2025_12_09_131838_camp_location.js deleted file mode 100644 index 3982c21..0000000 --- a/scripts/2025_12_09_131838_camp_location.js +++ /dev/null @@ -1,105 +0,0 @@ -require('dotenv').config(); -const fs = require('fs').promises; -const path = require('path'); -const connectDB = require('../config/database'); -const CampLocation = require('../models/campLocation'); - -async function validateCampLocationData(data) { - const requiredFields = [ - 'metadata', - 'hero', - 'camps', - 'locations', - 'intro', - 'faq', - 'welcomeQuote', - 'securityConcept' - ]; - - const missingFields = requiredFields.filter(field => !data[field]); - if (missingFields.length > 0) { - throw new Error(`Missing required fields: ${missingFields.join(', ')}`); - } - - // Validate camps array - if (!Array.isArray(data.camps)) { - throw new Error('Camps must be an array'); - } - - // Validate each camp has required fields - data.camps.forEach((camp, index) => { - if (!camp.id) { - throw new Error(`Camp at index ${index} is missing required field: id`); - } - if (!camp.title) { - throw new Error(`Camp at index ${index} is missing required field: title`); - } - }); - - // Validate locations array - if (!Array.isArray(data.locations) || data.locations.length === 0) { - throw new Error('Locations must be a non-empty array'); - } - - // Validate FAQ array - if (!Array.isArray(data.faq) || data.faq.length === 0) { - throw new Error('FAQ must be a non-empty array'); - } - - // Validate security concept items - if (!Array.isArray(data.securityConcept.items) || data.securityConcept.items.length === 0) { - throw new Error('Security concept items must be a non-empty array'); - } - - console.log('✓ Data validation passed'); -} - -/** - * Migration: camp_location - * Created: 13:18:38 9/12/2025 - * Imports camp location data including camps, locations, FAQ, and security information - */ -async function migrate() { - try { - // Kết nối database - await connectDB(); - console.log('Starting migration: camp_location...'); - - // Delete existing data - const deleteResult = await CampLocation.deleteMany({}); - console.log(`✓ Deleted ${deleteResult.deletedCount} existing records`); - - // Read JSON file - const campLocationData = JSON.parse( - await fs.readFile(path.join(__dirname, '../data/camp-location.json'), 'utf8') - ); - console.log('✓ Loaded camp-location.json'); - - // Validate data - await validateCampLocationData(campLocationData); - - // Create new record - const result = await CampLocation.create(campLocationData); - console.log('✓ Created camp location record'); - console.log(` - ${result.camps.length} camps`); - console.log(` - ${result.locations.length} locations`); - console.log(` - ${result.faq.length} FAQ items`); - console.log(` - ${result.securityConcept.items.length} security measures`); - - console.log('Migration camp_location completed successfully!'); - - const mongoose = require('mongoose'); - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error('Migration error:', error); - process.exit(1); - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate }; diff --git a/scripts/2025_12_10_111338_insurance.js b/scripts/2025_12_10_111338_insurance.js deleted file mode 100644 index 2715e43..0000000 --- a/scripts/2025_12_10_111338_insurance.js +++ /dev/null @@ -1,81 +0,0 @@ -require('dotenv').config(); -const fs = require('fs').promises; -const path = require('path'); -const connectDB = require('../config/database'); -const Insurance = require('../models/insurance'); -const mongoose = require('mongoose'); - -/** - * Migration: insurance - * Created: 11:13:38 10/12/2025 - * Updated: 14/12/2025 - Simplified for new structure only - */ -async function migrate() { - try { - console.log('Starting migration: insurance...'); - await connectDB(); - - // Đọc file insurance.json - const insuranceJsonPath = path.join(__dirname, '../data/insurance.json'); - console.log('Reading JSON file from:', insuranceJsonPath); - - const insuranceData = JSON.parse(await fs.readFile(insuranceJsonPath, 'utf8')); - console.log('Insurance data loaded successfully'); - - // Sử dụng phương thức migrateFromJson của model - await Insurance.migrateFromJson(insuranceData); - - console.log('Insurance migration completed successfully!'); - - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error('Migration error:', error); - process.exit(1); - } -} - -/** - * Custom migration logic cho insurance data - */ -async function migrateInsuranceData(insuranceData) { - try { - console.log('Starting custom migration logic...'); - - // Xóa dữ liệu cũ nếu có - const existingInsurance = await Insurance.find({}); - if (existingInsurance.length > 0) { - console.log(`Found ${existingInsurance.length} existing insurance documents`); - await Insurance.deleteMany({}); - console.log('Cleared existing insurance data'); - } - - // Tạo document mới - const insuranceDocument = new Insurance({ - name: 'default', - version: '2.0.0', - language: 'en', - hero: insuranceData.hero, - page: insuranceData.page, - content: insuranceData.content, - createdAt: new Date(), - updatedAt: new Date(), - isActive: true - }); - - // Lưu vào database - await insuranceDocument.save(); - console.log('Insurance document saved successfully'); - - } catch (error) { - console.error('Error in migrateInsuranceData:', error); - throw error; - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate, migrateInsuranceData }; diff --git a/scripts/2025_12_10_120047_terms.js b/scripts/2025_12_10_120047_terms.js deleted file mode 100644 index fc9524b..0000000 --- a/scripts/2025_12_10_120047_terms.js +++ /dev/null @@ -1,308 +0,0 @@ -require('dotenv').config(); -const fs = require('fs').promises; -const path = require('path'); -const connectDB = require('../config/database'); -const Terms = require('../models/terms'); -const mongoose = require('mongoose'); - -/** - * Migration: terms - * Migrate Terms & Conditions data từ terms-conditions.json - * Đã sửa để phù hợp với cấu trúc mới: hero, page, content - * Created: 12:00:47 10/12/2025 - */ -async function migrate() { - try { - // Kết nối database - await connectDB(); - console.log('Starting migration: terms...'); - - // Đọc file terms-conditions.json - const termsJsonPath = path.join(__dirname, '../data/terms-conditions.json'); - console.log('Reading JSON file from:', termsJsonPath); - - const termsData = JSON.parse(await fs.readFile(termsJsonPath, 'utf8')); - console.log('Terms data loaded successfully'); - console.log('Data structure keys:', Object.keys(termsData)); - - // Kiểm tra cấu trúc và gọi method phù hợp - if (termsData.hero && termsData.page && termsData.content) { - // Cấu trúc mới - sử dụng migrateFromNewJson - console.log('Detected new structure, using migrateFromNewJson...'); - if (typeof Terms.migrateFromNewJson === 'function') { - await Terms.migrateFromNewJson(termsData); - console.log('Migration completed using migrateFromNewJson method'); - } else { - console.log('migrateFromNewJson not found, using custom logic...'); - await migrateTermsData(termsData); - } - } else if (termsData.hero && termsData.termsHeader && termsData.sections) { - // Cấu trúc cũ - sử dụng migrateFromJson - console.log('Detected old structure, using migrateFromJson...'); - if (typeof Terms.migrateFromJson === 'function') { - await Terms.migrateFromJson(termsData); - console.log('Migration completed using migrateFromJson method'); - } else { - await migrateTermsData(termsData); - } - } else { - // Không xác định được cấu trúc - console.error('Unknown data structure. Keys:', Object.keys(termsData)); - throw new Error('Unknown terms data structure'); - } - - console.log('Terms migration completed successfully!'); - - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error('Migration error:', error); - process.exit(1); - } -} - -/** - * Custom migration logic cho terms data với cấu trúc mới - * Chỉ có 3 phần: hero, page, content - */ -async function migrateTermsData(termsData) { - try { - console.log('Starting custom migration logic with new structure...'); - - // 1. Xóa dữ liệu cũ (tùy chọn) - const existingTerms = await Terms.find({}); - if (existingTerms.length > 0) { - console.log(`Found ${existingTerms.length} existing terms documents`); - // Có thể bỏ comment để xóa dữ liệu cũ nếu cần - // await Terms.deleteMany({}); - // console.log('Cleared existing terms data'); - } - - // 2. Chuyển đổi từ cấu trúc cũ sang cấu trúc mới nếu cần - let heroData, pageData, contentData; - - // Kiểm tra xem data có cấu trúc cũ hay mới - if (termsData.hero && termsData.page && termsData.content) { - // Đây là cấu trúc mới, sử dụng trực tiếp - console.log('Using new structure (hero, page, content)'); - heroData = termsData.hero; - pageData = termsData.page; - contentData = termsData.content; - - // Debug: kiểm tra content data - console.log('contentData keys:', Object.keys(contentData)); - console.log('contentData.content exists?', !!contentData.content); - console.log('contentData.content length:', contentData.content ? contentData.content.length : 0); - if (contentData.content && contentData.content.length > 0) { - console.log('First content item type:', contentData.content[0].type); - } - } else if (termsData.hero && termsData.termsHeader && termsData.sections) { - // Đây là cấu trúc cũ, cần chuyển đổi sang cấu trúc mới - console.log('Converting from old structure to new structure...'); - heroData = termsData.hero; - pageData = convertOldPageToNew(termsData); - contentData = convertOldSectionsToNew(termsData); - } else { - throw new Error('Unknown terms data structure'); - } - - // 3. Tạo document mới cho terms - const termsDocument = new Terms({ - version: '2.0.0', // Tăng version vì cấu trúc thay đổi - language: 'en', - - // Cấu trúc mới chỉ có 3 phần chính - hero: { - title: heroData.title, - backgroundImage: heroData.backgroundImage, - sectionClass: heroData.sectionClass || 'uk-section-default uk-section-overlap uk-preserve-color uk-light uk-position-relative', - backgroundClasses: heroData.backgroundClasses || 'uk-background-norepeat uk-background-cover uk-background-top-center uk-section uk-section-xlarge', - overlayStyle: heroData.overlayStyle || { backgroundColor: 'rgba(0, 0, 0, 0)' }, - titleClass: heroData.titleClass || 'text-white text-[5vw] uk-text-center', - enableScrollspy: heroData.enableScrollspy !== undefined ? heroData.enableScrollspy : true - }, - - page: { - title: pageData.title, - divider: pageData.divider !== undefined ? pageData.divider : true, - sectionClass: pageData.sectionClass || 'uk-section-default uk-section-overlap uk-section', - titleClass: pageData.titleClass || 'text-[2.5vw] text-[#292c3d] uk-text-left@m uk-text-center', - dividerClass: pageData.dividerClass || 'uk-divider-small uk-text-left@m uk-text-center' - }, - - content: { - sectionClass: contentData.sectionClass || 'uk-section-muted uk-section-overlap uk-section', - textClass: contentData.textClass || 'uk-panel uk-margin text-[1vw]', - content: contentData.content || [] - }, - - // Metadata - createdAt: new Date(), - updatedAt: new Date(), - isActive: true, - migratedFromOldStructure: !termsData.content // Đánh dấu nếu được chuyển từ cấu trúc cũ - }); - - // 4. Lưu vào database - await termsDocument.save(); - console.log('Terms document saved successfully with new structure'); - - // 5. Log thông tin - console.log(`Created terms document with ID: ${termsDocument._id}`); - console.log(`Hero title: ${termsDocument.hero.title}`); - console.log(`Page title: ${termsDocument.page.title}`); - console.log(`Content items count: ${termsDocument.content.content.length}`); - - // 6. Tạo thêm bản German nếu có - const germanJsonPath = path.join(__dirname, '../data/terms-conditions.de.json'); - try { - const germanData = JSON.parse(await fs.readFile(germanJsonPath, 'utf8')); - - // Xác định cấu trúc của German data - let germanHero, germanPage, germanContent; - - if (germanData.hero && germanData.page && germanData.content) { - germanHero = germanData.hero; - germanPage = germanData.page; - germanContent = germanData.content; - } else if (germanData.hero && germanData.termsHeader && germanData.sections) { - germanHero = germanData.hero; - germanPage = convertOldPageToNew(germanData); - germanContent = convertOldSectionsToNew(germanData); - } - - const germanTerms = new Terms({ - ...termsDocument.toObject(), - _id: new mongoose.Types.ObjectId(), // Tạo ID mới - language: 'de', - hero: { - ...termsDocument.hero, - title: germanHero.title || termsDocument.hero.title - }, - page: { - ...termsDocument.page, - title: germanPage.title || termsDocument.page.title - }, - content: { - ...termsDocument.content, - content: germanContent.content || termsDocument.content.content - }, - isActive: true - }); - - await germanTerms.save(); - console.log('German terms document created successfully'); - } catch (error) { - console.log('German version not found or error:', error.message); - console.log('Continuing with English version only...'); - } - - } catch (error) { - console.error('Error in migrateTermsData:', error); - throw error; - } -} - -/** - * Chuyển đổi từ cấu trúc page cũ sang cấu trúc page mới - */ -function convertOldPageToNew(oldData) { - return { - title: oldData.termsHeader?.title || 'Terms & Conditions', - divider: oldData.termsHeader?.divider !== false, - sectionClass: oldData.termsHeader?.sectionClass || 'uk-section-default uk-section-overlap uk-section', - titleClass: oldData.termsHeader?.titleClass || 'text-[2.5vw] text-[#292c3d] uk-text-left@m uk-text-center', - dividerClass: oldData.termsHeader?.dividerClass || 'uk-divider-small uk-text-left@m uk-text-center' - }; -} - -/** - * Chuyển đổi từ cấu trúc sections cũ sang cấu trúc content mới - */ -function convertOldSectionsToNew(oldData) { - const contentItems = []; - - // Thêm disclaimer đầu tiên nếu có - if (oldData.disclaimer?.text) { - contentItems.push({ - type: 'paragraph', - text: oldData.disclaimer.text - }); - } - - // Thêm các sections - if (oldData.sections && Array.isArray(oldData.sections)) { - oldData.sections.forEach(section => { - if (section.title && section.content) { - const contentItem = { - type: 'section', - title: section.title, - content: section.content - }; - - // Thêm subsections nếu có - if (section.subsections && section.subsections.length > 0) { - contentItem.subsections = section.subsections.map(sub => ({ - type: 'note', - text: sub - })); - } - - // Thêm cancellation table nếu có - if (section.fees) { - contentItem.subsections = contentItem.subsections || []; - contentItem.subsections.push({ - type: 'cancellation_table', - title: 'Standard Cancellation Fees', - items: Object.entries(section.fees).map(([key, value]) => `${key}: ${value}`) - }); - } - - contentItems.push(contentItem); - } - }); - } - - // Thêm footer note nếu có - if (oldData.footerNote?.text) { - contentItems.push({ - type: 'paragraph', - text: oldData.footerNote.text - }); - } - - return { - sectionClass: oldData.layout?.termsSectionClass || 'uk-section-muted uk-section-overlap uk-section', - textClass: oldData.layout?.textContentClass || 'uk-panel uk-margin text-[1vw]', - content: contentItems - }; -} - -/** - * Hàm backup data trước khi migration - */ -async function backupExistingData() { - try { - console.log('Creating backup of existing terms data...'); - const existingTerms = await Terms.find({}); - - if (existingTerms.length > 0) { - const backupPath = path.join(__dirname, '../backups/terms-backup-' + Date.now() + '.json'); - - // Tạo thư mục backup nếu chưa có - await fs.mkdir(path.dirname(backupPath), { recursive: true }); - - await fs.writeFile(backupPath, JSON.stringify(existingTerms, null, 2)); - console.log(`Backup created at: ${backupPath}`); - } - } catch (error) { - console.error('Backup error:', error); - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate, migrateTermsData }; \ No newline at end of file diff --git a/scripts/2025_12_10_125800_activities.js b/scripts/2025_12_10_125800_activities.js deleted file mode 100644 index 57ac039..0000000 --- a/scripts/2025_12_10_125800_activities.js +++ /dev/null @@ -1,219 +0,0 @@ -require("dotenv").config(); -const fs = require("fs").promises; -const path = require("path"); -const connectDB = require("../config/database"); -const Activity = require("../models/activity"); - -/** - * Transform activities.json data to match Activity model schema - */ -function transformActivity(source, index, heroData) { - // Return a document that preserves the main activity fields and also - // keeps the detailed camp information (from `camp-detail`) under the - // `campDetail` key so it can be queried later. - return { - // Add hero section from global hero data if available (support activities/booking variants) - hero: heroData && Array.isArray(heroData) && heroData.length > 0 ? { - titleActivities: heroData[0].titleActivities || heroData[0].title || "", - titleBooking: heroData[0].titleBooking || heroData[0].title || "", - bannerImageActivities: heroData[0].bannerImageActivities || heroData[0].bannerImage || "", - bannerImageBooking: heroData[0].bannerImageBooking || heroData[0].bannerImage || "", - } : { - titleActivities: "", - titleBooking: "", - bannerImageActivities: "", - bannerImageBooking: "", - }, - name: source.name || "", - price: source.price || 0, - priceText: source.priceText || `from ${source.price || 0} USD`, - season: Array.isArray(source.season) ? source.season : [], - age: - Array.isArray(source.age) && source.age.length === 2 - ? source.age - : [12, 18], - locations: Array.isArray(source.locations) ? source.locations : [], - image: source.image || "", - link: source.link || "", - program: source.program || "", - rating: source.rating || 4, - isActive: typeof source.isActive === 'boolean' ? source.isActive : true, - order: typeof source.order === 'number' ? source.order : index, - // Keep the rich camp detail under a schema-friendly key - campDetail: source['camp-detail'] || source.campDetail || {}, - }; -} - -/** - * Migration: activities - * Import activities from data/activities.json into MongoDB - */ -async function migrate() { - try { - await connectDB(); - console.log("Starting migration: activities..."); - - // Read data file - const dataPath = path.join(__dirname, "../data/activities.json"); - console.log(`Reading data from ${dataPath}...`); - - // Use fs.existsSync and fs.readFileSync for synchronous check and read - const fsSync = require("fs"); - if (!fsSync.existsSync(dataPath)) { - throw new Error("Data file not found!"); - } - - const rawData = fsSync.readFileSync(dataPath, "utf8"); - const data = JSON.parse(rawData); - - // Handle new data structure - const activitiesData = Array.isArray(data) ? data : data.camps || []; - const filtersData = Array.isArray(data) ? [] : data.filter || []; - const heroData = Array.isArray(data) ? null : data.hero || null; - - console.log( - `Found ${activitiesData.length} activities and ${filtersData.length} filter groups to migrate.` - ); - - // --- Migrate Activities --- - if (activitiesData.length > 0) { - console.log("Migrating activities..."); - - // Transform data if needed (using the existing transformActivity for consistency, or a new one if structure changed) - const activitiesToInsert = activitiesData.map( - (source, index) => transformActivity(source, index, heroData) // Pass heroData to transform function - ); - - const insertedActivities = await Activity.insertMany(activitiesToInsert, { - ordered: false, - }); - console.log(`Inserted ${insertedActivities.length} activities.`); - } else { - console.log("No activities to migrate."); - } - - // --- Migrate Filters --- - if (filtersData.length > 0) { - console.log("Migrating activity filters..."); - - // Deduplicate filters by value (value must be unique per model) - const seen = new Map(); - const filtersToUpsert = []; - filtersData.forEach((item, index) => { - // sanitize incoming filter items (remove any unexpected keys such as `count`) - const sanitizeItems = (arr) => - (Array.isArray(arr) ? arr : []) - .map((it) => ({ - value: (it && it.value) ? it.value.toString().trim() : "", - label: (it && it.label) ? it.label.toString().trim() : "", - })) - .filter((it) => it.value && it.label); - - const f = { - label: item.label || item.name || `Filter ${index + 1}`, - value: (item.value || (item.label || item.name || `filter-${index + 1}`)) - .toString() - .trim(), - items: sanitizeItems(item.items), - order: item.order || index + 1, - }; - - if (!f.value) return; // skip invalid - - if (seen.has(f.value)) { - // merge items if duplicate in source (merge by `value`, prefer first occurrence) - const existing = seen.get(f.value); - const mergedMap = new Map(); - [...existing.items, ...f.items].forEach((it) => { - if (it && it.value) mergedMap.set(it.value, it); - }); - existing.items = Array.from(mergedMap.values()); - existing.order = Math.min(existing.order, f.order); - } else { - seen.set(f.value, f); - filtersToUpsert.push(f); - } - }); - - if (filtersToUpsert.length === 0) { - console.log("No valid activity filters to migrate after dedupe."); - } else { - // Use bulkWrite with upsert to avoid duplicate-key errors and to update existing docs - const bulkOps = filtersToUpsert.map((f) => ({ - updateOne: { - filter: { value: f.value }, - update: { $set: { label: f.label, items: f.items, order: f.order } }, - upsert: true, - }, - })); - - // Upsert the consolidated filters into a single Activity document - // that is used to store global filter definitions (marked by isFiltersDoc: true) - const upsertResult = await Activity.findOneAndUpdate( - { isFiltersDoc: true }, - { $set: { filters: filtersToUpsert, isFiltersDoc: true } }, - { upsert: true, new: true } - ); - console.log(`Upserted filters into Activity document id=${upsertResult._id} groups=${(upsertResult.filters || []).length}`); - } - } else { - console.log("No activity filters to migrate."); - } - - console.log("Migration activities completed successfully!"); - - const mongoose = require("mongoose"); - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error("Migration error:", error); - - // If some documents failed but others succeeded, log partial success - if (error.insertedDocs && error.insertedDocs.length > 0) { - console.log( - `Partial success: ${error.insertedDocs.length} documents inserted` - ); - } - - process.exit(1); - } -} - -/** - * Rollback: Delete all activities (use with caution!) - */ -async function rollback() { - try { - await connectDB(); - console.log("Starting rollback..."); - - const actResult = await Activity.deleteMany({}); - console.log(`✅ Deleted ${actResult.deletedCount} activities`); - - // Remove any filters document stored as an Activity with isFiltersDoc=true - const filterResult = await Activity.deleteMany({ isFiltersDoc: true }); - console.log(`✅ Deleted ${filterResult.deletedCount} activity filters documents`); - - console.log("Rollback completed successfully!"); - - const mongoose = require("mongoose"); - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error("Rollback error:", error); - process.exit(1); - } -} - -// Run migration or rollback based on command line arguments -if (require.main === module) { - const args = process.argv.slice(2); - - if (args.includes("--rollback")) { - rollback(); - } else { - migrate(); - } -} - -module.exports = {migrate, rollback}; diff --git a/scripts/2025_12_10_migrate_home.js b/scripts/2025_12_10_migrate_home.js deleted file mode 100644 index d5f182c..0000000 --- a/scripts/2025_12_10_migrate_home.js +++ /dev/null @@ -1,192 +0,0 @@ -require('dotenv').config(); -const fs = require('fs').promises; -const path = require('path'); -const mongoose = require('mongoose'); -const Home = require('../models/home'); // Đảm bảo đường dẫn đúng tới file model - -// 1. Đọc file JSON -async function loadHomeData() { - // Đảm bảo đường dẫn đúng tới file json - const filePath = path.join(__dirname, '..', 'data', 'home.json'); - const raw = await fs.readFile(filePath, 'utf8'); - return JSON.parse(raw); -} - -// 2. Hàm Transform: Đổ dữ liệu từ JSON (source) vào đúng Schema -function transformHome(source) { - return { - // --- Hero Section --- - hero: { - title: source.hero?.title || "", - description: source.hero?.description || "", - backgroundImage: source.hero?.backgroundImage || "", - button: { - label: source.hero?.button?.label || "Book Now", - href: source.hero?.button?.href || "/booking" - }, - contactBox: { - welcomeText: source.hero?.contactBox?.welcomeText || "", - phone: { - label: source.hero?.contactBox?.phone?.label || "Call us", - number: source.hero?.contactBox?.phone?.number || "", - href: source.hero?.contactBox?.phone?.href || "" - }, - email: { - label: source.hero?.contactBox?.email?.label || "Email", - address: source.hero?.contactBox?.email?.address || "", - href: source.hero?.contactBox?.email?.href || "" - }, - workingHours: { - label: source.hero?.contactBox?.workingHours?.label || "Working Hours", - hours: source.hero?.contactBox?.workingHours?.hours || "" - } - } - }, - - // --- About Section --- - about: { - title: source.about?.title || "", - subtitle: source.about?.subtitle || "", - description: source.about?.description || "", - images: { - mainImage1: source.about?.images?.mainImage1 || "", - mainImage2: source.about?.images?.mainImage2 || "", - avatars: Array.isArray(source.about?.images?.avatars) ? source.about.images.avatars : [] - }, - features: Array.isArray(source.about?.features) ? source.about.features : [], - quote: source.about?.quote || "", - button: { - label: source.about?.button?.label || "", - href: source.about?.button?.href || "" - }, - stats: { - customerCount: source.about?.stats?.customerCount || 0, - customerLabel: source.about?.stats?.customerLabel || "" - } - }, - - // --- Mission & Vision --- - missionVision: { - title: source.missionVision?.title || "", - subtitle: source.missionVision?.subtitle || "", - backgroundImage: source.missionVision?.backgroundImage || "", - cards: Array.isArray(source.missionVision?.cards) ? source.missionVision.cards : [] - }, - - // --- Why Choose Us --- - whyChooseUs: { - title: source.whyChooseUs?.title || "", - subtitle: source.whyChooseUs?.subtitle || "", - description: source.whyChooseUs?.description || "", - button: source.whyChooseUs?.button || {}, - features: Array.isArray(source.whyChooseUs?.features) ? source.whyChooseUs.features : [], - tags: Array.isArray(source.whyChooseUs?.tags) ? source.whyChooseUs.tags : [], - cta: source.whyChooseUs?.cta || {} - }, - - // --- Activities --- - activities: { - cards: Array.isArray(source.activities?.cards) ? source.activities.cards : [] - }, - - // --- FAQ --- - faq: { - title: source.faq?.title || "", - subtitle: source.faq?.subtitle || "", - description: source.faq?.description || "", - image: source.faq?.image || "", - contact: source.faq?.contact || {}, - questions: Array.isArray(source.faq?.questions) ? source.faq.questions : [] - }, - - // --- Partners --- - partners: { - title: source.partners?.title || "", - subtitle: source.partners?.subtitle || "", - backgroundImage: source.partners?.backgroundImage || "", - logos: Array.isArray(source.partners?.logos) ? source.partners.logos : [], - cta: source.partners?.cta || {} - }, - - // --- Programs --- - programs: { - title: source.programs?.title || "", - subtitle: source.programs?.subtitle || "", - button: source.programs?.button || {}, - card: { - pricePrefix: source.programs?.card?.pricePrefix || "from", - priceSuffix: source.programs?.card?.priceSuffix || "USD", - buttonLabel: source.programs?.card?.buttonLabel || "Camp Detail", - buttonHref: source.programs?.card?.buttonHref || "/camp-profiles" - }, - items: Array.isArray(source.programs?.items) ? source.programs.items : [] - }, - - - - // --- Newsletter --- - newsletter: { - title: source.newsletter?.title || "", - subtitle: source.newsletter?.subtitle || "", - description: source.newsletter?.description || "", - image: source.newsletter?.image || "", - decorativeImage: source.newsletter?.decorativeImage || "", - button: { - label: source.newsletter?.button?.label || "", - placeholder: source.newsletter?.button?.placeholder || "", - href: source.newsletter?.button?.href || "" - } - }, - - // --- Latest Posts --- - latestPosts: { - title: source.latestPosts?.title || "", - subtitle: source.latestPosts?.subtitle || "", - searchPlaceholder: source.latestPosts?.searchPlaceholder || "", - sidebarTitle: source.latestPosts?.sidebarTitle || "", - blogPosts: Array.isArray(source.latestPosts?.blogPosts) ? source.latestPosts.blogPosts : [], - sidebarPosts: Array.isArray(source.latestPosts?.sidebarPosts) ? source.latestPosts.sidebarPosts : [], - featuredCard: source.latestPosts?.featuredCard || {} - }, - - updatedAt: new Date() - }; -} - -// 3. Chạy Migration -async function migrate() { - try { - // Kết nối DB - await mongoose.connect(process.env.MONGODB_URI); - console.log('✅ Connected to MongoDB'); - - // A. Lấy dữ liệu thô - const rawData = await loadHomeData(); - console.log('📖 Data loaded from JSON'); - - // B. Chuẩn hóa dữ liệu theo Schema - const homeData = transformHome(rawData); - - // C. Lưu vào DB (Upsert: Có rồi thì update, chưa có thì tạo) - const existingDoc = await Home.findOne().sort({ updatedAt: -1 }); - - if (existingDoc) { - console.log('📝 Updating existing Home document...'); - await Home.findByIdAndUpdate(existingDoc._id, { $set: homeData }, { new: true }); - } else { - console.log('📝 Creating NEW Home document...'); - await Home.create(homeData); - } - - console.log('✨ Migration completed successfully!'); - - } catch (error) { - console.error('❌ Migration failed:', error); - process.exit(1); - } finally { - await mongoose.connection.close(); - process.exit(0); - } -} - -migrate(); \ No newline at end of file diff --git a/scripts/2025_12_11_075500_booking.js b/scripts/2025_12_11_075500_booking.js deleted file mode 100644 index 2bc3e72..0000000 --- a/scripts/2025_12_11_075500_booking.js +++ /dev/null @@ -1,178 +0,0 @@ -const mongoose = require('mongoose'); -const Booking = require('../models/booking'); -const fs = require('fs'); -const path = require('path'); -require('dotenv').config(); - - - -const filePath = path.join(__dirname, '..', 'data', 'booking.json'); -let raw = '{}'; -try { - raw = fs.readFileSync(filePath, 'utf8'); -} catch (e) { - console.error('Could not read booking.json at', filePath); - process.exit(2); -} - -let data; -try { - data = JSON.parse(raw || '{}'); -} catch (e) { - console.error('Invalid JSON in booking.json:', e.message); - process.exit(3); -} - -// Remove _id fields recursively to avoid conflicts -function stripIds(obj) { - if (Array.isArray(obj)) return obj.map(i => stripIds(i)); - if (obj && typeof obj === 'object') { - const out = {}; - for (const k in obj) { - if (k !== '_id') out[k] = stripIds(obj[k]); - } - return out; - } - return obj; -} - -data = stripIds(data); - -// Normalize vouchers to an array of objects so Mongoose casting won't fail -function normalizeVouchers(doc) { - if (!doc) return; - // support root-level `vouchers` and `configuration.vouchers` - let v = doc.vouchers || (doc.configuration && doc.configuration.vouchers); - if (!v) return; - - // Try to parse stringified arrays (may use single quotes or JS literal) - if (typeof v === 'string') { - try { - v = JSON.parse(v); - } catch (e1) { - try { - // try parsing JS object-literal style strings - // eslint-disable-next-line no-new-func - v = (new Function('return ' + v))(); - } catch (e2) { - // fallback: attempt to extract codes by splitting on commas - v = v.split && v.split(',').map(s => s.trim()).filter(Boolean).map(s => ({ validCodes: s, type: 'unknown', value: null })); - } - } - } - - // Normalize array items into objects - if (Array.isArray(v)) { - v = v.map(item => { - if (typeof item === 'string') return { validCodes: item, type: 'unknown', value: null }; - if (item && typeof item === 'object') { - return { - validCodes: item.validCodes || item.code || '', - type: item.type || '', - value: typeof item.value === 'number' ? item.value : (item.amount && Number(item.amount)) || null, - }; - } - return item; - }); - } - - // If Booking schema expects array of strings, convert objects -> string codes - try { - const Booking = require('../models/booking'); - const pathType = Booking.schema.path('vouchers') || Booking.schema.path('configuration.vouchers'); - if (pathType && pathType.instance === 'Array' && pathType.caster && pathType.caster.instance === 'String') { - const mapped = (v || []).map(it => (typeof it === 'string' ? it : (it && typeof it === 'object' ? it.validCodes || JSON.stringify(it) : String(it)))); - if (doc.vouchers) doc.vouchers = mapped; - else if (doc.configuration) doc.configuration.vouchers = mapped; - return; - } - } catch (e) { - // ignore and keep object form - } - - if (doc.vouchers) doc.vouchers = v; - else if (doc.configuration) doc.configuration.vouchers = v; -} - -normalizeVouchers(data); - -// Also normalize discounts (some inputs have them stringified or as objects) -function normalizeDiscounts(doc) { - if (!doc) return; - let d = doc.discounts || (doc.configuration && doc.configuration.discounts); - if (!d) return; - - if (typeof d === 'string') { - try { - d = JSON.parse(d); - } catch (e1) { - try { - // eslint-disable-next-line no-new-func - d = (new Function('return ' + d))(); - } catch (e2) { - d = d.split && d.split('\n').map(s => s.trim()).filter(Boolean).map(s => ({ id: '', name: s })); - } - } - } - - if (Array.isArray(d)) { - d = d.map(item => { - if (typeof item === 'string') return { id: '', name: item }; - if (item && typeof item === 'object') return item; - return item; - }); - } - - try { - const Booking = require('../models/booking'); - const pathType = Booking.schema.path('discounts') || Booking.schema.path('configuration.discounts'); - if (pathType && pathType.instance === 'Array' && pathType.caster && pathType.caster.instance === 'String') { - const mapped = (d || []).map(it => (typeof it === 'string' ? it : (it.id || it.name || JSON.stringify(it)))); - if (doc.discounts) doc.discounts = mapped; - else if (doc.configuration) doc.configuration.discounts = mapped; - return; - } - } catch (e) { - // ignore - } - - if (doc.discounts) doc.discounts = d; - else if (doc.configuration) doc.configuration.discounts = d; -} - -normalizeDiscounts(data); - -const dryRun = process.argv.includes('--dry-run') || process.argv.includes('-n'); - -async function run() { - try { - const dbUri = process.env.MONGODB_URI || process.env.MONGO_URL || 'mongodb://127.0.0.1:27017/cms'; - console.log('Using DB URI:', dbUri); - - if (dryRun) { - console.log('\nDRY RUN - preview of document to upsert:\n'); - console.log(JSON.stringify(data, null, 2)); - process.exit(0); - } - - await mongoose.connect(dbUri, { useNewUrlParser: true, useUnifiedTopology: true }); - console.log('Connected to MongoDB'); - - const result = await Booking.findOneAndUpdate({}, data, { - upsert: true, - new: true, - setDefaultsOnInsert: true, - }); - - console.log('Upsert complete. Document id:', result._id.toString()); - console.log('Summary: programs=', (data.programs || []).length, 'camps=', (data.camps || []).length); - - await mongoose.disconnect(); - process.exit(0); - } catch (err) { - console.error('Migration failed:', err && err.message ? err.message : err); - process.exit(1); - } -} - -run(); diff --git a/scripts/2025_12_11_214026_migrate_header_ggcamp.js b/scripts/2025_12_11_214026_migrate_header_ggcamp.js deleted file mode 100644 index 0c36da9..0000000 --- a/scripts/2025_12_11_214026_migrate_header_ggcamp.js +++ /dev/null @@ -1,48 +0,0 @@ -require('dotenv').config(); -const connectDB = require('../config/database'); - -/** - * Migration: migrate_header_ggcamp - * Created: 21:40:26 11/12/2025 - */ -async function migrate() { - try { - // Kết nối database - await connectDB(); - console.log('Starting migration: migrate_header_ggcamp...'); - - const mongoose = require('mongoose'); - const fs = require('fs').promises; - const path = require('path'); - const Header = require('../models/header'); - - // Đọc dữ liệu từ header.json - const headerDataPath = path.join(__dirname, '../data/header.json'); - const headerData = JSON.parse(await fs.readFile(headerDataPath, 'utf8')); - - // Xóa tất cả documents header cũ - await Header.deleteMany({}); - - // Tạo header mới với dữ liệu từ JSON (topbar và logo) - await Header.create({ - name: 'default', - topbar: headerData.topbar, - logo: headerData.logo - }); - - console.log('✅ Header migration completed successfully!'); - - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error('❌ Migration error:', error); - process.exit(1); - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate }; diff --git a/scripts/2025_12_11_214038_migrate_menu_header_ggcamp.js b/scripts/2025_12_11_214038_migrate_menu_header_ggcamp.js deleted file mode 100644 index 347b136..0000000 --- a/scripts/2025_12_11_214038_migrate_menu_header_ggcamp.js +++ /dev/null @@ -1,57 +0,0 @@ -require('dotenv').config(); -const connectDB = require('../config/database'); - -/** - * Migration: migrate_menu_header_ggcamp - * Created: 21:40:38 11/12/2025 - */ -async function migrate() { - try { - // Kết nối database - await connectDB(); - console.log('Starting migration: migrate_menu_header_ggcamp...'); - - const mongoose = require('mongoose'); - const fs = require('fs').promises; - const path = require('path'); - const Menu = require('../models/menuHeader'); - - // Xóa tất cả dữ liệu menu cũ - await Menu.deleteMany({}); - - // Đọc JSON file - const jsonPath = path.join(__dirname, '../data/menu-header.json'); - const jsonData = JSON.parse(await fs.readFile(jsonPath, 'utf8')); - - // Tạo menu items (đơn giản, không có fetch/programmes) - for (const menuData of jsonData.menus) { - // Chỉ giữ lại các field cần thiết: menuid, parent, title, url, order, type - const menuItem = { - menuid: menuData.menuid, - parent: menuData.parent || null, - title: menuData.title, - url: menuData.url, - order: menuData.order || 0, - type: menuData.type || 'static', - fetch: false, // Không dùng fetch - isActive: true // Mặc định active - }; - await Menu.create(menuItem); - } - - console.log('✅ Menu header migration completed successfully!'); - - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error('❌ Migration error:', error); - process.exit(1); - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate }; diff --git a/scripts/2025_12_11_214103_migrate_footer_ggcamp.js b/scripts/2025_12_11_214103_migrate_footer_ggcamp.js deleted file mode 100644 index d0cefea..0000000 --- a/scripts/2025_12_11_214103_migrate_footer_ggcamp.js +++ /dev/null @@ -1,41 +0,0 @@ -require('dotenv').config(); -const connectDB = require('../config/database'); - -/** - * Migration: migrate_footer_ggcamp - * Created: 21:41:03 11/12/2025 - */ -async function migrate() { - try { - // Kết nối database - await connectDB(); - console.log('Starting migration: migrate_footer_ggcamp...'); - - const mongoose = require('mongoose'); - const fs = require('fs').promises; - const path = require('path'); - const Footer = require('../models/footer'); - - // Read footer.json file - const footerJsonPath = path.join(__dirname, '../data/footer.json'); - const footerData = JSON.parse(await fs.readFile(footerJsonPath, 'utf8')); - - // Migrate data using the model's static method - await Footer.migrateFromJson(footerData); - - console.log('✅ Footer migration completed successfully!'); - - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error('❌ Migration error:', error); - process.exit(1); - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate }; diff --git a/scripts/2025_12_13_travel.js b/scripts/2025_12_13_travel.js deleted file mode 100644 index f57470e..0000000 --- a/scripts/2025_12_13_travel.js +++ /dev/null @@ -1,273 +0,0 @@ -require('dotenv').config(); -const fs = require('fs').promises; -const path = require('path'); -const connectDB = require('../config/database'); -const Travel = require('../models/travel'); -const mongoose = require('mongoose'); - -/** - * Migration: travel - * Migrate Travel data từ travel.json - * Created: 2025-12-13 - */ -async function migrate() { - try { - // Kết nối database - await connectDB(); - console.log('Starting migration: travel...'); - - // Đọc file travel.json - const travelJsonPath = path.join(__dirname, '../data/travel.json'); - console.log('Reading JSON file from:', travelJsonPath); - - const travelData = JSON.parse(await fs.readFile(travelJsonPath, 'utf8')); - console.log('Travel data loaded successfully'); - console.log('Data structure keys:', Object.keys(travelData)); - - // Thực hiện migration - await migrateTravelData(travelData); - - console.log('Travel migration completed successfully!'); - - await mongoose.disconnect(); - process.exit(0); - } catch (error) { - console.error('Migration error:', error); - process.exit(1); - } -} - -/** - * Custom migration logic cho travel data - */ -async function migrateTravelData(travelData) { - try { - console.log('Starting custom migration logic...'); - - // 1. Kiểm tra dữ liệu cũ - const existingTravel = await Travel.findOne({}); - if (existingTravel) { - console.log(`Found existing travel document: ${existingTravel._id}`); - console.log('Deleting existing travel data...'); - await Travel.deleteMany({}); - console.log('Cleared existing travel data'); - } - - // 2. Chuyển đổi locations thành blog content blocks - const contentBlocks = []; - - // If travelData already has posts (blog format), use the first post - if (Array.isArray(travelData.posts) && travelData.posts.length > 0) { - const firstPost = travelData.posts[0]; - if (firstPost.content && Array.isArray(firstPost.content.blocks)) { - contentBlocks.push(...firstPost.content.blocks); - } - } else { - // Thêm general description (legacy format) - if (travelData.general) { - contentBlocks.push({ - type: 'paragraph', - data: { - text: travelData.general.description - } - }); - - // Thêm additional info như conclusion - if (travelData.general.additionalInfo) { - contentBlocks.push({ - type: 'conclusion', - data: { - text: travelData.general.additionalInfo, - callToAction: { - text: '', - link: '' - } - } - }); - } - } - - // Chuyển đổi từng location thành blog blocks - if (travelData.locations && Array.isArray(travelData.locations)) { - travelData.locations.forEach(location => { - // Header cho location - contentBlocks.push({ - type: 'header', - data: { - text: location.title, - level: 2 - } - }); - - // Address information - const addressItems = []; - if (location.address) { - if (location.address.name) { - addressItems.push(`Name: ${location.address.name}`); - } - if (location.address.line2) { - addressItems.push(location.address.line2); - } - if (location.address.street) { - addressItems.push(`Street: ${location.address.street}`); - } - if (location.address.postalCode && location.address.city) { - const country = location.address.country ? `, ${location.address.country}` : ''; - addressItems.push(`Location: ${location.address.postalCode} ${location.address.city}${country}`); - } - if (location.address.googleMapsUrl) { - addressItems.push(`Google Maps: View on Map`); - } - } - - if (addressItems.length > 0) { - contentBlocks.push({ - type: 'list', - data: { - style: 'unordered', - items: addressItems - } - }); - } - - // Address note as conclusion - if (location.address?.note) { - contentBlocks.push({ - type: 'conclusion', - data: { - text: location.address.note, - callToAction: { - text: '', - link: '' - } - } - }); - } - - // Contact information - const contactItems = []; - if (location.contact) { - if (location.contact.email) { - contactItems.push(`Email: ${location.contact.email}`); - } - if (location.contact.phone) { - contactItems.push(`Phone: ${location.contact.phone}`); - } - } - - if (contactItems.length > 0) { - contentBlocks.push({ - type: 'paragraph', - data: { - text: 'Contact Information:' - } - }); - contentBlocks.push({ - type: 'list', - data: { - style: 'unordered', - items: contactItems - } - }); - } - - // Schedule information - const scheduleItems = []; - if (location.schedule) { - if (location.schedule.arrival) { - scheduleItems.push(`Arrival: ${location.schedule.arrival}`); - } - if (location.schedule.departure) { - scheduleItems.push(`Departure: ${location.schedule.departure}`); - } - } - - if (scheduleItems.length > 0) { - contentBlocks.push({ - type: 'paragraph', - data: { - text: 'Schedule:' - } - }); - contentBlocks.push({ - type: 'list', - data: { - style: 'unordered', - items: scheduleItems - } - }); - } - }); - } - } - - // 3. Tạo document mới cho travel - const travelDocument = new Travel({ - hero: { - title: travelData.hero?.title || 'Travel Information', - backgroundImage: travelData.hero?.backgroundImage || '' - }, - - page: { - title: travelData.page?.title || travelData.general?.title || 'Go and Grow Camp Travel Information', - description: travelData.page?.description || travelData.general?.description || '', - year: travelData.page?.year || travelData.pageYear || undefined, - metadata: { - title: 'Travel Guide - Go and Grow Camp', - description: 'Everything you need to know about traveling to our camps' - } - }, - - content: { - blocks: contentBlocks - }, - - enableScrollspy: true, - lastUpdated: new Date() - }); - - // 4. Lưu vào database - await travelDocument.save(); - console.log('Travel document saved successfully'); - - // 5. Log thông tin - console.log(`Created travel document with ID: ${travelDocument._id}`); - console.log(`Hero title: ${travelDocument.hero.title}`); - console.log(`Page title: ${travelDocument.page.title}`); - console.log(`Content blocks count: ${travelDocument.content.blocks.length}`); - console.log(`Converted ${travelData.locations?.length || 0} locations to blog blocks`); - - } catch (error) { - console.error('Error in migrateTravelData:', error); - throw error; - } -} - -/** - * Hàm backup data trước khi migration - */ -async function backupExistingData() { - try { - console.log('Creating backup of existing travel data...'); - const existingTravel = await Travel.find({}); - - if (existingTravel.length > 0) { - const backupPath = path.join(__dirname, '../backups/travel-backup-' + Date.now() + '.json'); - - // Tạo thư mục backup nếu chưa có - await fs.mkdir(path.dirname(backupPath), { recursive: true }); - - await fs.writeFile(backupPath, JSON.stringify(existingTravel, null, 2)); - console.log(`Backup created at: ${backupPath}`); - } - } catch (error) { - console.error('Backup error:', error); - } -} - -// Chạy migration nếu được gọi trực tiếp -if (require.main === module) { - migrate(); -} - -module.exports = { migrate, migrateTravelData }; diff --git a/scripts/2026_02_02_170000_create_blog_system.js b/scripts/2026_02_02_170000_create_blog_system.js new file mode 100644 index 0000000..35e1f28 --- /dev/null +++ b/scripts/2026_02_02_170000_create_blog_system.js @@ -0,0 +1,182 @@ +require('dotenv').config(); +const fs = require('fs').promises; +const path = require('path'); +const connectDB = require('../config/database'); + +/** + * Migration: create_complete_blog_system + * Created: 17:00:00 2/2/2026 + * Description: Tạo hoàn chỉnh hệ thống blog với categories, tags, posts và comments + */ +async function migrate() { + try { + // Kết nối database + await connectDB(); + console.log('🚀 Starting migration: create_complete_blog_system...'); + + // Import models + const Blog = require('../models/blog'); + const BlogCategory = require('../models/blogCategory'); + const BlogTag = require('../models/blogTag'); + const BlogComment = require('../models/blogComment'); + const RecentPost = require('../models/recentPost'); + + console.log('✅ Blog models registered successfully'); + + // Load complete data + const dataPath = path.join(__dirname, '..', 'data', 'blog.json'); + const rawData = await fs.readFile(dataPath, 'utf8'); + const data = JSON.parse(rawData); + + console.log('📖 Complete blog data loaded from JSON'); + + // Clear existing data + console.log('🧹 Clearing existing blog data...'); + await BlogComment.deleteMany({}); + await Blog.deleteMany({}); + await BlogCategory.deleteMany({}); + await BlogTag.deleteMany({}); + await RecentPost.deleteMany({}); + console.log('✅ Existing data cleared'); + + // 1. Create categories + console.log('📝 Creating categories...'); + const createdCategories = []; + for (const categoryData of data.categories) { + const category = new BlogCategory(categoryData); + await category.save(); + createdCategories.push(category); + console.log(`✅ Created category: ${category.name}`); + } + + // 2. Create tags + console.log('📝 Creating tags...'); + const createdTags = []; + for (const tagData of data.tags) { + const tag = new BlogTag(tagData); + await tag.save(); + createdTags.push(tag); + console.log(`✅ Created tag: ${tag.name}`); + } + + // 3. Create blog posts + console.log('📝 Creating blog posts...'); + const createdPosts = []; + for (const postData of data.posts) { + const post = new Blog(postData); + await post.save(); + createdPosts.push(post); + console.log(`✅ Created blog post: ${post.title}`); + } + + // 4. Create comments + console.log('💬 Creating comments...'); + let createdCommentsCount = 0; + + for (const commentData of data.comments) { + // Find the blog post by slug + const blog = await Blog.findOne({ + slug: commentData.postSlug, + status: 'published' + }); + + if (blog) { + const comment = new BlogComment({ + postId: blog._id, + authorName: commentData.authorName, + authorAvatar: commentData.authorAvatar, + content: commentData.content, + createdAt: commentData.createdAt, + status: commentData.status + }); + + await comment.save(); + createdCommentsCount++; + console.log(`✅ Created comment by ${comment.authorName} for: ${blog.title}`); + } else { + console.log(`⚠️ Blog post not found for slug: ${commentData.postSlug}`); + } + } + + // 5. Update category post counts + console.log('📊 Updating category post counts...'); + for (const category of createdCategories) { + await category.updatePostCount(); + console.log(`📊 Category "${category.name}": ${category.postCount} posts`); + } + + // 6. Update tag post counts + console.log('📊 Updating tag post counts...'); + for (const tag of createdTags) { + await tag.updatePostCount(); + console.log(`📊 Tag "${tag.name}": ${tag.postCount} posts`); + } + + // 7. Update comments count in blog posts + console.log('📊 Updating comments count in blog posts...'); + const blogs = await Blog.find({ status: 'published' }); + + for (const blog of blogs) { + const commentsCount = await BlogComment.countDocuments({ + postId: blog._id, + status: 'approved' + }); + + blog.commentsCount = commentsCount; + await blog.save(); + + if (commentsCount > 0) { + console.log(`📊 Updated comments count for "${blog.title}": ${commentsCount} comments`); + } + } + + // 8. Sync recent posts + console.log('🔄 Syncing recent posts...'); + await RecentPost.syncFromBlogs(5); + const recentPostsCount = await RecentPost.countDocuments(); + console.log(`🔄 Synced ${recentPostsCount} recent posts`); + + // Final summary + console.log('\n🎉 Migration create_complete_blog_system completed successfully!'); + console.log('=' .repeat(60)); + console.log('📊 MIGRATION SUMMARY:'); + console.log(` ✅ Categories: ${createdCategories.length}`); + console.log(` ✅ Tags: ${createdTags.length}`); + console.log(` ✅ Blog Posts: ${createdPosts.length}`); + console.log(` ✅ Comments: ${createdCommentsCount}`); + console.log(` ✅ Recent Posts: ${recentPostsCount}`); + + // Statistics + const totalPublishedPosts = await Blog.countDocuments({ status: 'published' }); + const totalFeaturedPosts = await Blog.countDocuments({ status: 'published', isFeatured: true }); + const totalApprovedComments = await BlogComment.countDocuments({ status: 'approved' }); + + console.log('\n📈 SYSTEM STATISTICS:'); + console.log(` 📝 Published Posts: ${totalPublishedPosts}`); + console.log(` ⭐ Featured Posts: ${totalFeaturedPosts}`); + console.log(` 💬 Approved Comments: ${totalApprovedComments}`); + + console.log('\n🌐 ACCESS POINTS:'); + console.log(' 📱 Admin Panel: http://localhost:3001/admin/blog'); + console.log(' 🔗 API Endpoint: http://localhost:3001/api/blog'); + console.log(' 📊 Categories API: http://localhost:3001/api/blog-categories'); + console.log(' 🏷️ Tags API: http://localhost:3001/api/blog-tags'); + + console.log('\n✨ Blog system is now ready for use!'); + console.log('=' .repeat(60)); + + const mongoose = require('mongoose'); + await mongoose.disconnect(); + process.exit(0); + } catch (error) { + console.error('❌ Migration error:', error); + process.exit(1); + } +} + +// Chạy migration nếu được gọi trực tiếp +if (require.main === module) { + migrate(); +} + +module.exports = { migrate }; \ No newline at end of file