Critical/Major:
- Remove dead functions syncRepoNameFromArg and repoNameFromArg (used url pkg without import, would cause compile error)
- Migrate forge.lthn.ai/core/config → dappco.re/go/core/config in forge/config.go and gitea/config.go
- Propagate ListIssueCommentsIter errors in forge/meta.go and gitea/meta.go (was silently returning truncated count)
- Add RedactedToken() to gitea/client.go to avoid exposing raw API tokens
- Add 30s timeout to http.DefaultClient usage in gitea/prs.go via package-level httpClient
- Fix stringsx.Fields (bufio 64KiB limit), Repeat (wrong for negative/zero), Replace (ignored n param) to match stdlib
- Fix fmtx.Println to use fmt.Sprintln so spaces appear between operands
- Fix filepathx.Abs to use path/filepath for OS-aware path handling; wrap Getwd error
- Fix stdio.Write to return io.ErrShortWrite on partial writes
- Add mutex lock to jobrunner.Journal.Query to prevent data race with Append
- Add sync.RWMutex to ScmProvider; protect p.index reads/writes in pkg/api/provider.go
- Fix cmd/scm/cmd_index.go: append dir to repoPaths only after ReadDir confirms existence
- Fix manifest/compile.go: copy manifest before applying version override to avoid mutating caller
- Fix forge/labels.go: use ListOrgLabelsIter/ListRepoLabelsIter names in iterator error logs
- Wrap single-segment validation error in syncutil.ParseRepoName with function context
Minor:
- Fix import ordering (stdlib → forge.lthn.ai → third-party) in cmd/forge, cmd/collect, repos, cmd/gitea files
- Add t.Setenv("HOME", t.TempDir()) to gitea testhelpers and forge/labels_test.go
- Add iterator yield guard in forge/orgs_test.go
- Convert syncutil/repo_name_test.go to table-driven tests
- Use json.Marshal in pkg/api/provider_test.go instead of string concatenation
- Fix test naming (redundant/conflicting _Good/_Bad suffixes) across 10 test files
Co-Authored-By: Virgil <virgil@lethean.io>
133 lines
3.1 KiB
Go
133 lines
3.1 KiB
Go
// SPDX-License-Identifier: EUPL-1.2
|
|
|
|
package collect
|
|
|
|
import (
|
|
"time"
|
|
|
|
"forge.lthn.ai/core/cli/pkg/cli"
|
|
|
|
"dappco.re/go/core/i18n"
|
|
collectpkg "dappco.re/go/core/scm/collect"
|
|
fmt "dappco.re/go/core/scm/internal/ax/fmtx"
|
|
)
|
|
|
|
// addDispatchCommand adds the 'dispatch' subcommand to the collect parent.
|
|
func addDispatchCommand(parent *cli.Command) {
|
|
dispatchCmd := &cli.Command{
|
|
Use: "dispatch <event>",
|
|
Short: i18n.T("cmd.collect.dispatch.short"),
|
|
Long: i18n.T("cmd.collect.dispatch.long"),
|
|
Args: cli.MinimumNArgs(1),
|
|
RunE: func(cmd *cli.Command, args []string) error {
|
|
return runDispatch(args[0])
|
|
},
|
|
}
|
|
|
|
// Add hooks subcommand group
|
|
hooksCmd := &cli.Command{
|
|
Use: "hooks",
|
|
Short: i18n.T("cmd.collect.dispatch.hooks.short"),
|
|
}
|
|
|
|
addHooksListCommand(hooksCmd)
|
|
addHooksRegisterCommand(hooksCmd)
|
|
|
|
dispatchCmd.AddCommand(hooksCmd)
|
|
parent.AddCommand(dispatchCmd)
|
|
}
|
|
|
|
func runDispatch(eventType string) error {
|
|
cfg := newConfig()
|
|
setupVerboseLogging(cfg)
|
|
|
|
// Validate event type
|
|
switch eventType {
|
|
case collectpkg.EventStart,
|
|
collectpkg.EventProgress,
|
|
collectpkg.EventItem,
|
|
collectpkg.EventError,
|
|
collectpkg.EventComplete:
|
|
// Valid event type
|
|
default:
|
|
return cli.Err("unknown event type: %s (valid: start, progress, item, error, complete)", eventType)
|
|
}
|
|
|
|
event := collectpkg.Event{
|
|
Type: eventType,
|
|
Source: "cli",
|
|
Message: fmt.Sprintf("Manual dispatch of %s event", eventType),
|
|
Time: time.Now(),
|
|
}
|
|
|
|
cfg.Dispatcher.Emit(event)
|
|
cli.Success(fmt.Sprintf("Dispatched %s event", eventType))
|
|
|
|
return nil
|
|
}
|
|
|
|
// addHooksListCommand adds the 'hooks list' subcommand.
|
|
func addHooksListCommand(parent *cli.Command) {
|
|
listCmd := &cli.Command{
|
|
Use: "list",
|
|
Short: i18n.T("cmd.collect.dispatch.hooks.list.short"),
|
|
RunE: func(cmd *cli.Command, args []string) error {
|
|
return runHooksList()
|
|
},
|
|
}
|
|
|
|
parent.AddCommand(listCmd)
|
|
}
|
|
|
|
func runHooksList() error {
|
|
eventTypes := []string{
|
|
collectpkg.EventStart,
|
|
collectpkg.EventProgress,
|
|
collectpkg.EventItem,
|
|
collectpkg.EventError,
|
|
collectpkg.EventComplete,
|
|
}
|
|
|
|
table := cli.NewTable("Event", "Status")
|
|
for _, et := range eventTypes {
|
|
table.AddRow(et, dimStyle.Render("no hooks registered"))
|
|
}
|
|
|
|
cli.Blank()
|
|
cli.Print("%s\n\n", cli.HeaderStyle.Render("Registered Hooks"))
|
|
table.Render()
|
|
cli.Blank()
|
|
|
|
return nil
|
|
}
|
|
|
|
// addHooksRegisterCommand adds the 'hooks register' subcommand.
|
|
func addHooksRegisterCommand(parent *cli.Command) {
|
|
registerCmd := &cli.Command{
|
|
Use: "register <event> <command>",
|
|
Short: i18n.T("cmd.collect.dispatch.hooks.register.short"),
|
|
Args: cli.ExactArgs(2),
|
|
RunE: func(cmd *cli.Command, args []string) error {
|
|
return runHooksRegister(args[0], args[1])
|
|
},
|
|
}
|
|
|
|
parent.AddCommand(registerCmd)
|
|
}
|
|
|
|
func runHooksRegister(eventType, command string) error {
|
|
// Validate event type
|
|
switch eventType {
|
|
case collectpkg.EventStart,
|
|
collectpkg.EventProgress,
|
|
collectpkg.EventItem,
|
|
collectpkg.EventError,
|
|
collectpkg.EventComplete:
|
|
// Valid
|
|
default:
|
|
return cli.Err("unknown event type: %s (valid: start, progress, item, error, complete)", eventType)
|
|
}
|
|
|
|
cli.Success(fmt.Sprintf("Registered hook for %s: %s", eventType, command))
|
|
return nil
|
|
}
|