Skip to content
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
Show all changes
51 commits
Select commit Hold shift + click to select a range
fad641b
added DeleteOrgRepos func
karthikbhandary2 Feb 24, 2026
226ace7
Merge branch 'go-gitea:main' into main
karthikbhandary2 Feb 25, 2026
acf9dc0
Merge branch 'go-gitea:main' into main
karthikbhandary2 Feb 26, 2026
e940dd2
Verified and tested the DeleteRepoOrgs
karthikbhandary2 Feb 26, 2026
c319345
ran make generate-swagger
karthikbhandary2 Feb 26, 2026
e4c5c38
error handling and swagger updated
karthikbhandary2 Feb 26, 2026
ed4b335
Removed additional check for permissions in DeleteOrgRepos
karthikbhandary2 Feb 26, 2026
5b819b4
updated DeleteOrgRepos to delete in the background
karthikbhandary2 Feb 27, 2026
9f31e6c
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 6, 2026
a5092e5
added system notice
karthikbhandary2 Mar 6, 2026
20e3fe4
added batch processing and other fixes
karthikbhandary2 Mar 7, 2026
988ec17
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 7, 2026
008415e
added retry logic
karthikbhandary2 Mar 7, 2026
63f9dbf
reverted back to old logic
karthikbhandary2 Mar 7, 2026
204100a
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 7, 2026
9f7deee
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 8, 2026
bef6617
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 8, 2026
ae7b1a1
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 9, 2026
84e3487
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 9, 2026
974def8
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 10, 2026
e29e9bf
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 17, 2026
ce000b0
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 20, 2026
95a0d70
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 21, 2026
cff469d
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 22, 2026
84ef882
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 22, 2026
348feff
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 22, 2026
cfcbdca
updated DeleteOrgRepos to use GetRepositoryByID
Mar 22, 2026
a36a3b5
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 23, 2026
aea3d2b
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 23, 2026
02343a2
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 24, 2026
1633be5
Merge branch 'main' into main
karthikbhandary2 Mar 27, 2026
53a1aac
merge conflict fix
Mar 27, 2026
8b494d6
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 28, 2026
41a1ed0
Merge branch 'main' into main
karthikbhandary2 Apr 2, 2026
35587c0
Merge branch 'go-gitea:main' into main
karthikbhandary2 Apr 3, 2026
e402e5f
Merge branch 'go-gitea:main' into main
karthikbhandary2 Apr 3, 2026
c8b09c8
Merge branch 'main' into main
karthikbhandary2 Apr 4, 2026
88381aa
add test to check if deletes were successful
Apr 4, 2026
0f0ec79
Merge branch 'main' into main
karthikbhandary2 Apr 5, 2026
41cff2f
refactor
wxiaoguang Apr 5, 2026
f723d35
refactor
wxiaoguang Apr 5, 2026
8ed094e
Merge branch 'go-gitea:main' into main
karthikbhandary2 Apr 6, 2026
76189cc
tests updated and delete endpoint updated
karthikbhandary2 Apr 6, 2026
bc073ce
Merge branch 'main' into main
karthikbhandary2 Apr 6, 2026
81fc298
Merge branch 'main' into main
karthikbhandary2 Apr 7, 2026
9e71545
clean up AI slop
wxiaoguang Apr 7, 2026
115b582
fine tune
wxiaoguang Apr 7, 2026
8017f5d
fine tune
wxiaoguang Apr 7, 2026
bea4c1c
Merge branch 'main' into main
karthikbhandary2 Apr 7, 2026
bbc3491
remove unnecessary test
wxiaoguang Apr 7, 2026
7af8f5f
Merge branch 'main' into main
karthikbhandary2 Apr 8, 2026
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
22 changes: 22 additions & 0 deletions modules/structs/repo.go
Original file line number Diff line number Diff line change
Expand Up @@ -440,3 +440,25 @@ type UpdateRepoAvatarOption struct {
// image must be base64 encoded
Image string `json:"image" binding:"Required"`
}

// DeleteOrgReposResponse represents the response for deleting organization repositories
// swagger:model
type DeleteOrgReposResponse struct {
// Number of repositories successfully deleted
SuccessCount int `json:"success_count"`
Comment thread
wxiaoguang marked this conversation as resolved.
Outdated
// Number of repositories that failed to delete
FailureCount int `json:"failure_count"`
// List of repository names that were deleted
Deleted []string `json:"deleted"`
// Details about repositories that failed to delete
Failed []DeleteRepoFailure `json:"failed"`
}

// DeleteRepoFailure represents a repository that failed to delete
Comment thread
karthikbhandary2 marked this conversation as resolved.
Outdated
// swagger:model
type DeleteRepoFailure struct {
// Repository name
RepoName string `json:"repo_name"`
// Message to be displayed
Message string `json:"reason"`
}
3 changes: 2 additions & 1 deletion routers/api/v1/api.go
Original file line number Diff line number Diff line change
Expand Up @@ -1621,7 +1621,8 @@ func Routes() *web.Router {
Delete(reqToken(), reqOrgOwnership(), org.Delete)
m.Post("/rename", reqToken(), reqOrgOwnership(), bind(api.RenameOrgOption{}), org.Rename)
m.Combo("/repos").Get(user.ListOrgRepos).
Post(reqToken(), bind(api.CreateRepoOption{}), repo.CreateOrgRepo)
Post(reqToken(), bind(api.CreateRepoOption{}), repo.CreateOrgRepo).
Delete(reqToken(), reqOrgOwnership(), org.DeleteOrgRepos)
Comment thread
karthikbhandary2 marked this conversation as resolved.
Outdated
Comment thread
wxiaoguang marked this conversation as resolved.
Outdated
m.Group("/members", func() {
m.Get("", reqToken(), org.ListMembers)
m.Combo("/{username}").Get(reqToken(), org.IsMember).
Expand Down
48 changes: 48 additions & 0 deletions routers/api/v1/org/org.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,9 @@ import (
"code.gitea.io/gitea/models/db"
"code.gitea.io/gitea/models/organization"
"code.gitea.io/gitea/models/perm"
repo_model "code.gitea.io/gitea/models/repo"
user_model "code.gitea.io/gitea/models/user"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/optional"
api "code.gitea.io/gitea/modules/structs"
"code.gitea.io/gitea/modules/web"
Expand All @@ -21,6 +23,7 @@ import (
"code.gitea.io/gitea/services/convert"
feed_service "code.gitea.io/gitea/services/feed"
"code.gitea.io/gitea/services/org"
repo_service "code.gitea.io/gitea/services/repository"
user_service "code.gitea.io/gitea/services/user"
)

Expand Down Expand Up @@ -493,3 +496,48 @@ func ListOrgActivityFeeds(ctx *context.APIContext) {

ctx.JSON(http.StatusOK, convert.ToActivities(ctx, feeds, ctx.Doer))
}

func DeleteOrgRepos(ctx *context.APIContext) {
Comment thread
karthikbhandary2 marked this conversation as resolved.
// swagger:operation DELETE /orgs/{org}/repos organization orgDeleteRepos
// ---
// summary: Delete all repositories in an organization
// produces:
// - application/json
Comment thread
karthikbhandary2 marked this conversation as resolved.
// parameters:
// - name: org
// in: path
// description: name of the organization
// type: string
// required: true
// responses:
// "200":
// "$ref": "#/responses/DeleteOrgReposList"
// "403":
// "$ref": "#/responses/forbidden"
Comment thread
karthikbhandary2 marked this conversation as resolved.
// "404":
// "$ref": "#/responses/notFound"
Comment thread
karthikbhandary2 marked this conversation as resolved.
org := ctx.Org.Organization
repos, err := repo_model.GetOrgRepositories(ctx, org.ID)
Comment thread
karthikbhandary2 marked this conversation as resolved.
Outdated
if err != nil {
ctx.APIErrorInternal(err)
return
}
response := &api.DeleteOrgReposResponse{
Deleted: []string{},
Failed: []api.DeleteRepoFailure{},
}
for _, repo := range repos {
if err := repo_service.DeleteRepository(ctx, ctx.Doer, repo, true); err != nil {
log.Error("Error deleting repo %s: %v", repo.Name, err)
response.Failed = append(response.Failed, api.DeleteRepoFailure{
RepoName: repo.Name,
Message: "Failed to delete repository",
Comment thread
wxiaoguang marked this conversation as resolved.
Outdated
})
} else {
response.Deleted = append(response.Deleted, repo.Name)
}
}
response.SuccessCount = len(response.Deleted)
response.FailureCount = len(response.Failed)
ctx.JSON(http.StatusOK, response)
}
Comment thread
karthikbhandary2 marked this conversation as resolved.
8 changes: 8 additions & 0 deletions routers/api/v1/swagger/org.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,3 +41,11 @@ type swaggerResponseOrganizationPermissions struct {
// in:body
Body api.OrganizationPermissions `json:"body"`
}

// DeleteOrgReposList
// swagger:response DeleteOrgReposList
type swaggerDeleteOrgReposList struct {
// List of successfully deleted repositories and failures
//in:body
Body api.DeleteOrgReposResponse `json:"body"`
}
88 changes: 88 additions & 0 deletions templates/swagger/v1_json.tmpl

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

97 changes: 97 additions & 0 deletions tests/integration/api_org_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -249,3 +249,100 @@ func TestAPIOrgGeneral(t *testing.T) {
MakeRequest(t, req, http.StatusForbidden)
})
}

func TestAPIDeleteOrgRepos(t *testing.T) {
Comment thread
lunny marked this conversation as resolved.
Outdated
defer tests.PrepareTestEnv(t)()

t.Run("Delete all repos successfully", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
Comment thread
karthikbhandary2 marked this conversation as resolved.
Outdated

// Create test org with owner
session := loginUser(t, "user1")
token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteOrganization, auth_model.AccessTokenScopeWriteRepository)

orgName := "test_delete_org"
req := NewRequestWithJSON(t, "POST", "/api/v1/orgs", &api.CreateOrgOption{
UserName: orgName,
}).AddTokenAuth(token)
MakeRequest(t, req, http.StatusCreated)

// Create 60 repos to test efficiency
for i := range 60 {
Comment thread
wxiaoguang marked this conversation as resolved.
Outdated
repoName := fmt.Sprintf("test_repo_%d", i)
req := NewRequestWithJSON(t, "POST", fmt.Sprintf("/api/v1/org/%s/repos", orgName), &api.CreateRepoOption{
Name: repoName,
}).AddTokenAuth(token)
MakeRequest(t, req, http.StatusCreated)
}

// Delete all repos
req = NewRequest(t, "DELETE", fmt.Sprintf("/api/v1/orgs/%s/repos", orgName)).AddTokenAuth(token)
resp := MakeRequest(t, req, http.StatusOK)

var result api.DeleteOrgReposResponse
DecodeJSON(t, resp, &result)

assert.Equal(t, 60, result.SuccessCount)
assert.Equal(t, 0, result.FailureCount)
assert.Len(t, result.Deleted, 60)
assert.Empty(t, result.Failed)
})

t.Run("Verify response structure", func(t *testing.T) {
Comment thread
karthikbhandary2 marked this conversation as resolved.
Outdated
defer tests.PrintCurrentTest(t)()

session := loginUser(t, "user1")
token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteOrganization, auth_model.AccessTokenScopeWriteRepository)

orgName := "test_response_org"
req := NewRequestWithJSON(t, "POST", "/api/v1/orgs", &api.CreateOrgOption{
UserName: orgName,
}).AddTokenAuth(token)
MakeRequest(t, req, http.StatusCreated)

// Create a few repos
for i := range 3 {
req = NewRequestWithJSON(t, "POST", fmt.Sprintf("/api/v1/org/%s/repos", orgName), &api.CreateRepoOption{
Name: fmt.Sprintf("repo_%d", i),
}).AddTokenAuth(token)
MakeRequest(t, req, http.StatusCreated)
}

// Delete all repos
req = NewRequest(t, "DELETE", fmt.Sprintf("/api/v1/orgs/%s/repos", orgName)).AddTokenAuth(token)
resp := MakeRequest(t, req, http.StatusOK)

var result api.DeleteOrgReposResponse
DecodeJSON(t, resp, &result)

// Verify response structure
assert.Equal(t, 3, result.SuccessCount)
assert.Equal(t, 0, result.FailureCount)
assert.Len(t, result.Deleted, 3)
assert.Empty(t, result.Failed)
assert.NotNil(t, result.Deleted)
assert.NotNil(t, result.Failed)
})

t.Run("Fail without permissions", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()

// user2 is owner of org3
ownerSession := loginUser(t, "user2")
ownerToken := getTokenForLoggedInUser(t, ownerSession, auth_model.AccessTokenScopeWriteOrganization, auth_model.AccessTokenScopeWriteRepository)

// Create repo in org3
req := NewRequestWithJSON(t, "POST", "/api/v1/org/org3/repos", &api.CreateRepoOption{
Name: "test_perm_repo",
}).AddTokenAuth(ownerToken)
MakeRequest(t, req, http.StatusCreated)

// user4 is not owner of org3
nonOwnerSession := loginUser(t, "user4")
nonOwnerToken := getTokenForLoggedInUser(t, nonOwnerSession, auth_model.AccessTokenScopeWriteOrganization)

// Try to delete repos without owner permission
req = NewRequest(t, "DELETE", "/api/v1/orgs/org3/repos").AddTokenAuth(nonOwnerToken)
MakeRequest(t, req, http.StatusForbidden)
})
}
Loading