You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

package.go 4.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. // Copyright 2021 The Gitea 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 context
  5. import (
  6. gocontext "context"
  7. "fmt"
  8. "net/http"
  9. "code.gitea.io/gitea/models/organization"
  10. packages_model "code.gitea.io/gitea/models/packages"
  11. "code.gitea.io/gitea/models/perm"
  12. "code.gitea.io/gitea/models/unit"
  13. user_model "code.gitea.io/gitea/models/user"
  14. "code.gitea.io/gitea/modules/setting"
  15. "code.gitea.io/gitea/modules/structs"
  16. "code.gitea.io/gitea/modules/templates"
  17. )
  18. // Package contains owner, access mode and optional the package descriptor
  19. type Package struct {
  20. Owner *user_model.User
  21. AccessMode perm.AccessMode
  22. Descriptor *packages_model.PackageDescriptor
  23. }
  24. // PackageAssignment returns a middleware to handle Context.Package assignment
  25. func PackageAssignment() func(ctx *Context) {
  26. return func(ctx *Context) {
  27. packageAssignment(ctx, func(status int, title string, obj interface{}) {
  28. err, ok := obj.(error)
  29. if !ok {
  30. err = fmt.Errorf("%s", obj)
  31. }
  32. if status == http.StatusNotFound {
  33. ctx.NotFound(title, err)
  34. } else {
  35. ctx.ServerError(title, err)
  36. }
  37. })
  38. }
  39. }
  40. // PackageAssignmentAPI returns a middleware to handle Context.Package assignment
  41. func PackageAssignmentAPI() func(ctx *APIContext) {
  42. return func(ctx *APIContext) {
  43. packageAssignment(ctx.Context, ctx.Error)
  44. }
  45. }
  46. func packageAssignment(ctx *Context, errCb func(int, string, interface{})) {
  47. ctx.Package = &Package{
  48. Owner: ctx.ContextUser,
  49. }
  50. var err error
  51. ctx.Package.AccessMode, err = determineAccessMode(ctx)
  52. if err != nil {
  53. errCb(http.StatusInternalServerError, "determineAccessMode", err)
  54. return
  55. }
  56. packageType := ctx.Params("type")
  57. name := ctx.Params("name")
  58. version := ctx.Params("version")
  59. if packageType != "" && name != "" && version != "" {
  60. pv, err := packages_model.GetVersionByNameAndVersion(ctx, ctx.Package.Owner.ID, packages_model.Type(packageType), name, version)
  61. if err != nil {
  62. if err == packages_model.ErrPackageNotExist {
  63. errCb(http.StatusNotFound, "GetVersionByNameAndVersion", err)
  64. } else {
  65. errCb(http.StatusInternalServerError, "GetVersionByNameAndVersion", err)
  66. }
  67. return
  68. }
  69. ctx.Package.Descriptor, err = packages_model.GetPackageDescriptor(ctx, pv)
  70. if err != nil {
  71. errCb(http.StatusInternalServerError, "GetPackageDescriptor", err)
  72. return
  73. }
  74. }
  75. }
  76. func determineAccessMode(ctx *Context) (perm.AccessMode, error) {
  77. accessMode := perm.AccessModeNone
  78. if setting.Service.RequireSignInView && ctx.Doer == nil {
  79. return accessMode, nil
  80. }
  81. if ctx.Package.Owner.IsOrganization() {
  82. org := organization.OrgFromUser(ctx.Package.Owner)
  83. // 1. Get user max authorize level for the org (may be none, if user is not member of the org)
  84. if ctx.Doer != nil {
  85. var err error
  86. accessMode, err = org.GetOrgUserMaxAuthorizeLevel(ctx.Doer.ID)
  87. if err != nil {
  88. return accessMode, err
  89. }
  90. // If access mode is less than write check every team for more permissions
  91. if accessMode < perm.AccessModeWrite {
  92. teams, err := organization.GetUserOrgTeams(ctx, org.ID, ctx.Doer.ID)
  93. if err != nil {
  94. return accessMode, err
  95. }
  96. for _, t := range teams {
  97. perm := t.UnitAccessModeCtx(ctx, unit.TypePackages)
  98. if accessMode < perm {
  99. accessMode = perm
  100. }
  101. }
  102. }
  103. }
  104. // 2. If authorize level is none, check if org is visible to user
  105. if accessMode == perm.AccessModeNone && organization.HasOrgOrUserVisible(ctx, ctx.Package.Owner, ctx.Doer) {
  106. accessMode = perm.AccessModeRead
  107. }
  108. } else {
  109. if ctx.Doer != nil && !ctx.Doer.IsGhost() {
  110. // 1. Check if user is package owner
  111. if ctx.Doer.ID == ctx.Package.Owner.ID {
  112. accessMode = perm.AccessModeOwner
  113. } else if ctx.Package.Owner.Visibility == structs.VisibleTypePublic || ctx.Package.Owner.Visibility == structs.VisibleTypeLimited { // 2. Check if package owner is public or limited
  114. accessMode = perm.AccessModeRead
  115. }
  116. } else if ctx.Package.Owner.Visibility == structs.VisibleTypePublic { // 3. Check if package owner is public
  117. accessMode = perm.AccessModeRead
  118. }
  119. }
  120. return accessMode, nil
  121. }
  122. // PackageContexter initializes a package context for a request.
  123. func PackageContexter(ctx gocontext.Context) func(next http.Handler) http.Handler {
  124. _, rnd := templates.HTMLRenderer(ctx)
  125. return func(next http.Handler) http.Handler {
  126. return http.HandlerFunc(func(resp http.ResponseWriter, req *http.Request) {
  127. ctx := Context{
  128. Resp: NewResponse(resp),
  129. Data: map[string]interface{}{},
  130. Render: rnd,
  131. }
  132. defer ctx.Close()
  133. ctx.Req = WithContext(req, &ctx)
  134. next.ServeHTTP(ctx.Resp, ctx.Req)
  135. })
  136. }
  137. }