organizations.go 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653
  1. // Copyright 2022 The Gogs Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package db
  5. import (
  6. "context"
  7. "fmt"
  8. "os"
  9. "strings"
  10. "github.com/pkg/errors"
  11. "gorm.io/gorm"
  12. "gogs.io/gogs/internal/dbutil"
  13. "gogs.io/gogs/internal/errutil"
  14. "gogs.io/gogs/internal/repoutil"
  15. "gogs.io/gogs/internal/userutil"
  16. )
  17. // OrganizationsStore is the persistent interface for organizations.
  18. type OrganizationsStore interface {
  19. // Create creates a new organization with the initial owner and persists to
  20. // database. It returns ErrNameNotAllowed if the given name or pattern of the
  21. // name is not allowed as an organization name, or ErrOrganizationAlreadyExist
  22. // when a user or an organization with same name already exists.
  23. Create(ctx context.Context, name string, ownerID int64, opts CreateOrganizationOptions) (*Organization, error)
  24. // GetByName returns the organization with given name.
  25. GetByName(ctx context.Context, name string) (*Organization, error)
  26. // SearchByName returns a list of organizations whose username or full name
  27. // matches the given keyword case-insensitively. Results are paginated by given
  28. // page and page size, and sorted by the given order (e.g. "id DESC"). A total
  29. // count of all results is also returned. If the order is not given, it's up to
  30. // the database to decide.
  31. SearchByName(ctx context.Context, keyword string, page, pageSize int, orderBy string) ([]*Organization, int64, error)
  32. // List returns a list of organizations filtered by options.
  33. List(ctx context.Context, opts ListOrganizationsOptions) ([]*Organization, error)
  34. // CountByUser returns the number of organizations the user is a member of.
  35. CountByUser(ctx context.Context, userID int64) (int64, error)
  36. // Count returns the total number of organizations.
  37. Count(ctx context.Context) int64
  38. // DeleteByID deletes the given organization and all their resources. It returns
  39. // ErrOrganizationOwnRepos when the user still has repository ownership.
  40. DeleteByID(ctx context.Context, orgID int64) error
  41. // AddMember adds a new member to the given organization.
  42. AddMember(ctx context.Context, orgID, userID int64) error
  43. // RemoveMember removes a member from the given organization.
  44. RemoveMember(ctx context.Context, orgID, userID int64) error
  45. // HasMember returns whether the given user is a member of the organization
  46. // (first), and whether the organization membership is public (second).
  47. HasMember(ctx context.Context, orgID, userID int64) (bool, bool)
  48. // ListMembers returns all members of the given organization, and sorted by the
  49. // given order (e.g. "id ASC").
  50. ListMembers(ctx context.Context, orgID int64, opts ListOrgMembersOptions) ([]*User, error)
  51. // IsOwnedBy returns true if the given user is an owner of the organization.
  52. IsOwnedBy(ctx context.Context, orgID, userID int64) bool
  53. // SetMemberVisibility sets the visibility of the given user in the organization.
  54. SetMemberVisibility(ctx context.Context, orgID, userID int64, public bool) error
  55. // GetTeamByName returns the team with given name under the given organization.
  56. // It returns ErrTeamNotExist whe not found.
  57. GetTeamByName(ctx context.Context, orgID int64, name string) (*Team, error)
  58. // AccessibleRepositoriesByUser returns a range of repositories in the
  59. // organization that the user has access to and the total number of it. Results
  60. // are paginated by given page and page size, and sorted by the given order
  61. // (e.g. "updated_unix DESC").
  62. AccessibleRepositoriesByUser(ctx context.Context, orgID, userID int64, page, pageSize int, opts AccessibleRepositoriesByUserOptions) ([]*Repository, int64, error)
  63. }
  64. var Organizations OrganizationsStore
  65. var _ OrganizationsStore = (*organizations)(nil)
  66. type organizations struct {
  67. *gorm.DB
  68. }
  69. // NewOrganizationsStore returns a persistent interface for orgs with given
  70. // database connection.
  71. func NewOrganizationsStore(db *gorm.DB) OrganizationsStore {
  72. return &organizations{DB: db}
  73. }
  74. func (*organizations) recountMembers(tx *gorm.DB, orgID int64) error {
  75. /*
  76. Equivalent SQL for PostgreSQL:
  77. UPDATE "user"
  78. SET num_members = (
  79. SELECT COUNT(*) FROM org_user WHERE org_id = @orgID
  80. )
  81. WHERE id = @orgID
  82. */
  83. err := tx.Model(&User{}).
  84. Where("id = ?", orgID).
  85. Update(
  86. "num_members",
  87. tx.Model(&OrgUser{}).Select("COUNT(*)").Where("org_id = ?", orgID),
  88. ).
  89. Error
  90. if err != nil {
  91. return errors.Wrap(err, `update "user.num_members"`)
  92. }
  93. return nil
  94. }
  95. func (db *organizations) AddMember(ctx context.Context, orgID, userID int64) error {
  96. return db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
  97. ou := &OrgUser{
  98. UserID: userID,
  99. OrgID: orgID,
  100. }
  101. result := tx.FirstOrCreate(ou, ou)
  102. if result.Error != nil {
  103. return errors.Wrap(result.Error, "upsert")
  104. } else if result.RowsAffected <= 0 {
  105. return nil // Relation already exists
  106. }
  107. return db.recountMembers(tx, orgID)
  108. })
  109. }
  110. type ErrLastOrgOwner struct {
  111. args map[string]any
  112. }
  113. func IsErrLastOrgOwner(err error) bool {
  114. return errors.As(err, &ErrLastOrgOwner{})
  115. }
  116. func (err ErrLastOrgOwner) Error() string {
  117. return fmt.Sprintf("user is the last owner of the organization: %v", err.args)
  118. }
  119. func (db *organizations) RemoveMember(ctx context.Context, orgID, userID int64) error {
  120. ou, err := db.getOrgUser(ctx, orgID, userID)
  121. if err != nil {
  122. if errors.Is(err, gorm.ErrRecordNotFound) {
  123. return nil // Not a member
  124. }
  125. return errors.Wrap(err, "check organization membership")
  126. }
  127. // Check if the member to remove is the last owner.
  128. if ou.IsOwner {
  129. t, err := db.GetTeamByName(ctx, orgID, TeamNameOwners)
  130. if err != nil {
  131. return errors.Wrap(err, "get owners team")
  132. } else if t.NumMembers == 1 {
  133. return ErrLastOrgOwner{args: map[string]any{"orgID": orgID, "userID": userID}}
  134. }
  135. }
  136. return db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
  137. repoIDsConds := db.accessibleRepositoriesByUser(tx, orgID, userID, accessibleRepositoriesByUserOptions{}).Select("repository.id")
  138. err := tx.Where("user_id = ? AND repo_id IN (?)", userID, repoIDsConds).Delete(&Watch{}).Error
  139. if err != nil {
  140. return errors.Wrap(err, "unwatch repositories")
  141. }
  142. err = tx.Table("repository").
  143. Where("id IN (?)", repoIDsConds).
  144. UpdateColumn("num_watches", gorm.Expr("num_watches - 1")).
  145. Error
  146. if err != nil {
  147. return errors.Wrap(err, `decrease "repository.num_watches"`)
  148. }
  149. err = tx.Where("user_id = ? AND repo_id IN (?)", userID, repoIDsConds).Delete(&Access{}).Error
  150. if err != nil {
  151. return errors.Wrap(err, "delete repository accesses")
  152. }
  153. err = tx.Where("user_id = ? AND repo_id IN (?)", userID, repoIDsConds).Delete(&Collaboration{}).Error
  154. if err != nil {
  155. return errors.Wrap(err, "delete repository collaborations")
  156. }
  157. /*
  158. Equivalent SQL for PostgreSQL:
  159. UPDATE "team"
  160. SET num_members = num_members - 1
  161. WHERE id IN (
  162. SELECT team_id FROM "team_user"
  163. WHERE team_user.org_id = @orgID AND uid = @userID)
  164. )
  165. */
  166. err = tx.Table("team").
  167. Where(`id IN (?)`, tx.
  168. Select("team_id").
  169. Table("team_user").
  170. Where("org_id = ? AND uid = ?", orgID, userID),
  171. ).
  172. UpdateColumn("num_members", gorm.Expr("num_members - 1")).
  173. Error
  174. if err != nil {
  175. return errors.Wrap(err, `decrease "team.num_members"`)
  176. }
  177. err = tx.Where("uid = ? AND org_id = ?", userID, orgID).Delete(&TeamUser{}).Error
  178. if err != nil {
  179. return errors.Wrap(err, "delete team membership")
  180. }
  181. err = tx.Where("uid = ? AND org_id = ?", userID, orgID).Delete(&OrgUser{}).Error
  182. if err != nil {
  183. return errors.Wrap(err, "delete organization membership")
  184. }
  185. return db.recountMembers(tx, orgID)
  186. })
  187. }
  188. type OrderBy int
  189. const (
  190. OrderByIDAsc OrderBy = iota + 1
  191. OrderByUpdatedDesc
  192. )
  193. type accessibleRepositoriesByUserOptions struct {
  194. orderBy OrderBy
  195. page int
  196. pageSize int
  197. }
  198. func (*organizations) accessibleRepositoriesByUser(tx *gorm.DB, orgID, userID int64, opts accessibleRepositoriesByUserOptions) *gorm.DB {
  199. /*
  200. Equivalent SQL for PostgreSQL:
  201. <SELECT * FROM "repository">
  202. JOIN team_repo ON repository.id = team_repo.repo_id
  203. WHERE
  204. owner_id = @orgID
  205. AND (
  206. team_repo.team_id IN (
  207. SELECT team_id FROM "team_user"
  208. WHERE team_user.org_id = @orgID AND uid = @userID)
  209. )
  210. OR (repository.is_private = FALSE AND repository.is_unlisted = FALSE)
  211. )
  212. [ORDER BY updated_unix DESC]
  213. [LIMIT @limit OFFSET @offset]
  214. */
  215. conds := tx.
  216. Joins("JOIN team_repo ON repository.id = team_repo.repo_id").
  217. Where("owner_id = ? AND (?)", orgID, tx.
  218. Where("team_repo.team_id IN (?)", tx.
  219. Select("team_id").
  220. Table("team_user").
  221. Where("team_user.org_id = ? AND uid = ?", orgID, userID),
  222. ).
  223. Or("repository.is_private = ? AND repository.is_unlisted = ?", false, false),
  224. )
  225. if opts.orderBy == OrderByUpdatedDesc {
  226. conds.Order("updated_unix DESC")
  227. }
  228. if opts.page > 0 && opts.pageSize > 0 {
  229. conds.Limit(opts.pageSize).Offset((opts.page - 1) * opts.pageSize)
  230. }
  231. return conds
  232. }
  233. type AccessibleRepositoriesByUserOptions struct {
  234. // Whether to skip counting the total number of repositories.
  235. SkipCount bool
  236. }
  237. func (db *organizations) AccessibleRepositoriesByUser(ctx context.Context, orgID, userID int64, page, pageSize int, opts AccessibleRepositoriesByUserOptions) ([]*Repository, int64, error) {
  238. conds := db.accessibleRepositoriesByUser(
  239. db.DB,
  240. orgID,
  241. userID,
  242. accessibleRepositoriesByUserOptions{
  243. orderBy: OrderByUpdatedDesc,
  244. page: page,
  245. pageSize: pageSize,
  246. },
  247. ).WithContext(ctx)
  248. repos := make([]*Repository, 0, pageSize)
  249. err := conds.Find(&repos).Error
  250. if err != nil {
  251. return nil, 0, errors.Wrap(err, "list repositories")
  252. }
  253. if opts.SkipCount {
  254. return repos, 0, nil
  255. }
  256. var count int64
  257. err = conds.Model(&Repository{}).Count(&count).Error
  258. if err != nil {
  259. return nil, 0, errors.Wrap(err, "count repositories")
  260. }
  261. return repos, count, nil
  262. }
  263. func (db *organizations) getOrgUser(ctx context.Context, orgID, userID int64) (*OrgUser, error) {
  264. var ou OrgUser
  265. return &ou, db.WithContext(ctx).Where("org_id = ? AND uid = ?", orgID, userID).First(&ou).Error
  266. }
  267. func (db *organizations) IsOwnedBy(ctx context.Context, orgID, userID int64) bool {
  268. ou, err := db.getOrgUser(ctx, orgID, userID)
  269. return err == nil && ou.IsOwner
  270. }
  271. func (db *organizations) SetMemberVisibility(ctx context.Context, orgID, userID int64, public bool) error {
  272. return db.Table("org_user").Where("org_id = ? AND uid = ?", orgID, userID).UpdateColumn("is_public", public).Error
  273. }
  274. func (db *organizations) HasMember(ctx context.Context, orgID, userID int64) (bool, bool) {
  275. ou, err := db.getOrgUser(ctx, orgID, userID)
  276. return err == nil, ou != nil && ou.IsPublic
  277. }
  278. type ListOrgMembersOptions struct {
  279. // The maximum number of members to return.
  280. Limit int
  281. }
  282. func (db *organizations) ListMembers(ctx context.Context, orgID int64, opts ListOrgMembersOptions) ([]*User, error) {
  283. /*
  284. Equivalent SQL for PostgreSQL:
  285. SELECT * FROM "user"
  286. JOIN org_user ON org_user.uid = user.id
  287. WHERE
  288. org_user.org_id = @orgID
  289. ORDER BY user.id ASC
  290. [LIMIT @limit]
  291. */
  292. conds := db.WithContext(ctx).
  293. Joins(dbutil.Quote("JOIN org_user ON org_user.uid = %s.id", "user")).
  294. Where("org_user.org_id = ?", orgID).
  295. Order(dbutil.Quote("%s.id ASC", "user"))
  296. if opts.Limit > 0 {
  297. conds.Limit(opts.Limit)
  298. }
  299. var users []*User
  300. return users, conds.Find(&users).Error
  301. }
  302. type ListOrganizationsOptions struct {
  303. // Filter by the membership with the given user ID. It cannot be set when the
  304. // OwnerID is also set.
  305. MemberID int64
  306. // Filter by the ownership with the given user ID. It cannot be set when the
  307. // MemberID is also set.
  308. OwnerID int64
  309. // Whether to include private memberships.
  310. IncludePrivateMembers bool
  311. // Order by the given field and direction. Default is OrderByIDAsc.
  312. OrderBy OrderBy
  313. // 1-based page number.
  314. Page int
  315. // Number of results per page.
  316. PageSize int
  317. }
  318. func (db *organizations) List(ctx context.Context, opts ListOrganizationsOptions) ([]*Organization, error) {
  319. if opts.MemberID > 0 && opts.OwnerID > 0 {
  320. return nil, errors.New("cannot filter by both MemberID and OwnerID")
  321. }
  322. /*
  323. Equivalent SQL for PostgreSQL:
  324. SELECT * FROM "user"
  325. [JOIN org_user ON org_user.org_id = user.id]
  326. WHERE
  327. type = @type
  328. [AND org_user.uid = (@memberID | @ownerID)
  329. AND org_user.is_public = @includePrivateMembers
  330. AND org_user.is_owner = @ownerID > 0]
  331. ORDER BY (user.id ASC | user.updated_unix DESC)
  332. [LIMIT @limit OFFSET @offset]
  333. */
  334. conds := db.WithContext(ctx).Where("type = ?", UserTypeOrganization)
  335. if opts.MemberID > 0 || opts.OwnerID > 0 || !opts.IncludePrivateMembers {
  336. conds.Joins(dbutil.Quote("JOIN org_user ON org_user.org_id = %s.id", "user"))
  337. }
  338. if opts.MemberID > 0 {
  339. conds.Where("org_user.uid = ?", opts.MemberID)
  340. } else if opts.OwnerID > 0 {
  341. conds.Where("org_user.uid = ? AND org_user.is_owner = ?", opts.OwnerID, true)
  342. }
  343. if !opts.IncludePrivateMembers {
  344. conds.Where("org_user.is_public = ?", true)
  345. }
  346. if opts.OrderBy == OrderByUpdatedDesc {
  347. conds.Order(dbutil.Quote("%s.updated_unix DESC", "user"))
  348. } else {
  349. conds.Order(dbutil.Quote("%s.id ASC", "user"))
  350. }
  351. if opts.Page > 0 && opts.PageSize > 0 {
  352. conds.Limit(opts.PageSize).Offset((opts.Page - 1) * opts.PageSize)
  353. }
  354. var orgs []*Organization
  355. return orgs, conds.Find(&orgs).Error
  356. }
  357. type CreateOrganizationOptions struct {
  358. FullName string
  359. Email string
  360. Location string
  361. Website string
  362. Description string
  363. }
  364. type ErrOrganizationAlreadyExist struct {
  365. args errutil.Args
  366. }
  367. // IsErrOrganizationAlreadyExist returns true if the underlying error has the
  368. // type ErrOrganizationAlreadyExist.
  369. func IsErrOrganizationAlreadyExist(err error) bool {
  370. return errors.As(err, &ErrOrganizationAlreadyExist{})
  371. }
  372. func (err ErrOrganizationAlreadyExist) Error() string {
  373. return fmt.Sprintf("organization already exists: %v", err.args)
  374. }
  375. func (db *organizations) Create(ctx context.Context, name string, ownerID int64, opts CreateOrganizationOptions) (*Organization, error) {
  376. err := isUsernameAllowed(name)
  377. if err != nil {
  378. return nil, err
  379. }
  380. if NewUsersStore(db.DB).IsUsernameUsed(ctx, name, 0) {
  381. return nil, ErrOrganizationAlreadyExist{
  382. args: errutil.Args{
  383. "name": name,
  384. },
  385. }
  386. }
  387. org := &Organization{
  388. LowerName: strings.ToLower(name),
  389. Name: name,
  390. FullName: opts.FullName,
  391. Email: opts.Email,
  392. Type: UserTypeOrganization,
  393. Location: opts.Location,
  394. Website: opts.Website,
  395. MaxRepoCreation: -1,
  396. IsActive: true,
  397. UseCustomAvatar: true,
  398. Description: opts.Description,
  399. NumTeams: 1, // The default "owners" team
  400. NumMembers: 1, // The initial owner
  401. }
  402. org.Rands, err = userutil.RandomSalt()
  403. if err != nil {
  404. return nil, err
  405. }
  406. org.Salt, err = userutil.RandomSalt()
  407. if err != nil {
  408. return nil, err
  409. }
  410. return org, db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
  411. err := tx.Create(org).Error
  412. if err != nil {
  413. return errors.Wrap(err, "create organization")
  414. }
  415. err = tx.Create(&OrgUser{
  416. UserID: ownerID,
  417. OrgID: org.ID,
  418. IsOwner: true,
  419. NumTeams: 1,
  420. }).Error
  421. if err != nil {
  422. return errors.Wrap(err, "create org-user relation")
  423. }
  424. team := &Team{
  425. OrgID: org.ID,
  426. LowerName: strings.ToLower(TeamNameOwners),
  427. Name: TeamNameOwners,
  428. Authorize: AccessModeOwner,
  429. NumMembers: 1,
  430. }
  431. err = tx.Create(team).Error
  432. if err != nil {
  433. return errors.Wrap(err, "create owner team")
  434. }
  435. err = tx.Create(&TeamUser{
  436. UID: ownerID,
  437. OrgID: org.ID,
  438. TeamID: team.ID,
  439. }).Error
  440. if err != nil {
  441. return errors.Wrap(err, "create team-user relation")
  442. }
  443. err = userutil.GenerateRandomAvatar(org.ID, org.Name, org.Email)
  444. if err != nil {
  445. return errors.Wrap(err, "generate organization avatar")
  446. }
  447. err = os.MkdirAll(repoutil.UserPath(org.Name), os.ModePerm)
  448. if err != nil {
  449. return errors.Wrap(err, "create organization directory")
  450. }
  451. return nil
  452. })
  453. }
  454. var _ errutil.NotFound = (*ErrUserNotExist)(nil)
  455. type ErrOrganizationNotExist struct {
  456. args errutil.Args
  457. }
  458. // IsErrOrganizationNotExist returns true if the underlying error has the type
  459. // ErrOrganizationNotExist.
  460. func IsErrOrganizationNotExist(err error) bool {
  461. return errors.As(err, &ErrOrganizationNotExist{})
  462. }
  463. func (err ErrOrganizationNotExist) Error() string {
  464. return fmt.Sprintf("organization does not exist: %v", err.args)
  465. }
  466. func (ErrOrganizationNotExist) NotFound() bool {
  467. return true
  468. }
  469. func (db *organizations) GetByName(ctx context.Context, name string) (*Organization, error) {
  470. org, err := getUserByUsername(ctx, db.DB, UserTypeOrganization, name)
  471. if err != nil {
  472. if IsErrUserNotExist(err) {
  473. return nil, ErrOrganizationNotExist{args: map[string]any{"name": name}}
  474. }
  475. return nil, errors.Wrap(err, "get organization by name")
  476. }
  477. return org, nil
  478. }
  479. func (db *organizations) SearchByName(ctx context.Context, keyword string, page, pageSize int, orderBy string) ([]*Organization, int64, error) {
  480. return searchUserByName(ctx, db.DB, UserTypeOrganization, keyword, page, pageSize, orderBy)
  481. }
  482. func (db *organizations) CountByUser(ctx context.Context, userID int64) (int64, error) {
  483. var count int64
  484. return count, db.WithContext(ctx).Model(&OrgUser{}).Where("uid = ?", userID).Count(&count).Error
  485. }
  486. func (db *organizations) Count(ctx context.Context) int64 {
  487. var count int64
  488. db.WithContext(ctx).Model(&User{}).Where("type = ?", UserTypeOrganization).Count(&count)
  489. return count
  490. }
  491. type ErrOrganizationOwnRepos struct {
  492. args errutil.Args
  493. }
  494. // IsErrOrganizationOwnRepos returns true if the underlying error has the type
  495. // ErrOrganizationOwnRepos.
  496. func IsErrOrganizationOwnRepos(err error) bool {
  497. return errors.As(errors.Cause(err), &ErrOrganizationOwnRepos{})
  498. }
  499. func (err ErrOrganizationOwnRepos) Error() string {
  500. return fmt.Sprintf("organization still has repository ownership: %v", err.args)
  501. }
  502. func (db *organizations) DeleteByID(ctx context.Context, orgID int64) error {
  503. return db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
  504. for _, t := range []any{&Team{}, &OrgUser{}, &TeamUser{}} {
  505. err := tx.Where("org_id = ?", orgID).Delete(t).Error
  506. if err != nil {
  507. return errors.Wrapf(err, "clean up table %T", t)
  508. }
  509. }
  510. err := NewUsersStore(tx).DeleteByID(ctx, orgID, false)
  511. if err != nil {
  512. return errors.Wrap(err, "delete organization")
  513. }
  514. return nil
  515. })
  516. }
  517. var _ errutil.NotFound = (*ErrTeamNotExist)(nil)
  518. type ErrTeamNotExist struct {
  519. args map[string]any
  520. }
  521. func IsErrTeamNotExist(err error) bool {
  522. return errors.As(err, &ErrTeamNotExist{})
  523. }
  524. func (err ErrTeamNotExist) Error() string {
  525. return fmt.Sprintf("team does not exist: %v", err.args)
  526. }
  527. func (ErrTeamNotExist) NotFound() bool {
  528. return true
  529. }
  530. func (db *organizations) GetTeamByName(ctx context.Context, orgID int64, name string) (*Team, error) {
  531. var team Team
  532. err := db.WithContext(ctx).Where("org_id = ? AND lower_name = ?", orgID, strings.ToLower(name)).First(&team).Error
  533. if err != nil {
  534. if errors.Is(err, gorm.ErrRecordNotFound) {
  535. return nil, ErrTeamNotExist{args: map[string]any{"orgID": orgID, "name": name}}
  536. }
  537. return nil, errors.Wrap(err, "get team by name")
  538. }
  539. return &team, nil
  540. }
  541. type Organization = User
  542. func (u *Organization) TableName() string {
  543. return "user"
  544. }
  545. // IsOwnedBy returns true if the given user is an owner of the organization.
  546. //
  547. // TODO(unknwon): This is also used in templates, which should be fixed by
  548. // having a dedicated type `template.Organization`.
  549. func (u *Organization) IsOwnedBy(userID int64) bool {
  550. return Organizations.IsOwnedBy(context.TODO(), u.ID, userID)
  551. }
  552. // OrgUser represents relations of organizations and their members.
  553. type OrgUser struct {
  554. ID int64 `gorm:"primaryKey"`
  555. UserID int64 `xorm:"uid INDEX UNIQUE(s)" gorm:"column:uid;uniqueIndex:org_user_user_org_unique;index;not null" json:"Uid"`
  556. OrgID int64 `xorm:"INDEX UNIQUE(s)" gorm:"uniqueIndex:org_user_user_org_unique;index;not null"`
  557. IsPublic bool `gorm:"not null;default:FALSE"`
  558. IsOwner bool `gorm:"not null;default:FALSE"`
  559. NumTeams int `gorm:"not null;default:0"`
  560. }