feat(task): JIRA형 작업 — 설명·우선순위·라벨 필드 + 댓글 스레드
All checks were successful
build-and-push / build (push) Successful in 33s

- ProjectTask에 description/priority/labels 추가
- TaskComment 모델 + 엔드포인트(GET/POST /tasks/{id}/comments, DELETE /comments/{id})
- 댓글 작성자=요청자, 삭제는 작성자 또는 관리자
- PATCH /tasks: labels/dependsOn JSON 슬라이스 map 패치 처리

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
theorose49 2026-06-30 09:08:02 +09:00
parent dcf8b415db
commit f46f135dbf
4 changed files with 89 additions and 3 deletions

View File

@ -1,6 +1,7 @@
package httpapi
import (
"encoding/json"
"fmt"
"net/http"
"strings"
@ -9,6 +10,7 @@ import (
"spin/internal/models"
"github.com/go-chi/chi/v5"
"gorm.io/datatypes"
)
// ---- company / product / version (master data) ----------------------------
@ -374,11 +376,78 @@ func (s *Server) handlePatchTask(w http.ResponseWriter, r *http.Request) {
}
delete(patch, "id")
delete(patch, "projectId")
// JSON 슬라이스 컬럼(labels/dependsOn)은 map 패치 시 driver.Valuer로 변환해 줘야 한다.
for _, k := range []string{"labels", "dependsOn"} {
if v, ok := patch[k]; ok {
if b, err := json.Marshal(v); err == nil {
patch[k] = datatypes.JSON(b)
}
}
}
s.db.Model(&t).Updates(patch)
s.db.First(&t, "id = ?", t.ID)
writeJSON(w, http.StatusOK, t)
}
// ---- task comments (JIRA형 댓글) -----------------------------------------
func (s *Server) taskProjectID(taskID string) (string, bool) {
var t models.ProjectTask
if err := s.db.Select("project_id").First(&t, "id = ?", taskID).Error; err != nil {
return "", false
}
return t.ProjectID, true
}
func (s *Server) handleListTaskComments(w http.ResponseWriter, r *http.Request) {
tid := chi.URLParam(r, "tId")
pid, ok := s.taskProjectID(tid)
if !ok || !s.canSeeProject(r, pid) {
writeError(w, http.StatusForbidden, "권한이 없습니다")
return
}
var out []models.TaskComment
s.db.Where("task_id = ?", tid).Order("created_at asc").Find(&out)
writeJSON(w, http.StatusOK, out)
}
func (s *Server) handleCreateTaskComment(w http.ResponseWriter, r *http.Request) {
tid := chi.URLParam(r, "tId")
pid, ok := s.taskProjectID(tid)
if !ok || !s.canSeeProject(r, pid) {
writeError(w, http.StatusForbidden, "권한이 없습니다")
return
}
var c models.TaskComment
if err := decodeJSON(r, &c); err != nil {
writeError(w, http.StatusBadRequest, err.Error())
return
}
if strings.TrimSpace(c.Body) == "" {
writeError(w, http.StatusBadRequest, "내용을 입력하세요")
return
}
c.ID = ""
c.TaskID = tid
c.AuthorEmail = s.email(r)
s.db.Create(&c)
writeJSON(w, http.StatusCreated, c)
}
func (s *Server) handleDeleteTaskComment(w http.ResponseWriter, r *http.Request) {
var c models.TaskComment
if err := s.db.First(&c, "id = ?", chi.URLParam(r, "cId")).Error; err != nil {
writeError(w, http.StatusNotFound, "댓글을 찾을 수 없습니다")
return
}
if !s.isAdmin(r) && !s.owns(r, c.AuthorEmail) {
writeError(w, http.StatusForbidden, "본인 댓글만 삭제할 수 있습니다")
return
}
s.db.Delete(&models.TaskComment{}, "id = ?", c.ID)
writeJSON(w, http.StatusOK, map[string]bool{"ok": true})
}
func (s *Server) handleDeleteTask(w http.ResponseWriter, r *http.Request) {
var t models.ProjectTask
if err := s.db.First(&t, "id = ?", chi.URLParam(r, "tId")).Error; err != nil {

View File

@ -123,6 +123,9 @@ func NewRouter(db *gorm.DB, store *storage.Storage, cfg config.Config, pusher *p
r.Post("/projects/{id}/tasks", s.handleCreateTask)
r.Patch("/tasks/{tId}", s.handlePatchTask)
r.Delete("/tasks/{tId}", s.handleDeleteTask)
r.Get("/tasks/{tId}/comments", s.handleListTaskComments)
r.Post("/tasks/{tId}/comments", s.handleCreateTaskComment)
r.Delete("/comments/{cId}", s.handleDeleteTaskComment)
// admin-only commercial block
r.Get("/projects/{id}/contract", s.handleGetContract)
r.Put("/projects/{id}/contract", s.handlePutContract)

View File

@ -28,7 +28,7 @@ func All() []interface{} {
&Attendance{}, &LeaveRequest{}, &OvertimeRequest{}, &WorkPolicy{}, &LeaveBalance{},
// slice 3 — projects
&Company{}, &Product{}, &Version{}, &Project{}, &ProjectMember{},
&ClientContact{}, &ProjectTask{}, &Contract{}, &ContractFile{}, &PaymentSplit{},
&ClientContact{}, &ProjectTask{}, &TaskComment{}, &Contract{}, &ContractFile{}, &PaymentSplit{},
// slice 4 — incentive
&IncentiveConfig{}, &PaymentStage{}, &UserIncentive{}, &QuarterlySettlement{},
// slice 5 — accounting

View File

@ -97,7 +97,10 @@ type ProjectTask struct {
Base
ProjectID string `gorm:"index" json:"projectId"`
Title string `json:"title"`
Description string `json:"description"` // 상세 설명 (JIRA형 카드 본문)
Lane string `json:"lane"` // todo | doing | review | done
Priority string `json:"priority"` // low | medium | high | urgent
Labels datatypes.JSONSlice[string] `json:"labels"` // 라벨/태그
Start string `json:"start"` // YYYY-MM-DD
End string `json:"end"`
Assignee string `json:"assignee"`
@ -111,6 +114,17 @@ type ProjectTask struct {
func (m *ProjectTask) BeforeCreate(*gorm.DB) error { m.ensureID(); return nil }
// TaskComment is a JIRA-style comment thread on a task (작성자·본문·시각).
type TaskComment struct {
Base
TaskID string `gorm:"index" json:"taskId"`
AuthorEmail string `json:"authorEmail"`
Body string `json:"body"`
CreatedAt time.Time `json:"createdAt"`
}
func (m *TaskComment) BeforeCreate(*gorm.DB) error { m.ensureID(); return nil }
// Contract holds the [admin-only] commercial terms of a project. BE is the
// break-even floor (손해가 안 나는 최소 금액). Exposed ONLY to admins.
type Contract struct {