Abuse report system

This commit is contained in:
LittleSheep 2024-09-19 21:02:21 +08:00
parent 5bcce9d44e
commit 817c60c4e0
5 changed files with 183 additions and 0 deletions

View File

@ -27,6 +27,7 @@ var AutoMaintainRange = []any{
&models.ApiKey{}, &models.ApiKey{},
&models.SignRecord{}, &models.SignRecord{},
&models.PreferenceNotification{}, &models.PreferenceNotification{},
&models.AbuseReport{},
} }
func RunMigration(source *gorm.DB) error { func RunMigration(source *gorm.DB) error {

View File

@ -0,0 +1,19 @@
package models
const (
ReportStatusPending = "pending"
ReportStatusReviewing = "reviewing"
ReportStatusConfirmed = "confirmed"
ReportStatusRejected = "rejected"
ReportStatusProcessed = "processed"
)
type AbuseReport struct {
BaseModel
Resource string `json:"resource"`
Reason string `json:"reason"`
Status string `json:"status"`
AccountID uint `json:"account_id"`
Account Account `json:"account"`
}

View File

@ -30,6 +30,17 @@ func MapAPIs(app *fiber.App, baseURL string) {
preferences.Put("/notifications", updateNotificationPreference) preferences.Put("/notifications", updateNotificationPreference)
} }
reports := api.Group("/reports").Name("Reports API")
{
abuse := reports.Group("/abuse").Name("Abuse Reports")
{
abuse.Get("/", listAbuseReports)
abuse.Get("/:id", getAbuseReport)
abuse.Put("/:id/status", updateAbuseReportStatus)
abuse.Post("/", createAbuseReport)
}
}
api.Get("/users/lookup", lookupAccount) api.Get("/users/lookup", lookupAccount)
api.Get("/users/search", searchAccount) api.Get("/users/search", searchAccount)

View File

@ -0,0 +1,77 @@
package api
import (
"git.solsynth.dev/hydrogen/passport/pkg/internal/models"
"git.solsynth.dev/hydrogen/passport/pkg/internal/server/exts"
"git.solsynth.dev/hydrogen/passport/pkg/internal/services"
"github.com/gofiber/fiber/v2"
)
func listAbuseReports(c *fiber.Ctx) error {
if err := exts.EnsureAuthenticated(c); err != nil {
return err
}
user := c.Locals("user").(models.Account)
reports, err := services.ListAbuseReport(user)
if err != nil {
return fiber.NewError(fiber.StatusBadRequest, err.Error())
}
return c.JSON(reports)
}
func getAbuseReport(c *fiber.Ctx) error {
id, _ := c.ParamsInt("id")
report, err := services.GetAbuseReport(uint(id))
if err != nil {
return fiber.NewError(fiber.StatusNotFound, err.Error())
}
return c.JSON(report)
}
func updateAbuseReportStatus(c *fiber.Ctx) error {
if err := exts.EnsureGrantedPerm(c, "DealAbuseReport", true); err != nil {
return err
}
var data struct {
Status string `json:"status" validate:"required"`
}
if err := exts.BindAndValidate(c, &data); err != nil {
return err
}
id, _ := c.ParamsInt("id")
if err := services.UpdateAbuseReportStatus(uint(id), data.Status); err != nil {
return fiber.NewError(fiber.StatusBadRequest, err.Error())
}
return c.SendStatus(fiber.StatusOK)
}
func createAbuseReport(c *fiber.Ctx) error {
if err := exts.EnsureAuthenticated(c); err != nil {
return err
}
user := c.Locals("user").(models.Account)
var data struct {
Resource string `json:"resource" validate:"required"`
Reason string `json:"reason" validate:"required,min=16,max=4096"`
}
if err := exts.BindAndValidate(c, &data); err != nil {
return err
}
report, err := services.NewAbuseReport(data.Resource, data.Reason, user)
if err != nil {
return fiber.NewError(fiber.StatusBadRequest, err.Error())
}
return c.JSON(report)
}

View File

@ -0,0 +1,75 @@
package services
import (
"fmt"
"git.solsynth.dev/hydrogen/passport/pkg/internal/database"
"git.solsynth.dev/hydrogen/passport/pkg/internal/models"
)
func ListAbuseReport(account models.Account) ([]models.AbuseReport, error) {
var reports []models.AbuseReport
err := database.C.
Where("account_id = ?", account.ID).
Find(&reports).Error
return reports, err
}
func GetAbuseReport(id uint) (models.AbuseReport, error) {
var report models.AbuseReport
err := database.C.
Where("id = ?", id).
First(&report).Error
return report, err
}
func UpdateAbuseReportStatus(id uint, status string) error {
var report models.AbuseReport
err := database.C.
Where("id = ?", id).
Preload("Account").
First(&report).Error
if err != nil {
return err
}
report.Status = status
account := report.Account
err = database.C.Save(&report).Error
if err != nil {
return err
}
NewNotification(models.Notification{
Topic: "reports.feedback",
Title: "Abuse report status has been changed.",
Body: fmt.Sprintf("The report created by you with ID #%d's status has been changed to %s", id, status),
Account: account,
AccountID: account.ID,
})
return nil
}
func NewAbuseReport(resource string, reason string, account models.Account) (models.AbuseReport, error) {
var report models.AbuseReport
if err := database.C.
Where(
"resource = ? AND account_id = ? AND status IN ?",
resource,
account.ID,
[]string{models.ReportStatusPending, models.ReportStatusReviewing},
).First(&report).Error; err == nil {
return report, fmt.Errorf("you already reported this resource and it still in process")
}
report = models.AbuseReport{
Resource: resource,
Reason: reason,
AccountID: account.ID,
}
err := database.C.Create(&report).Error
return report, err
}