♻️ Interactive v2 #1
@ -19,10 +19,6 @@
|
|||||||
<option name="SPACES_WITHIN_OBJECT_LITERAL_BRACES" value="true" />
|
<option name="SPACES_WITHIN_OBJECT_LITERAL_BRACES" value="true" />
|
||||||
<option name="SPACES_WITHIN_IMPORTS" value="true" />
|
<option name="SPACES_WITHIN_IMPORTS" value="true" />
|
||||||
</TypeScriptCodeStyleSettings>
|
</TypeScriptCodeStyleSettings>
|
||||||
<VueCodeStyleSettings>
|
|
||||||
<option name="INTERPOLATION_NEW_LINE_AFTER_START_DELIMITER" value="false" />
|
|
||||||
<option name="INTERPOLATION_NEW_LINE_BEFORE_END_DELIMITER" value="false" />
|
|
||||||
</VueCodeStyleSettings>
|
|
||||||
<codeStyleSettings language="HTML">
|
<codeStyleSettings language="HTML">
|
||||||
<option name="SOFT_MARGINS" value="120" />
|
<option name="SOFT_MARGINS" value="120" />
|
||||||
<indentOptions>
|
<indentOptions>
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<project version="4">
|
<project version="4">
|
||||||
<component name="SqlDialectMappings">
|
<component name="SqlDialectMappings">
|
||||||
<file url="file://$PROJECT_DIR$/pkg/server/posts_api.go" dialect="PostgreSQL" />
|
<file url="file://$PROJECT_DIR$/pkg/server/moments_api.go" dialect="PostgreSQL" />
|
||||||
|
<file url="file://$APPLICATION_CONFIG_DIR$/consoles/db/2e2101b2-4037-47ee-88ed-456dc2cb4423/console.sql" dialect="PostgreSQL" />
|
||||||
</component>
|
</component>
|
||||||
</project>
|
</project>
|
87
pkg/server/feed_api.go
Normal file
87
pkg/server/feed_api.go
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import "C"
|
||||||
|
import (
|
||||||
|
"code.smartsheep.studio/hydrogen/interactive/pkg/database"
|
||||||
|
"code.smartsheep.studio/hydrogen/interactive/pkg/models"
|
||||||
|
"fmt"
|
||||||
|
"github.com/gofiber/fiber/v2"
|
||||||
|
"github.com/spf13/viper"
|
||||||
|
)
|
||||||
|
|
||||||
|
type FeedItem struct {
|
||||||
|
models.BaseModel
|
||||||
|
|
||||||
|
Title string `json:"title"`
|
||||||
|
Description string `json:"description"`
|
||||||
|
Content string `json:"content"`
|
||||||
|
ModelType string `json:"model_type"`
|
||||||
|
CommentCount int64 `json:"comment_count"`
|
||||||
|
ReactionCount int64 `json:"reaction_count"`
|
||||||
|
AuthorID uint `json:"author_id"`
|
||||||
|
RealmID *uint `json:"realm_id"`
|
||||||
|
|
||||||
|
Author models.Account `json:"author" gorm:"embedded"`
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
queryArticle = "id, created_at, updated_at, title, content, description, realm_id, author_id, 'article' as model_type"
|
||||||
|
queryMoment = "id, created_at, updated_at, NULL as title, content, NULL as description, realm_id, author_id, 'moment' as model_type"
|
||||||
|
)
|
||||||
|
|
||||||
|
func listFeed(c *fiber.Ctx) error {
|
||||||
|
take := c.QueryInt("take", 0)
|
||||||
|
offset := c.QueryInt("offset", 0)
|
||||||
|
realmId := c.QueryInt("realmId", 0)
|
||||||
|
|
||||||
|
if take > 20 {
|
||||||
|
take = 20
|
||||||
|
}
|
||||||
|
|
||||||
|
var whereCondition string
|
||||||
|
|
||||||
|
if realmId > 0 {
|
||||||
|
whereCondition += fmt.Sprintf("feed.realm_id = %d", realmId)
|
||||||
|
} else {
|
||||||
|
whereCondition += "feed.realm_id IS NULL"
|
||||||
|
}
|
||||||
|
|
||||||
|
var author models.Account
|
||||||
|
if len(c.Query("authorId")) > 0 {
|
||||||
|
if err := database.C.Where(&models.Account{Name: c.Query("authorId")}).First(&author).Error; err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusNotFound, err.Error())
|
||||||
|
} else {
|
||||||
|
whereCondition += fmt.Sprintf("AND feed.author_id = %d", author.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var result []FeedItem
|
||||||
|
|
||||||
|
userTable := viper.GetString("database.prefix") + "accounts"
|
||||||
|
commentTable := viper.GetString("database.prefix") + "comments"
|
||||||
|
reactionTable := viper.GetString("database.prefix") + "reactions"
|
||||||
|
|
||||||
|
database.C.Raw(fmt.Sprintf(`SELECT feed.*, author.*,
|
||||||
|
COALESCE(comment_count, 0) as comment_count,
|
||||||
|
COALESCE(reaction_count, 0) as reaction_count
|
||||||
|
FROM (? UNION ALL ?) as feed
|
||||||
|
INNER JOIN %s as author ON author_id = author.id
|
||||||
|
LEFT JOIN (SELECT article_id, moment_id, COUNT(*) as comment_count
|
||||||
|
FROM %s
|
||||||
|
GROUP BY article_id, moment_id) as comments
|
||||||
|
ON (feed.model_type = 'article' AND feed.id = comments.article_id) OR
|
||||||
|
(feed.model_type = 'moment' AND feed.id = comments.moment_id)
|
||||||
|
LEFT JOIN (SELECT article_id, moment_id, COUNT(*) as reaction_count
|
||||||
|
FROM %s
|
||||||
|
GROUP BY article_id, moment_id) as reactions
|
||||||
|
ON (feed.model_type = 'article' AND feed.id = reactions.article_id) OR
|
||||||
|
(feed.model_type = 'moment' AND feed.id = reactions.moment_id)
|
||||||
|
WHERE %s LIMIT ? OFFSET ?`, userTable, commentTable, reactionTable, whereCondition),
|
||||||
|
database.C.Select(queryArticle).Model(&models.Article{}),
|
||||||
|
database.C.Select(queryMoment).Model(&models.Moment{}),
|
||||||
|
take,
|
||||||
|
offset,
|
||||||
|
).Scan(&result)
|
||||||
|
|
||||||
|
return c.JSON(result)
|
||||||
|
}
|
@ -69,7 +69,7 @@ func NewServer() {
|
|||||||
}), openAttachment)
|
}), openAttachment)
|
||||||
api.Post("/attachments", authMiddleware, uploadAttachment)
|
api.Post("/attachments", authMiddleware, uploadAttachment)
|
||||||
|
|
||||||
// TODO Feed (aka. Union source)
|
api.Get("/feed", listFeed)
|
||||||
|
|
||||||
moments := api.Group("/moments").Name("Moments API")
|
moments := api.Group("/moments").Name("Moments API")
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user