Files
adopt-a-street/backend/models/Task.js
William Valentin 7c70a8d098 feat(backend): implement comments, image uploads, and data consistency
Implement additional backend features and improve data models:

Comments System:
- Create Comment model with user and post relationships
- Add comments routes: GET /api/posts/:postId/comments (paginated), POST (create), DELETE (own comments)
- Update Post model with commentsCount field
- Emit Socket.IO events for newComment and commentDeleted
- Pagination support for comment lists
- Authorization checks (users can only delete own comments)
- 500 character limit on comments

Image Upload System:
- Implement Cloudinary configuration (config/cloudinary.js)
- Add uploadImage() and deleteImage() helper functions
- Image optimization: max 1000x1000, auto quality, auto format (WebP)
- Integrate image upload in users routes (profile pictures)
- Integrate image upload in posts routes (post images with add/update endpoints)
- File validation: 5MB limit, JPG/PNG/GIF/WebP only
- Automatic image deletion when removing posts/reports

Data Consistency Improvements:
- Add cascade deletes in Street model (remove from user, delete associated tasks)
- Add cascade deletes in Task model (remove from user completedTasks)
- Add cascade deletes in Post model (remove from user posts)
- Update user relationships on save (adoptedStreets, completedTasks, posts, events)
- Add proper indexes for performance (2dsphere for location, compound indexes)
- Add virtual relationships and toJSON configurations

Model Updates:
- Street: Add cascade hooks, location 2dsphere index
- Task: Add cascade hooks, compound indexes for queries
- Post: Add imageUrl, cloudinaryPublicId, commentsCount fields
- Event: Add participants tracking
- Report: Add image upload support
- User: Add earnedBadges virtual, profilePicture, cloudinaryPublicId

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-01 10:43:08 -07:00

63 lines
1.4 KiB
JavaScript

const mongoose = require("mongoose");
const TaskSchema = new mongoose.Schema(
{
street: {
type: mongoose.Schema.Types.ObjectId,
ref: "Street",
required: true,
index: true,
},
description: {
type: String,
required: true,
},
completedBy: {
type: mongoose.Schema.Types.ObjectId,
ref: "User",
index: true,
},
status: {
type: String,
enum: ["pending", "completed"],
default: "pending",
index: true,
},
},
{
timestamps: true,
},
);
// Compound indexes for common queries
TaskSchema.index({ street: 1, status: 1 });
TaskSchema.index({ completedBy: 1, status: 1 });
// Update user relationship when task is completed
TaskSchema.post("save", async function (doc) {
if (doc.completedBy && doc.status === "completed") {
const User = mongoose.model("User");
// Add task to user's completedTasks if not already there
await User.updateOne(
{ _id: doc.completedBy },
{ $addToSet: { completedTasks: doc._id } }
);
}
});
// Cascade cleanup when a task is deleted
TaskSchema.pre("deleteOne", { document: true, query: false }, async function () {
const User = mongoose.model("User");
// Remove task from user's completedTasks
if (this.completedBy) {
await User.updateOne(
{ _id: this.completedBy },
{ $pull: { completedTasks: this._id } }
);
}
});
module.exports = mongoose.model("Task", TaskSchema);