mirror of
https://github.com/ForFarmTeam/ForFarm.git
synced 2025-12-19 22:14:08 +01:00
291 lines
7.8 KiB
Go
291 lines
7.8 KiB
Go
package api
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"net/http"
|
|
"time"
|
|
|
|
"github.com/danielgtaylor/huma/v2"
|
|
"github.com/forfarm/backend/internal/domain"
|
|
"github.com/go-chi/chi/v5"
|
|
"github.com/gofrs/uuid"
|
|
)
|
|
|
|
func (a *api) registerKnowledgeHubRoutes(_ chi.Router, api huma.API) {
|
|
tags := []string{"knowledge-hub"}
|
|
|
|
prefix := "/knowledge-hub"
|
|
|
|
huma.Register(api, huma.Operation{
|
|
OperationID: "getAllKnowledgeArticles",
|
|
Method: http.MethodGet,
|
|
Path: prefix,
|
|
Tags: tags,
|
|
}, a.getAllKnowledgeArticlesHandler)
|
|
|
|
huma.Register(api, huma.Operation{
|
|
OperationID: "getKnowledgeArticleByID",
|
|
Method: http.MethodGet,
|
|
Path: prefix + "/{uuid}",
|
|
Tags: tags,
|
|
}, a.getKnowledgeArticleByIDHandler)
|
|
|
|
huma.Register(api, huma.Operation{
|
|
OperationID: "getKnowledgeArticlesByCategory",
|
|
Method: http.MethodGet,
|
|
Path: prefix + "/category/{category}",
|
|
Tags: tags,
|
|
}, a.getKnowledgeArticlesByCategoryHandler)
|
|
|
|
huma.Register(api, huma.Operation{
|
|
OperationID: "createOrUpdateKnowledgeArticle",
|
|
Method: http.MethodPost,
|
|
Path: prefix,
|
|
Tags: tags,
|
|
}, a.createOrUpdateKnowledgeArticleHandler)
|
|
|
|
huma.Register(api, huma.Operation{
|
|
OperationID: "getArticleTableOfContents",
|
|
Method: http.MethodGet,
|
|
Path: prefix + "/{uuid}/toc",
|
|
Tags: tags,
|
|
}, a.getArticleTableOfContentsHandler)
|
|
|
|
huma.Register(api, huma.Operation{
|
|
OperationID: "getArticleRelatedArticles",
|
|
Method: http.MethodGet,
|
|
Path: prefix + "/{uuid}/related",
|
|
Tags: tags,
|
|
}, a.getArticleRelatedArticlesHandler)
|
|
|
|
huma.Register(api, huma.Operation{
|
|
OperationID: "createRelatedArticle",
|
|
Method: http.MethodPost,
|
|
Path: prefix + "/{uuid}/related",
|
|
Tags: tags,
|
|
}, a.createRelatedArticleHandler)
|
|
}
|
|
|
|
type GetKnowledgeArticlesOutput struct {
|
|
Body struct {
|
|
Articles []domain.KnowledgeArticle `json:"articles"`
|
|
} `json:"body"`
|
|
}
|
|
|
|
type GetKnowledgeArticleByIDOutput struct {
|
|
Body struct {
|
|
Article domain.KnowledgeArticle `json:"article"`
|
|
} `json:"body"`
|
|
}
|
|
|
|
type CreateOrUpdateKnowledgeArticleInput struct {
|
|
Body struct {
|
|
UUID string `json:"uuid,omitempty"` // Optional for create, required for update
|
|
Title string `json:"title"`
|
|
Content string `json:"content"`
|
|
Author string `json:"author"`
|
|
PublishDate time.Time `json:"publish_date"`
|
|
ReadTime string `json:"read_time"`
|
|
Categories []string `json:"categories"`
|
|
} `json:"body"`
|
|
}
|
|
|
|
type CreateOrUpdateKnowledgeArticleOutput struct {
|
|
Body struct {
|
|
Article domain.KnowledgeArticle `json:"article"`
|
|
} `json:"body"`
|
|
}
|
|
|
|
type GetTableOfContentsOutput struct {
|
|
Body struct {
|
|
TableOfContents []domain.TableOfContent `json:"table_of_contents"`
|
|
} `json:"body"`
|
|
}
|
|
|
|
type GetRelatedArticlesOutput struct {
|
|
Body struct {
|
|
RelatedArticles []domain.RelatedArticle `json:"related_articles"`
|
|
} `json:"body"`
|
|
}
|
|
|
|
type CreateRelatedArticleInput struct {
|
|
UUID string `path:"uuid"`
|
|
Body struct {
|
|
RelatedTitle string `json:"related_title"`
|
|
RelatedTag string `json:"related_tag"`
|
|
} `json:"body"`
|
|
}
|
|
|
|
func (a *api) getAllKnowledgeArticlesHandler(ctx context.Context, input *struct{}) (*GetKnowledgeArticlesOutput, error) {
|
|
resp := &GetKnowledgeArticlesOutput{}
|
|
|
|
articles, err := a.knowledgeHubRepo.GetAllArticles(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
resp.Body.Articles = articles
|
|
return resp, nil
|
|
}
|
|
|
|
func (a *api) getKnowledgeArticleByIDHandler(ctx context.Context, input *struct {
|
|
UUID string `path:"uuid" example:"550e8400-e29b-41d4-a716-446655440000"`
|
|
}) (*GetKnowledgeArticleByIDOutput, error) {
|
|
resp := &GetKnowledgeArticleByIDOutput{}
|
|
|
|
if input.UUID == "" {
|
|
return nil, huma.Error400BadRequest("UUID parameter is required")
|
|
}
|
|
|
|
_, err := uuid.FromString(input.UUID)
|
|
if err != nil {
|
|
return nil, huma.Error400BadRequest("invalid UUID format")
|
|
}
|
|
|
|
article, err := a.knowledgeHubRepo.GetArticleByID(ctx, input.UUID)
|
|
if err != nil {
|
|
if errors.Is(err, domain.ErrNotFound) {
|
|
return nil, huma.Error404NotFound("article not found")
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
resp.Body.Article = article
|
|
return resp, nil
|
|
}
|
|
|
|
func (a *api) getKnowledgeArticlesByCategoryHandler(ctx context.Context, input *struct {
|
|
Category string `path:"category" example:"Sustainability"`
|
|
}) (*GetKnowledgeArticlesOutput, error) {
|
|
resp := &GetKnowledgeArticlesOutput{}
|
|
|
|
if input.Category == "" {
|
|
return nil, huma.Error400BadRequest("category parameter is required")
|
|
}
|
|
|
|
articles, err := a.knowledgeHubRepo.GetArticlesByCategory(ctx, input.Category)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
resp.Body.Articles = articles
|
|
return resp, nil
|
|
}
|
|
|
|
func (a *api) createOrUpdateKnowledgeArticleHandler(ctx context.Context, input *CreateOrUpdateKnowledgeArticleInput) (*CreateOrUpdateKnowledgeArticleOutput, error) {
|
|
resp := &CreateOrUpdateKnowledgeArticleOutput{}
|
|
|
|
if input.Body.Title == "" {
|
|
return nil, huma.Error400BadRequest("title is required")
|
|
}
|
|
if input.Body.Content == "" {
|
|
return nil, huma.Error400BadRequest("content is required")
|
|
}
|
|
if input.Body.Author == "" {
|
|
return nil, huma.Error400BadRequest("author is required")
|
|
}
|
|
if len(input.Body.Categories) == 0 {
|
|
return nil, huma.Error400BadRequest("at least one category is required")
|
|
}
|
|
|
|
if input.Body.UUID != "" {
|
|
_, err := uuid.FromString(input.Body.UUID)
|
|
if err != nil {
|
|
return nil, huma.Error400BadRequest("invalid UUID format")
|
|
}
|
|
}
|
|
|
|
article := &domain.KnowledgeArticle{
|
|
UUID: input.Body.UUID,
|
|
Title: input.Body.Title,
|
|
Content: input.Body.Content,
|
|
Author: input.Body.Author,
|
|
PublishDate: input.Body.PublishDate,
|
|
ReadTime: input.Body.ReadTime,
|
|
Categories: input.Body.Categories,
|
|
}
|
|
|
|
err := a.knowledgeHubRepo.CreateOrUpdateArticle(ctx, article)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
resp.Body.Article = *article
|
|
return resp, nil
|
|
}
|
|
|
|
func (a *api) getArticleTableOfContentsHandler(ctx context.Context, input *struct {
|
|
UUID string `path:"uuid" example:"550e8400-e29b-41d4-a716-446655440000"`
|
|
}) (*GetTableOfContentsOutput, error) {
|
|
resp := &GetTableOfContentsOutput{}
|
|
|
|
if input.UUID == "" {
|
|
return nil, huma.Error400BadRequest("UUID parameter is required")
|
|
}
|
|
|
|
_, err := uuid.FromString(input.UUID)
|
|
if err != nil {
|
|
return nil, huma.Error400BadRequest("invalid UUID format")
|
|
}
|
|
|
|
toc, err := a.knowledgeHubRepo.GetTableOfContents(ctx, input.UUID)
|
|
if err != nil {
|
|
if errors.Is(err, domain.ErrNotFound) {
|
|
return nil, huma.Error404NotFound("article not found")
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
resp.Body.TableOfContents = toc
|
|
return resp, nil
|
|
}
|
|
|
|
func (a *api) getArticleRelatedArticlesHandler(ctx context.Context, input *struct {
|
|
UUID string `path:"uuid" example:"550e8400-e29b-41d4-a716-446655440000"`
|
|
}) (*GetRelatedArticlesOutput, error) {
|
|
resp := &GetRelatedArticlesOutput{}
|
|
|
|
if input.UUID == "" {
|
|
return nil, huma.Error400BadRequest("UUID parameter is required")
|
|
}
|
|
|
|
_, err := uuid.FromString(input.UUID)
|
|
if err != nil {
|
|
return nil, huma.Error400BadRequest("invalid UUID format")
|
|
}
|
|
|
|
related, err := a.knowledgeHubRepo.GetRelatedArticles(ctx, input.UUID)
|
|
if err != nil {
|
|
if errors.Is(err, domain.ErrNotFound) {
|
|
return nil, huma.Error404NotFound("article not found")
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
resp.Body.RelatedArticles = related
|
|
return resp, nil
|
|
}
|
|
|
|
func (a *api) createRelatedArticleHandler(
|
|
ctx context.Context,
|
|
input *CreateRelatedArticleInput,
|
|
) (*struct{}, error) {
|
|
// Validate main article exists
|
|
if _, err := a.knowledgeHubRepo.GetArticleByID(ctx, input.UUID); err != nil {
|
|
return nil, huma.Error404NotFound("main article not found")
|
|
}
|
|
|
|
// Create related article
|
|
related := &domain.RelatedArticle{
|
|
RelatedTitle: input.Body.RelatedTitle,
|
|
RelatedTag: input.Body.RelatedTag,
|
|
}
|
|
|
|
if err := a.knowledgeHubRepo.CreateRelatedArticle(ctx, input.UUID, related); err != nil {
|
|
return nil, huma.Error500InternalServerError("failed to create related article")
|
|
}
|
|
|
|
return nil, nil // HTTP 204 No Content
|
|
}
|