Skip to content

Commit 8ad2697

Browse files
davidsvantessonzeripath
authored andcommitted
Recalculate repository access only for specific user (#8481)
* Recalculate repository access only for specific user Signed-off-by: David Svantesson <davidsvantesson@gmail.com> * Handle user repositories as well, and only add access if minimum mode * Need to get repo owner to check if organization
1 parent 733c898 commit 8ad2697

File tree

3 files changed

+64
-8
lines changed

3 files changed

+64
-8
lines changed

models/access.go

+49
Original file line numberDiff line numberDiff line change
@@ -246,6 +246,55 @@ func (repo *Repository) recalculateTeamAccesses(e Engine, ignTeamID int64) (err
246246
return repo.refreshAccesses(e, accessMap)
247247
}
248248

249+
// recalculateUserAccess recalculates new access for a single user
250+
// Usable if we know access only affected one user
251+
func (repo *Repository) recalculateUserAccess(e Engine, uid int64) (err error) {
252+
minMode := AccessModeRead
253+
if !repo.IsPrivate {
254+
minMode = AccessModeWrite
255+
}
256+
257+
accessMode := AccessModeNone
258+
collaborator, err := repo.getCollaboration(e, uid)
259+
if err != nil {
260+
return err
261+
} else if collaborator != nil {
262+
accessMode = collaborator.Mode
263+
}
264+
265+
if err = repo.getOwner(e); err != nil {
266+
return err
267+
} else if repo.Owner.IsOrganization() {
268+
var teams []Team
269+
if err := e.Join("INNER", "team_repo", "team_repo.team_id = team.id").
270+
Join("INNER", "team_user", "team_user.team_id = team.id").
271+
Where("team.org_id = ?", repo.OwnerID).
272+
And("team_repo.repo_id=?", repo.ID).
273+
And("team_user.uid=?", uid).
274+
Find(&teams); err != nil {
275+
return err
276+
}
277+
278+
for _, t := range teams {
279+
if t.IsOwnerTeam() {
280+
t.Authorize = AccessModeOwner
281+
}
282+
283+
accessMode = maxAccessMode(accessMode, t.Authorize)
284+
}
285+
}
286+
287+
// Delete old user accesses and insert new one for repository.
288+
if _, err = e.Delete(&Access{RepoID: repo.ID, UserID: uid}); err != nil {
289+
return fmt.Errorf("delete old user accesses: %v", err)
290+
} else if accessMode >= minMode {
291+
if _, err = e.Insert(&Access{RepoID: repo.ID, UserID: uid, Mode: accessMode}); err != nil {
292+
return fmt.Errorf("insert new user accesses: %v", err)
293+
}
294+
}
295+
return nil
296+
}
297+
249298
func (repo *Repository) recalculateAccesses(e Engine) error {
250299
if repo.Owner.IsOrganization() {
251300
return repo.recalculateTeamAccesses(e, 0)

models/org_team.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -723,7 +723,7 @@ func AddTeamMember(team *Team, userID int64) error {
723723

724724
// Give access to team repositories.
725725
for _, repo := range team.Repos {
726-
if err := repo.recalculateTeamAccesses(sess, 0); err != nil {
726+
if err := repo.recalculateUserAccess(sess, userID); err != nil {
727727
return err
728728
}
729729
if setting.Service.AutoWatchNewRepos {
@@ -768,7 +768,7 @@ func removeTeamMember(e *xorm.Session, team *Team, userID int64) error {
768768

769769
// Delete access to team repositories.
770770
for _, repo := range team.Repos {
771-
if err := repo.recalculateTeamAccesses(e, 0); err != nil {
771+
if err := repo.recalculateUserAccess(e, userID); err != nil {
772772
return err
773773
}
774774

models/repo_collaboration.go

+13-6
Original file line numberDiff line numberDiff line change
@@ -41,12 +41,7 @@ func (repo *Repository) AddCollaborator(u *User) error {
4141
return err
4242
}
4343

44-
if repo.Owner.IsOrganization() {
45-
err = repo.recalculateTeamAccesses(sess, 0)
46-
} else {
47-
err = repo.recalculateAccesses(sess)
48-
}
49-
if err != nil {
44+
if err = repo.recalculateUserAccess(sess, u.ID); err != nil {
5045
return fmt.Errorf("recalculateAccesses 'team=%v': %v", repo.Owner.IsOrganization(), err)
5146
}
5247

@@ -89,6 +84,18 @@ func (repo *Repository) GetCollaborators() ([]*Collaborator, error) {
8984
return repo.getCollaborators(x)
9085
}
9186

87+
func (repo *Repository) getCollaboration(e Engine, uid int64) (*Collaboration, error) {
88+
collaboration := &Collaboration{
89+
RepoID: repo.ID,
90+
UserID: uid,
91+
}
92+
has, err := e.Get(collaboration)
93+
if !has {
94+
collaboration = nil
95+
}
96+
return collaboration, err
97+
}
98+
9299
func (repo *Repository) isCollaborator(e Engine, userID int64) (bool, error) {
93100
return e.Get(&Collaboration{RepoID: repo.ID, UserID: userID})
94101
}

0 commit comments

Comments
 (0)