diff options
| author | Paul Buetow <paul@buetow.org> | 2026-04-08 15:56:03 +0300 |
|---|---|---|
| committer | Paul Buetow <paul@buetow.org> | 2026-04-08 16:14:42 +0300 |
| commit | 8ce0a453f355dd3f9239ee33bf4a426f6951ac9f (patch) | |
| tree | ecaa2f2cd8113df0110ec9109b219995fdd63939 | |
| parent | 4855df91e6ef9b7cc3e8278e2cb349f8e0d36dad (diff) | |
Fix reload error handling for task 1
| -rw-r--r-- | internal/ui/handlers.go | 34 | ||||
| -rw-r--r-- | internal/ui/keyhandlers.go | 22 | ||||
| -rw-r--r-- | internal/ui/table.go | 21 | ||||
| -rw-r--r-- | internal/ui/table_test.go | 59 |
4 files changed, 116 insertions, 20 deletions
diff --git a/internal/ui/handlers.go b/internal/ui/handlers.go index 5428ebd..831687a 100644 --- a/internal/ui/handlers.go +++ b/internal/ui/handlers.go @@ -58,7 +58,9 @@ func (m *Model) handleAnnotationMode(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) { return err } } - _ = m.reload() + if err := m.reload(); err != nil { + return fmt.Errorf("reloading tasks: %w", err) + } return nil } @@ -84,7 +86,9 @@ func (m *Model) handleDescriptionMode(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) if err := task.SetDescription(m.descID, value); err != nil { return err } - _ = m.reload() + if err := m.reload(); err != nil { + return fmt.Errorf("reloading tasks: %w", err) + } return nil } @@ -133,7 +137,9 @@ func (m *Model) handleTagsMode(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) { return err } } - _ = m.reload() + if err := m.reload(); err != nil { + return fmt.Errorf("reloading tasks: %w", err) + } return nil } @@ -164,7 +170,9 @@ func (m *Model) handleDueEditMode(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) { return m, cmd } m.dueEditing = false - m.reload() + if !m.reloadAndReport() { + return m, nil + } var cmd tea.Cmd if m.showTaskDetail { // In detail view, blink the due field @@ -202,7 +210,9 @@ func (m *Model) handleRecurrenceMode(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) { if err := task.SetRecurrence(m.recurID, value); err != nil { return err } - _ = m.reload() + if err := m.reload(); err != nil { + return fmt.Errorf("reloading tasks: %w", err) + } return nil } @@ -233,7 +243,7 @@ func (m *Model) handleProjectMode(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) { onExit := func() { m.projEditing = false - m.reload() + m.reloadAndReport() } model, cmd := m.handleTextInput(msg, &m.projInput, onEnter, onExit) @@ -267,7 +277,9 @@ func (m *Model) handlePriorityMode(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) { return m, cmd } m.prioritySelecting = false - m.reload() + if !m.reloadAndReport() { + return m, nil + } var cmd tea.Cmd if m.showTaskDetail { // In detail view, blink the priority field @@ -343,7 +355,9 @@ func (m *Model) handleAddTaskMode(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) { m.addingTask = false m.addInput.Blur() - m.reload() + if !m.reloadAndReport() { + return m, nil + } // Find the newly added task var newID int @@ -408,7 +422,9 @@ func (m *Model) handleSearchMode(msg tea.KeyPressMsg) (tea.Model, tea.Cmd) { } m.searching = false m.searchInput.Blur() - m.reload() + if !m.reloadAndReport() { + return m, nil + } m.updateTableHeight() if len(m.searchMatches) > 0 { diff --git a/internal/ui/keyhandlers.go b/internal/ui/keyhandlers.go index 33c7dc0..de16dd0 100644 --- a/internal/ui/keyhandlers.go +++ b/internal/ui/keyhandlers.go @@ -197,7 +197,7 @@ func (m *Model) handleQuitKey() (tea.Model, tea.Cmd) { m.searchRegex = nil m.searchMatches = nil m.searchIndex = 0 - m.reload() + m.reloadAndReport() return m, nil } return m, tea.Quit @@ -252,7 +252,7 @@ func (m *Model) handleEscapeKey() (tea.Model, tea.Cmd) { m.searchRegex = nil m.searchMatches = nil m.searchIndex = 0 - m.reload() + m.reloadAndReport() return m, nil } return m, nil @@ -294,7 +294,9 @@ func (m *Model) handleToggleStart() (tea.Model, tea.Cmd) { } } - m.reload() + if !m.reloadAndReport() { + return m, nil + } return m, m.startBlink(id, false) } @@ -420,7 +422,9 @@ func (m *Model) handleRemoveDueDate() (tea.Model, tea.Cmd) { return m, nil } - m.reload() + if !m.reloadAndReport() { + return m, nil + } return m, m.startBlink(id, false) } @@ -438,7 +442,9 @@ func (m *Model) handleRandomDueDate() (tea.Model, tea.Cmd) { return m, nil } - m.reload() + if !m.reloadAndReport() { + return m, nil + } return m, m.startBlink(id, false) } @@ -575,7 +581,9 @@ func (m *Model) handleTagToProject() (tea.Model, tea.Cmd) { return m, nil } - m.reload() + if !m.reloadAndReport() { + return m, nil + } return m, m.startBlink(id, false) } @@ -607,7 +615,7 @@ func (m *Model) handleToggleBlink() (tea.Model, tea.Cmd) { } func (m *Model) handleRefresh() (tea.Model, tea.Cmd) { - m.reload() + m.reloadAndReport() return m, nil } diff --git a/internal/ui/table.go b/internal/ui/table.go index 0f569d2..20928fd 100644 --- a/internal/ui/table.go +++ b/internal/ui/table.go @@ -307,7 +307,7 @@ func (m *Model) startBlink(id int, markDone bool) tea.Cmd { } m.blinkID = 0 m.blinkMarkDone = false - m.reload() + m.reloadAndReport() return nil } @@ -462,6 +462,14 @@ func (m *Model) reload() error { return nil } +func (m *Model) reloadAndReport() bool { + if err := m.reload(); err != nil { + m.showError(fmt.Errorf("reloading tasks: %w", err)) + return false + } + return true +} + // Init implements tea.Model. func (m Model) Init() tea.Cmd { return nil } @@ -572,7 +580,10 @@ func (m *Model) handleEditDone(msg editDoneMsg) (tea.Model, tea.Cmd) { if m.showUltra { m.ultraFocusedID = m.editID } - m.reload() + if !m.reloadAndReport() { + m.editID = 0 + return m, nil + } cmd := m.startBlink(m.editID, false) m.editID = 0 return m, cmd @@ -616,7 +627,9 @@ func (m *Model) handleDescEditDone(msg descEditDoneMsg) (tea.Model, tea.Cmd) { } // Reload and start blinking - m.reload() + if !m.reloadAndReport() { + return m, nil + } return m, m.startDetailBlink(m.detailDescriptionFieldIndex()) } @@ -667,7 +680,7 @@ func (m *Model) handleBlinkMsg() (tea.Model, tea.Cmd) { m.showError(err) } } - m.reload() + m.reloadAndReport() return m, nil } diff --git a/internal/ui/table_test.go b/internal/ui/table_test.go index 95f9b1b..7ad079b 100644 --- a/internal/ui/table_test.go +++ b/internal/ui/table_test.go @@ -228,6 +228,65 @@ func TestHandleDescEditDoneRemovesTempFileOnEditorError(t *testing.T) { } } +func TestHandleFilterModeReportsReloadError(t *testing.T) { + tmp := t.TempDir() + taskPath := filepath.Join(tmp, "task") + failFlag := filepath.Join(tmp, "fail-reload") + + script := "#!/bin/sh\n" + + "if echo \"$@\" | grep -q export; then\n" + + " if [ -f " + failFlag + " ]; then\n" + + " echo 'reload failed' >&2\n" + + " exit 42\n" + + " fi\n" + + " echo '{\"id\":1,\"uuid\":\"x\",\"description\":\"d\",\"status\":\"pending\",\"entry\":\"\",\"priority\":\"\",\"urgency\":0,\"annotations\":[]}'\n" + + " exit 0\n" + + "fi\n" + + if err := os.WriteFile(taskPath, []byte(script), 0o755); err != nil { + t.Fatal(err) + } + + origPath := os.Getenv("PATH") + os.Setenv("PATH", tmp+":"+origPath) + t.Cleanup(func() { os.Setenv("PATH", origPath) }) + + os.Setenv("TASKDATA", tmp) + os.Setenv("TASKRC", "/dev/null") + t.Cleanup(func() { + os.Unsetenv("TASKDATA") + os.Unsetenv("TASKRC") + }) + + m, err := New(nil, "firefox") + if err != nil { + t.Fatalf("New: %v", err) + } + + if err := os.WriteFile(failFlag, []byte("1"), 0o644); err != nil { + t.Fatal(err) + } + + m.filterEditing = true + m.filterInput.SetValue("project:alpha") + + mv, cmd := (&m).handleFilterMode(tea.KeyPressMsg{Code: tea.KeyEnter}) + m = *mv.(*Model) + + if !m.filterEditing { + t.Fatalf("filter editing was cleared after reload failure") + } + if cmd == nil { + t.Fatalf("reload failure did not return a clear-status command") + } + if !strings.Contains(m.statusMsg, "reloading tasks") { + t.Fatalf("unexpected status message: %q", m.statusMsg) + } + if got := strings.Join(m.filters, " "); got != "project:alpha" { + t.Fatalf("filters not applied before reload attempt: %q", got) + } +} + func TestDoneHotkey(t *testing.T) { tmp := t.TempDir() taskPath := filepath.Join(tmp, "task") |
