Files
ng-blog/internal/posts/controller.go
schreifuchs 893c49ec88
All checks were successful
Release / publish (push) Successful in 2m41s
feat: private posts
2025-10-17 23:59:34 +02:00

145 lines
3.4 KiB
Go

package posts
import (
"encoding/json"
"fmt"
"log"
"net/http"
"strconv"
"git.schreifuchs.ch/schreifuchs/ng-blog/internal/auth"
"git.schreifuchs.ch/schreifuchs/ng-blog/internal/model"
"github.com/google/uuid"
"github.com/gorilla/mux"
)
// SavePost handles saving a new post to the database after extracting user claims and decoding the request body.
func (s Service) SavePost(w http.ResponseWriter, r *http.Request) {
claims, ok := auth.ExtractClaims(r.Context())
if !ok {
log.Println("Err could not ExtractClaims")
w.WriteHeader(http.StatusInternalServerError)
return
}
var post model.Post
if err := json.NewDecoder(r.Body).Decode(&post); err != nil {
fmt.Fprint(w, err.Error())
w.WriteHeader(http.StatusBadRequest)
return
}
if post.Private {
secret := uuid.NewString()
post.Secret = &secret
} else {
post.Secret = nil
}
post.UserID = claims.UserID
if err := s.db.Save(&post).Error; err != nil {
fmt.Fprint(w, err.Error())
w.WriteHeader(http.StatusInternalServerError)
return
}
res, err := json.Marshal(&post)
if err != nil {
fmt.Fprint(w, err.Error())
w.WriteHeader(http.StatusInternalServerError)
return
}
w.Write(res)
}
// GetAllPosts retrieves all posts from the database, eager-loads comments, orders them by creation time, and returns them as JSON.
func (s Service) GetAllPosts(w http.ResponseWriter, r *http.Request) {
var posts []model.Post
claims, ok := auth.ExtractClaims(r.Context())
if !ok {
claims = nil
}
if err := s.db.Preload("Comments").Order("created_at DESC").Find(&posts).Error; err != nil {
fmt.Fprint(w, err.Error())
w.WriteHeader(http.StatusInternalServerError)
return
}
newPosts := make([]model.Post, 0, len(posts))
for _, p := range posts {
if p.Secret == nil {
newPosts = append(newPosts, p)
continue
}
if claims == nil {
continue
}
if claims.UserID == p.UserID {
newPosts = append(newPosts, p)
}
}
res, err := json.Marshal(&newPosts)
if err != nil {
fmt.Fprint(w, err.Error())
w.WriteHeader(http.StatusInternalServerError)
return
}
w.Write(res)
}
// GetPostBySecret retrieves a post by its secret from the database, eager-loads comments, and returns it as JSON.
func (s Service) GetPostBySecret(w http.ResponseWriter, r *http.Request) {
secret, ok := mux.Vars(r)["secret"]
if !ok {
w.WriteHeader(http.StatusNotFound)
return
}
var post model.Post
if err := s.db.Preload("Comments").Where("secret = ?", secret).First(&post).Error; err != nil {
fmt.Fprint(w, err.Error())
w.WriteHeader(http.StatusNotFound)
return
}
res, err := json.Marshal(&post)
if err != nil {
fmt.Fprint(w, err.Error())
w.WriteHeader(http.StatusInternalServerError)
return
}
w.Write(res)
}
// DeletePost handles deleting a post from the database based on its ID and user authentication.
func (s Service) DeletePost(w http.ResponseWriter, r *http.Request) {
idStr, ok := mux.Vars(r)["postID"]
if !ok {
w.WriteHeader(http.StatusNotFound)
return
}
id, err := strconv.Atoi(idStr)
if err != nil {
w.WriteHeader(http.StatusNotFound)
return
}
claims, ok := auth.ExtractClaims(r.Context())
if !ok {
log.Println("Err could not ExtractClaims")
w.WriteHeader(http.StatusInternalServerError)
return
}
err = s.db.Where("user_id = ?", claims.UserID).Delete(&model.Post{}, id).Error
if err != nil {
fmt.Fprint(w, err.Error())
w.WriteHeader(http.StatusInternalServerError)
}
w.WriteHeader(http.StatusNoContent)
}