|
| 1 | +package controllers |
| 2 | + |
| 3 | +import ( |
| 4 | + "ccsync_backend/models" |
| 5 | + "ccsync_backend/utils/tw" |
| 6 | + "encoding/json" |
| 7 | + "fmt" |
| 8 | + "io" |
| 9 | + "net/http" |
| 10 | +) |
| 11 | + |
| 12 | +// BulkCompleteTaskHandler godoc |
| 13 | +// @Summary Bulk complete tasks |
| 14 | +// @Description Mark multiple tasks as completed in Taskwarrior |
| 15 | +// @Tags Tasks |
| 16 | +// @Accept json |
| 17 | +// @Produce json |
| 18 | +// @Param task body models.BulkCompleteTaskRequestBody true "Bulk task completion details" |
| 19 | +// @Success 202 {string} string "Bulk task completion accepted for processing" |
| 20 | +// @Failure 400 {string} string "Invalid request - missing or empty taskuuids" |
| 21 | +// @Failure 405 {string} string "Method not allowed" |
| 22 | +// @Router /complete-tasks [post] |
| 23 | +func BulkCompleteTaskHandler(w http.ResponseWriter, r *http.Request) { |
| 24 | + if r.Method != http.MethodPost { |
| 25 | + http.Error(w, "Invalid request method", http.StatusMethodNotAllowed) |
| 26 | + return |
| 27 | + } |
| 28 | + |
| 29 | + body, err := io.ReadAll(r.Body) |
| 30 | + if err != nil { |
| 31 | + http.Error(w, fmt.Sprintf("error reading request body: %v", err), http.StatusBadRequest) |
| 32 | + return |
| 33 | + } |
| 34 | + defer r.Body.Close() |
| 35 | + |
| 36 | + var requestBody models.BulkCompleteTaskRequestBody |
| 37 | + |
| 38 | + if err := json.Unmarshal(body, &requestBody); err != nil { |
| 39 | + http.Error(w, fmt.Sprintf("error decoding request body: %v", err), http.StatusBadRequest) |
| 40 | + return |
| 41 | + } |
| 42 | + |
| 43 | + email := requestBody.Email |
| 44 | + encryptionSecret := requestBody.EncryptionSecret |
| 45 | + uuid := requestBody.UUID |
| 46 | + taskUUIDs := requestBody.TaskUUIDs |
| 47 | + |
| 48 | + if len(taskUUIDs) == 0 { |
| 49 | + http.Error(w, "taskuuids is required and cannot be empty", http.StatusBadRequest) |
| 50 | + return |
| 51 | + } |
| 52 | + |
| 53 | + logStore := models.GetLogStore() |
| 54 | + |
| 55 | + // Create a *single* job for all UUIDs |
| 56 | + job := Job{ |
| 57 | + Name: "Bulk Complete Tasks", |
| 58 | + Execute: func() error { |
| 59 | + logStore.AddLog("INFO", fmt.Sprintf("[Bulk Complete] Starting %d tasks", len(taskUUIDs)), uuid, "Bulk Complete Task") |
| 60 | + |
| 61 | + failedTasks, err := tw.CompleteTasksInTaskwarrior(email, encryptionSecret, uuid, taskUUIDs) |
| 62 | + |
| 63 | + for taskUUID, errMsg := range failedTasks { |
| 64 | + logStore.AddLog("ERROR", fmt.Sprintf("[Bulk Complete] Failed: %s (%s)", taskUUID, errMsg), uuid, "Bulk Complete Task") |
| 65 | + } |
| 66 | + |
| 67 | + if err != nil { |
| 68 | + logStore.AddLog("ERROR", fmt.Sprintf("[Bulk Complete] Sync error: %v", err), uuid, "Bulk Complete Task") |
| 69 | + return err |
| 70 | + } |
| 71 | + |
| 72 | + successCount := len(taskUUIDs) - len(failedTasks) |
| 73 | + logStore.AddLog("INFO", fmt.Sprintf("[Bulk Complete] Finished: %d succeeded, %d failed", successCount, len(failedTasks)), uuid, "Bulk Complete Task") |
| 74 | + |
| 75 | + return nil |
| 76 | + }, |
| 77 | + } |
| 78 | + |
| 79 | + GlobalJobQueue.AddJob(job) |
| 80 | + w.WriteHeader(http.StatusAccepted) |
| 81 | +} |
0 commit comments