// Vikunja is a to-do list application to facilitate your life. // Copyright 2018-2021 Vikunja and contributors. All rights reserved. // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public Licensee as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public Licensee for more details. // // You should have received a copy of the GNU Affero General Public Licensee // along with this program. If not, see . package models import ( "code.vikunja.io/api/pkg/user" "xorm.io/builder" "xorm.io/xorm" ) // ProjectUIDs hold all kinds of user IDs from accounts who have somehow access to a project type ProjectUIDs struct { ProjectOwnerID int64 `xorm:"projectOwner"` NamespaceUserID int64 `xorm:"unID"` ProjectUserID int64 `xorm:"ulID"` NamespaceOwnerUserID int64 `xorm:"nOwner"` TeamNamespaceUserID int64 `xorm:"tnUID"` TeamProjectUserID int64 `xorm:"tlUID"` } // ProjectUsersFromProject returns a project with all users who have access to a project, regardless of the method which gave them access func ProjectUsersFromProject(s *xorm.Session, l *Project, search string) (users []*user.User, err error) { userids := []*ProjectUIDs{} err = s. Select(`l.owner_id as projectOwner, un.user_id as unID, ul.user_id as ulID, n.owner_id as nOwner, tm.user_id as tnUID, tm2.user_id as tlUID`). Table("projects"). Alias("l"). // User stuff Join("LEFT", []string{"users_namespaces", "un"}, "un.namespace_id = l.namespace_id"). Join("LEFT", []string{"users_projects", "ul"}, "ul.project_id = l.id"). Join("LEFT", []string{"namespaces", "n"}, "n.id = l.namespace_id"). // Team stuff Join("LEFT", []string{"team_namespaces", "tn"}, " l.namespace_id = tn.namespace_id"). Join("LEFT", []string{"team_members", "tm"}, "tm.team_id = tn.team_id"). Join("LEFT", []string{"team_projects", "tl"}, "l.id = tl.project_id"). Join("LEFT", []string{"team_members", "tm2"}, "tm2.team_id = tl.team_id"). // The actual condition Where( builder.Or( builder.Or(builder.Eq{"ul.right": RightRead}), builder.Or(builder.Eq{"un.right": RightRead}), builder.Or(builder.Eq{"tl.right": RightRead}), builder.Or(builder.Eq{"tn.right": RightRead}), builder.Or(builder.Eq{"ul.right": RightWrite}), builder.Or(builder.Eq{"un.right": RightWrite}), builder.Or(builder.Eq{"tl.right": RightWrite}), builder.Or(builder.Eq{"tn.right": RightWrite}), builder.Or(builder.Eq{"ul.right": RightAdmin}), builder.Or(builder.Eq{"un.right": RightAdmin}), builder.Or(builder.Eq{"tl.right": RightAdmin}), builder.Or(builder.Eq{"tn.right": RightAdmin}), ), builder.Eq{"l.id": l.ID}, ). Find(&userids) if err != nil { return } // Remove duplicates from the project of ids and make it a slice uidmap := make(map[int64]bool) uidmap[l.OwnerID] = true for _, u := range userids { uidmap[u.ProjectUserID] = true uidmap[u.NamespaceOwnerUserID] = true uidmap[u.NamespaceUserID] = true uidmap[u.TeamProjectUserID] = true uidmap[u.TeamNamespaceUserID] = true } uids := make([]int64, 0, len(uidmap)) for id := range uidmap { uids = append(uids, id) } var cond builder.Cond if len(uids) > 0 { cond = builder.In("id", uids) } users, err = user.ProjectUsers(s, search, &user.ProjectUserOpts{ AdditionalCond: cond, ReturnAllIfNoSearchProvided: true, }) return }