✨ Question APIs
This commit is contained in:
parent
9d9b2ac866
commit
57b908e5a0
@ -142,6 +142,7 @@ func editArticle(c *fiber.Ctx) error {
|
|||||||
if err := database.C.Where(models.Post{
|
if err := database.C.Where(models.Post{
|
||||||
BaseModel: cruda.BaseModel{ID: uint(id)},
|
BaseModel: cruda.BaseModel{ID: uint(id)},
|
||||||
PublisherID: publisher.ID,
|
PublisherID: publisher.ID,
|
||||||
|
Type: models.PostTypeArticle,
|
||||||
}).Preload("Publisher").First(&item).Error; err != nil {
|
}).Preload("Publisher").First(&item).Error; err != nil {
|
||||||
return fiber.NewError(fiber.StatusNotFound, err.Error())
|
return fiber.NewError(fiber.StatusNotFound, err.Error())
|
||||||
}
|
}
|
||||||
|
@ -36,6 +36,12 @@ func MapAPIs(app *fiber.App, baseURL string) {
|
|||||||
articles.Post("/", createArticle)
|
articles.Post("/", createArticle)
|
||||||
articles.Put("/:postId", editArticle)
|
articles.Put("/:postId", editArticle)
|
||||||
}
|
}
|
||||||
|
questions := api.Group("/questions").Name("Question API")
|
||||||
|
{
|
||||||
|
questions.Post("/", createQuestion)
|
||||||
|
questions.Put("/:postId", editQuestion)
|
||||||
|
questions.Put("/:postId/answer", selectQuestionAnswer)
|
||||||
|
}
|
||||||
|
|
||||||
posts := api.Group("/posts").Name("Posts API")
|
posts := api.Group("/posts").Name("Posts API")
|
||||||
{
|
{
|
||||||
|
316
pkg/internal/http/api/questions_api.go
Normal file
316
pkg/internal/http/api/questions_api.go
Normal file
@ -0,0 +1,316 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"git.solsynth.dev/hypernet/interactive/pkg/internal/database"
|
||||||
|
"git.solsynth.dev/hypernet/interactive/pkg/internal/gap"
|
||||||
|
"git.solsynth.dev/hypernet/interactive/pkg/internal/http/exts"
|
||||||
|
"git.solsynth.dev/hypernet/interactive/pkg/internal/models"
|
||||||
|
"git.solsynth.dev/hypernet/interactive/pkg/internal/services"
|
||||||
|
"git.solsynth.dev/hypernet/nexus/pkg/nex/cruda"
|
||||||
|
"git.solsynth.dev/hypernet/nexus/pkg/nex/sec"
|
||||||
|
"git.solsynth.dev/hypernet/passport/pkg/authkit"
|
||||||
|
authm "git.solsynth.dev/hypernet/passport/pkg/authkit/models"
|
||||||
|
wproto "git.solsynth.dev/hypernet/wallet/pkg/proto"
|
||||||
|
"github.com/gofiber/fiber/v2"
|
||||||
|
jsoniter "github.com/json-iterator/go"
|
||||||
|
"github.com/samber/lo"
|
||||||
|
"strconv"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func createQuestion(c *fiber.Ctx) error {
|
||||||
|
if err := sec.EnsureGrantedPerm(c, "CreatePosts", true); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
user := c.Locals("user").(authm.Account)
|
||||||
|
|
||||||
|
var data struct {
|
||||||
|
Publisher uint `json:"publisher"`
|
||||||
|
Alias *string `json:"alias"`
|
||||||
|
Title *string `json:"title"`
|
||||||
|
Content string `json:"content" validate:"max=4096"`
|
||||||
|
Location *string `json:"location"`
|
||||||
|
Thumbnail *string `json:"thumbnail"`
|
||||||
|
Attachments []string `json:"attachments"`
|
||||||
|
Tags []models.Tag `json:"tags"`
|
||||||
|
Categories []models.Category `json:"categories"`
|
||||||
|
PublishedAt *time.Time `json:"published_at"`
|
||||||
|
PublishedUntil *time.Time `json:"published_until"`
|
||||||
|
VisibleUsers []uint `json:"visible_users_list"`
|
||||||
|
InvisibleUsers []uint `json:"invisible_users_list"`
|
||||||
|
Visibility *int8 `json:"visibility"`
|
||||||
|
IsDraft bool `json:"is_draft"`
|
||||||
|
Reward float64 `json:"reward"`
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := exts.BindAndValidate(c, &data); err != nil {
|
||||||
|
return err
|
||||||
|
} else if len(data.Content) == 0 && len(data.Attachments) == 0 {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, "content or attachments are required")
|
||||||
|
}
|
||||||
|
|
||||||
|
publisher, err := services.GetPublisher(data.Publisher, user.ID)
|
||||||
|
if err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Take charge
|
||||||
|
if data.Reward > 0 {
|
||||||
|
conn, err := gap.Nx.GetClientGrpcConn("wa")
|
||||||
|
if err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusInternalServerError, fmt.Sprintf("unable to connect Wallet: %v", err))
|
||||||
|
}
|
||||||
|
wc := wproto.NewPaymentServiceClient(conn)
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second*3)
|
||||||
|
defer cancel()
|
||||||
|
if _, err := wc.MakeTransactionWithAccount(ctx, &wproto.MakeTransactionWithAccountRequest{
|
||||||
|
Amount: data.Reward,
|
||||||
|
Remark: "As reward for posting a question",
|
||||||
|
PayerAccountId: lo.ToPtr(uint64(user.ID)),
|
||||||
|
}); err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusPaymentRequired, fmt.Sprintf("failed to handle payment: %v", err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
body := models.PostQuestionBody{
|
||||||
|
PostStoryBody: models.PostStoryBody{
|
||||||
|
Thumbnail: data.Thumbnail,
|
||||||
|
Title: data.Title,
|
||||||
|
Content: data.Content,
|
||||||
|
Location: data.Location,
|
||||||
|
Attachments: data.Attachments,
|
||||||
|
},
|
||||||
|
Reward: data.Reward,
|
||||||
|
}
|
||||||
|
|
||||||
|
var bodyMapping map[string]any
|
||||||
|
rawBody, _ := jsoniter.Marshal(body)
|
||||||
|
_ = jsoniter.Unmarshal(rawBody, &bodyMapping)
|
||||||
|
|
||||||
|
item := models.Post{
|
||||||
|
Alias: data.Alias,
|
||||||
|
Type: models.PostTypeQuestion,
|
||||||
|
Body: bodyMapping,
|
||||||
|
Language: services.DetectLanguage(data.Content),
|
||||||
|
Tags: data.Tags,
|
||||||
|
Categories: data.Categories,
|
||||||
|
PublishedAt: data.PublishedAt,
|
||||||
|
PublishedUntil: data.PublishedUntil,
|
||||||
|
IsDraft: data.IsDraft,
|
||||||
|
VisibleUsers: data.VisibleUsers,
|
||||||
|
InvisibleUsers: data.InvisibleUsers,
|
||||||
|
PublisherID: publisher.ID,
|
||||||
|
}
|
||||||
|
|
||||||
|
if item.PublishedAt == nil {
|
||||||
|
item.PublishedAt = lo.ToPtr(time.Now())
|
||||||
|
}
|
||||||
|
|
||||||
|
if data.Visibility != nil {
|
||||||
|
item.Visibility = *data.Visibility
|
||||||
|
} else {
|
||||||
|
item.Visibility = models.PostVisibilityAll
|
||||||
|
}
|
||||||
|
|
||||||
|
item, err = services.NewPost(publisher, item)
|
||||||
|
if err != nil {
|
||||||
|
// Failed to create post, refund the charge
|
||||||
|
if data.Reward > 0 {
|
||||||
|
conn, err := gap.Nx.GetClientGrpcConn("wa")
|
||||||
|
if err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusInternalServerError, fmt.Sprintf("unable to connect Wallet: %v", err))
|
||||||
|
}
|
||||||
|
wc := wproto.NewPaymentServiceClient(conn)
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second*3)
|
||||||
|
defer cancel()
|
||||||
|
if _, err := wc.MakeTransactionWithAccount(ctx, &wproto.MakeTransactionWithAccountRequest{
|
||||||
|
Amount: data.Reward,
|
||||||
|
Remark: "As reward for posting a question - Refund",
|
||||||
|
PayeeAccountId: lo.ToPtr(uint64(user.ID)),
|
||||||
|
}); err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusPaymentRequired, fmt.Sprintf("failed to handle payment: %v", err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||||
|
} else {
|
||||||
|
_ = authkit.AddEventExt(
|
||||||
|
gap.Nx,
|
||||||
|
"posts.new",
|
||||||
|
strconv.Itoa(int(item.ID)),
|
||||||
|
c,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.JSON(item)
|
||||||
|
}
|
||||||
|
|
||||||
|
func editQuestion(c *fiber.Ctx) error {
|
||||||
|
id, _ := c.ParamsInt("postId", 0)
|
||||||
|
if err := sec.EnsureAuthenticated(c); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
user := c.Locals("user").(authm.Account)
|
||||||
|
|
||||||
|
var data struct {
|
||||||
|
Publisher uint `json:"publisher"`
|
||||||
|
Alias *string `json:"alias"`
|
||||||
|
Title *string `json:"title"`
|
||||||
|
Content string `json:"content" validate:"max=4096"`
|
||||||
|
Thumbnail *string `json:"thumbnail"`
|
||||||
|
Location *string `json:"location"`
|
||||||
|
Attachments []string `json:"attachments"`
|
||||||
|
Tags []models.Tag `json:"tags"`
|
||||||
|
Categories []models.Category `json:"categories"`
|
||||||
|
PublishedAt *time.Time `json:"published_at"`
|
||||||
|
PublishedUntil *time.Time `json:"published_until"`
|
||||||
|
VisibleUsers []uint `json:"visible_users_list"`
|
||||||
|
InvisibleUsers []uint `json:"invisible_users_list"`
|
||||||
|
Visibility *int8 `json:"visibility"`
|
||||||
|
IsDraft bool `json:"is_draft"`
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := exts.BindAndValidate(c, &data); err != nil {
|
||||||
|
return err
|
||||||
|
} else if len(data.Content) == 0 && len(data.Attachments) == 0 {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, "content or attachments are required")
|
||||||
|
}
|
||||||
|
|
||||||
|
publisher, err := services.GetPublisher(data.Publisher, user.ID)
|
||||||
|
if err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
var item models.Post
|
||||||
|
if err := database.C.Where(models.Post{
|
||||||
|
BaseModel: cruda.BaseModel{ID: uint(id)},
|
||||||
|
PublisherID: publisher.ID,
|
||||||
|
Type: models.PostTypeQuestion,
|
||||||
|
}).Preload("Publisher").First(&item).Error; err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusNotFound, err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
if item.LockedAt != nil {
|
||||||
|
return fiber.NewError(fiber.StatusForbidden, "post was locked")
|
||||||
|
}
|
||||||
|
|
||||||
|
if !item.IsDraft && !data.IsDraft {
|
||||||
|
item.EditedAt = lo.ToPtr(time.Now())
|
||||||
|
}
|
||||||
|
|
||||||
|
if item.IsDraft && !data.IsDraft && data.PublishedAt == nil {
|
||||||
|
item.PublishedAt = lo.ToPtr(time.Now())
|
||||||
|
} else {
|
||||||
|
item.PublishedAt = data.PublishedAt
|
||||||
|
}
|
||||||
|
|
||||||
|
body := models.PostQuestionBody{
|
||||||
|
PostStoryBody: models.PostStoryBody{
|
||||||
|
Thumbnail: data.Thumbnail,
|
||||||
|
Title: data.Title,
|
||||||
|
Content: data.Content,
|
||||||
|
Location: data.Location,
|
||||||
|
Attachments: data.Attachments,
|
||||||
|
},
|
||||||
|
Reward: item.Body["reward"].(float64),
|
||||||
|
}
|
||||||
|
|
||||||
|
var bodyMapping map[string]any
|
||||||
|
rawBody, _ := jsoniter.Marshal(body)
|
||||||
|
_ = jsoniter.Unmarshal(rawBody, &bodyMapping)
|
||||||
|
|
||||||
|
item.Alias = data.Alias
|
||||||
|
item.Body = bodyMapping
|
||||||
|
item.Language = services.DetectLanguage(data.Content)
|
||||||
|
item.Tags = data.Tags
|
||||||
|
item.Categories = data.Categories
|
||||||
|
item.IsDraft = data.IsDraft
|
||||||
|
item.PublishedUntil = data.PublishedUntil
|
||||||
|
item.VisibleUsers = data.VisibleUsers
|
||||||
|
item.InvisibleUsers = data.InvisibleUsers
|
||||||
|
|
||||||
|
// Preload publisher data
|
||||||
|
item.Publisher = publisher
|
||||||
|
|
||||||
|
if item.PublishedAt == nil {
|
||||||
|
item.PublishedAt = data.PublishedAt
|
||||||
|
}
|
||||||
|
if data.Visibility != nil {
|
||||||
|
item.Visibility = *data.Visibility
|
||||||
|
}
|
||||||
|
|
||||||
|
if item, err = services.EditPost(item); err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||||
|
} else {
|
||||||
|
_ = authkit.AddEventExt(
|
||||||
|
gap.Nx,
|
||||||
|
"posts.edit",
|
||||||
|
strconv.Itoa(int(item.ID)),
|
||||||
|
c,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.JSON(item)
|
||||||
|
}
|
||||||
|
|
||||||
|
func selectQuestionAnswer(c *fiber.Ctx) error {
|
||||||
|
id, _ := c.ParamsInt("postId", 0)
|
||||||
|
if err := sec.EnsureAuthenticated(c); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
user := c.Locals("user").(authm.Account)
|
||||||
|
|
||||||
|
var data struct {
|
||||||
|
Publisher uint `json:"publisher"`
|
||||||
|
AnswerID uint `json:"answer_id" validate:"required"`
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := exts.BindAndValidate(c, &data); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
publisher, err := services.GetPublisher(data.Publisher, user.ID)
|
||||||
|
if err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
var item models.Post
|
||||||
|
if err := database.C.Where(models.Post{
|
||||||
|
BaseModel: cruda.BaseModel{ID: uint(id)},
|
||||||
|
PublisherID: publisher.ID,
|
||||||
|
Type: models.PostTypeQuestion,
|
||||||
|
}).Preload("Publisher").First(&item).Error; err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusNotFound, err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
if item.LockedAt != nil {
|
||||||
|
return fiber.NewError(fiber.StatusForbidden, "post was locked")
|
||||||
|
}
|
||||||
|
if val, ok := item.Body["answer"].(uint); ok && val > 0 {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, "question already has an answer")
|
||||||
|
}
|
||||||
|
|
||||||
|
var answer models.Post
|
||||||
|
if err := database.C.Where("id = ? AND reply_id = ?", data.AnswerID, item.ID).Preload("Publisher").First(&answer).Error; err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusNotFound, fmt.Sprintf("related answer was not found: %v", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
item.Body["answer"] = answer.ID
|
||||||
|
|
||||||
|
// Preload publisher data
|
||||||
|
item.Publisher = publisher
|
||||||
|
if item, err = services.EditPost(item); err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, err.Error())
|
||||||
|
} else {
|
||||||
|
_ = authkit.AddEventExt(
|
||||||
|
gap.Nx,
|
||||||
|
"posts.edit.answer",
|
||||||
|
strconv.Itoa(int(item.ID)),
|
||||||
|
c,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.JSON(item)
|
||||||
|
}
|
@ -166,6 +166,7 @@ func editStory(c *fiber.Ctx) error {
|
|||||||
if err := database.C.Where(models.Post{
|
if err := database.C.Where(models.Post{
|
||||||
BaseModel: cruda.BaseModel{ID: uint(id)},
|
BaseModel: cruda.BaseModel{ID: uint(id)},
|
||||||
PublisherID: publisher.ID,
|
PublisherID: publisher.ID,
|
||||||
|
Type: models.PostTypeStory,
|
||||||
}).Preload("Publisher").First(&item).Error; err != nil {
|
}).Preload("Publisher").First(&item).Error; err != nil {
|
||||||
return fiber.NewError(fiber.StatusNotFound, err.Error())
|
return fiber.NewError(fiber.StatusNotFound, err.Error())
|
||||||
}
|
}
|
||||||
|
@ -10,8 +10,9 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
PostTypeStory = "story"
|
PostTypeStory = "story"
|
||||||
PostTypeArticle = "article"
|
PostTypeArticle = "article"
|
||||||
|
PostTypeQuestion = "question"
|
||||||
)
|
)
|
||||||
|
|
||||||
type PostVisibilityLevel = int8
|
type PostVisibilityLevel = int8
|
||||||
@ -81,6 +82,12 @@ type PostArticleBody struct {
|
|||||||
Attachments []string `json:"attachments"`
|
Attachments []string `json:"attachments"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type PostQuestionBody struct {
|
||||||
|
PostStoryBody
|
||||||
|
Answer *uint `json:"answer"`
|
||||||
|
Reward float64 `json:"reward"`
|
||||||
|
}
|
||||||
|
|
||||||
type PostInsight struct {
|
type PostInsight struct {
|
||||||
cruda.BaseModel
|
cruda.BaseModel
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user