Adds a repo_update tool exposing PATCH /api/v1/repos/{owner}/{name}
with optional pointer fields (archived, description, private,
website, template). Only fields set by the caller are sent on the
wire, so the server patches exactly what was asked for.
Originally needed to archive ingestion-svc cleanly instead of
leaving a README tombstone, and to flip template-go-{agent,web}
to template=true so create_project_from_template stops failing
the "is not marked as template" guard.
Wire-level enforcement of "at least one field" returns ErrValidation
before any network call, preventing no-op PATCHes.
private=false (making a repo public) is allowed but flagged in the
tool description with a "verify intent before calling" warning.
The earlier issue draft suggested an ntfy confirmation hook for
that path — out of scope for this PR; the warning string is the
minimum that fits inside the tool surface today.
Wires NewRepoUpdate into cmd/gitea-mcp/main.go alongside the rest
of the repo_* family.
Closes #12
140 lines
4.9 KiB
Go
140 lines
4.9 KiB
Go
package tools_test
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"io"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"testing"
|
|
|
|
"gitea.d-ma.be/mathias/gitea-mcp/internal/allowlist"
|
|
"gitea.d-ma.be/mathias/gitea-mcp/internal/gitea"
|
|
"gitea.d-ma.be/mathias/gitea-mcp/internal/tools"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func newRepoUpdateTool(srvURL string) *tools.RepoUpdate {
|
|
return tools.NewRepoUpdate(gitea.NewClient(srvURL, "tok"), allowlist.New([]string{"mathias"}))
|
|
}
|
|
|
|
// TestRepoUpdateArchive: happy path — set archived=true.
|
|
func TestRepoUpdateArchive(t *testing.T) {
|
|
var patchedBody []byte
|
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
require.Equal(t, http.MethodPatch, r.Method)
|
|
require.Equal(t, "/api/v1/repos/mathias/old-svc", r.URL.Path)
|
|
patchedBody, _ = io.ReadAll(r.Body)
|
|
w.Header().Set("Content-Type", "application/json")
|
|
_, _ = w.Write([]byte(`{"name":"old-svc","full_name":"mathias/old-svc","default_branch":"main","template":false,"private":false}`))
|
|
}))
|
|
defer srv.Close()
|
|
|
|
tool := newRepoUpdateTool(srv.URL)
|
|
result, err := tool.Call(context.Background(), json.RawMessage(
|
|
`{"owner":"mathias","name":"old-svc","archived":true}`,
|
|
))
|
|
require.NoError(t, err)
|
|
|
|
// Wire payload only contains the field that was actually set.
|
|
var sent map[string]any
|
|
require.NoError(t, json.Unmarshal(patchedBody, &sent))
|
|
assert.Equal(t, true, sent["archived"])
|
|
assert.NotContains(t, sent, "description")
|
|
assert.NotContains(t, sent, "private")
|
|
assert.NotContains(t, sent, "website")
|
|
assert.NotContains(t, sent, "template")
|
|
|
|
var repo gitea.Repo
|
|
require.NoError(t, json.Unmarshal(result, &repo))
|
|
assert.Equal(t, "mathias/old-svc", repo.FullName)
|
|
}
|
|
|
|
// TestRepoUpdateMultipleFields: set description + template flag in one call.
|
|
func TestRepoUpdateMultipleFields(t *testing.T) {
|
|
var patchedBody []byte
|
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
patchedBody, _ = io.ReadAll(r.Body)
|
|
w.Header().Set("Content-Type", "application/json")
|
|
_, _ = w.Write([]byte(`{"name":"template-go-agent","full_name":"mathias/template-go-agent","description":"Go agent template","template":true}`))
|
|
}))
|
|
defer srv.Close()
|
|
|
|
tool := newRepoUpdateTool(srv.URL)
|
|
_, err := tool.Call(context.Background(), json.RawMessage(
|
|
`{"owner":"mathias","name":"template-go-agent","description":"Go agent template","template":true}`,
|
|
))
|
|
require.NoError(t, err)
|
|
|
|
var sent map[string]any
|
|
require.NoError(t, json.Unmarshal(patchedBody, &sent))
|
|
assert.Equal(t, "Go agent template", sent["description"])
|
|
assert.Equal(t, true, sent["template"])
|
|
assert.NotContains(t, sent, "archived")
|
|
assert.NotContains(t, sent, "private")
|
|
}
|
|
|
|
// TestRepoUpdateNoFieldsRejected: zero updatable fields → validation error before network.
|
|
func TestRepoUpdateNoFieldsRejected(t *testing.T) {
|
|
tool := tools.NewRepoUpdate(
|
|
gitea.NewClient("http://unused", ""),
|
|
allowlist.New([]string{"mathias"}),
|
|
)
|
|
_, err := tool.Call(context.Background(), json.RawMessage(
|
|
`{"owner":"mathias","name":"some-repo"}`,
|
|
))
|
|
require.Error(t, err)
|
|
assert.ErrorIs(t, err, gitea.ErrValidation)
|
|
}
|
|
|
|
// TestRepoUpdateMakePublic: explicit private=false is allowed; wire payload carries the false.
|
|
// (The destructive nature is warned about in the tool description, not blocked by the tool.)
|
|
func TestRepoUpdateMakePublic(t *testing.T) {
|
|
var patchedBody []byte
|
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
patchedBody, _ = io.ReadAll(r.Body)
|
|
w.Header().Set("Content-Type", "application/json")
|
|
_, _ = w.Write([]byte(`{"name":"open-repo","full_name":"mathias/open-repo","private":false}`))
|
|
}))
|
|
defer srv.Close()
|
|
|
|
tool := newRepoUpdateTool(srv.URL)
|
|
_, err := tool.Call(context.Background(), json.RawMessage(
|
|
`{"owner":"mathias","name":"open-repo","private":false}`,
|
|
))
|
|
require.NoError(t, err)
|
|
|
|
var sent map[string]any
|
|
require.NoError(t, json.Unmarshal(patchedBody, &sent))
|
|
assert.Equal(t, false, sent["private"])
|
|
}
|
|
|
|
// TestRepoUpdateAllowlistRejects: owner outside allowlist denied without network call.
|
|
func TestRepoUpdateAllowlistRejects(t *testing.T) {
|
|
tool := tools.NewRepoUpdate(
|
|
gitea.NewClient("http://unused", ""),
|
|
allowlist.New([]string{"mathias"}),
|
|
)
|
|
_, err := tool.Call(context.Background(), json.RawMessage(
|
|
`{"owner":"evil","name":"some-repo","archived":true}`,
|
|
))
|
|
require.Error(t, err)
|
|
}
|
|
|
|
// TestRepoUpdateUpstreamError: server 500 propagates as ErrUpstream.
|
|
func TestRepoUpdateUpstreamError(t *testing.T) {
|
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, _ *http.Request) {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
_, _ = w.Write([]byte(`{"message":"internal"}`))
|
|
}))
|
|
defer srv.Close()
|
|
|
|
tool := newRepoUpdateTool(srv.URL)
|
|
_, err := tool.Call(context.Background(), json.RawMessage(
|
|
`{"owner":"mathias","name":"some-repo","archived":true}`,
|
|
))
|
|
require.Error(t, err)
|
|
assert.ErrorIs(t, err, gitea.ErrUpstream)
|
|
}
|