2022-08-16 03:57:09 -05:00
|
|
|
package requestcontext
|
|
|
|
|
2022-11-18 12:35:28 -05:00
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
|
|
|
glob "github.com/bmatcuk/doublestar/v4" //nolint:gci
|
2023-01-09 15:37:44 -05:00
|
|
|
|
2022-11-18 12:35:28 -05:00
|
|
|
"zotregistry.io/zot/errors"
|
|
|
|
)
|
|
|
|
|
2022-08-16 03:57:09 -05:00
|
|
|
type Key int
|
|
|
|
|
|
|
|
// request-local context key.
|
2022-10-05 05:21:14 -05:00
|
|
|
var authzCtxKey = Key(0) //nolint: gochecknoglobals
|
2022-08-16 03:57:09 -05:00
|
|
|
|
|
|
|
// pointer needed for use in context.WithValue.
|
|
|
|
func GetContextKey() *Key {
|
|
|
|
return &authzCtxKey
|
|
|
|
}
|
|
|
|
|
2023-04-27 10:13:06 -05:00
|
|
|
// AccessControlContext - contains user authn/authz information.
|
2022-08-16 03:57:09 -05:00
|
|
|
type AccessControlContext struct {
|
2022-11-18 12:35:28 -05:00
|
|
|
// read method action
|
|
|
|
ReadGlobPatterns map[string]bool
|
|
|
|
// detectManifestCollision behaviour action
|
|
|
|
DmcGlobPatterns map[string]bool
|
|
|
|
IsAdmin bool
|
|
|
|
Username string
|
2023-03-08 14:47:15 -05:00
|
|
|
Groups []string
|
2022-11-18 12:35:28 -05:00
|
|
|
}
|
|
|
|
|
2023-04-27 10:13:06 -05:00
|
|
|
/*
|
|
|
|
GetAccessControlContext returns an AccessControlContext struct made available on all http requests
|
|
|
|
(using context.Context values) by authz and authn middlewares.
|
|
|
|
|
|
|
|
its methods and attributes can be used in http.Handlers to get user info for that specific request
|
|
|
|
(username, groups, if it's an admin, if it can access certain resources).
|
|
|
|
*/
|
2022-11-18 12:35:28 -05:00
|
|
|
func GetAccessControlContext(ctx context.Context) (*AccessControlContext, error) {
|
|
|
|
authzCtxKey := GetContextKey()
|
|
|
|
if authCtx := ctx.Value(authzCtxKey); authCtx != nil {
|
|
|
|
acCtx, ok := authCtx.(AccessControlContext)
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.ErrBadType
|
|
|
|
}
|
|
|
|
|
|
|
|
return &acCtx, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil //nolint: nilnil
|
|
|
|
}
|
|
|
|
|
2023-04-27 10:13:06 -05:00
|
|
|
// returns whether or not the user/anonymous who made the request has read permission on 'repository'.
|
|
|
|
func (acCtx *AccessControlContext) CanReadRepo(repository string) bool {
|
|
|
|
if acCtx.ReadGlobPatterns != nil {
|
|
|
|
return acCtx.matchesRepo(acCtx.ReadGlobPatterns, repository)
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
returns whether or not the user/anonymous who made the request
|
|
|
|
has detectManifestCollision permission on 'repository'.
|
|
|
|
*/
|
|
|
|
func (acCtx *AccessControlContext) CanDetectManifestCollision(repository string) bool {
|
|
|
|
if acCtx.DmcGlobPatterns != nil {
|
|
|
|
return acCtx.matchesRepo(acCtx.DmcGlobPatterns, repository)
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
returns whether or not 'repository' can be found in the list of patterns
|
|
|
|
on which the user who made the request has read permission on.
|
|
|
|
*/
|
2022-11-18 12:35:28 -05:00
|
|
|
func (acCtx *AccessControlContext) matchesRepo(globPatterns map[string]bool, repository string) bool {
|
|
|
|
var longestMatchedPattern string
|
|
|
|
|
|
|
|
// because of the longest path matching rule, we need to check all patterns from config
|
|
|
|
for pattern := range globPatterns {
|
|
|
|
matched, err := glob.Match(pattern, repository)
|
|
|
|
if err == nil {
|
|
|
|
if matched && len(pattern) > len(longestMatchedPattern) {
|
|
|
|
longestMatchedPattern = pattern
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
allowed := globPatterns[longestMatchedPattern]
|
|
|
|
|
|
|
|
return allowed
|
|
|
|
}
|
2023-07-07 11:27:10 -05:00
|
|
|
|
|
|
|
// request-local context key.
|
|
|
|
var amwCtxKey = Key(1) //nolint: gochecknoglobals
|
|
|
|
|
|
|
|
// pointer needed for use in context.WithValue.
|
|
|
|
func GetAuthnMiddlewareCtxKey() *Key {
|
|
|
|
return &amwCtxKey
|
|
|
|
}
|
|
|
|
|
|
|
|
type AuthnMiddlewareContext struct {
|
|
|
|
AuthnType string
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetAuthnMiddlewareContext(ctx context.Context) (*AuthnMiddlewareContext, error) {
|
|
|
|
authnMiddlewareCtxKey := GetAuthnMiddlewareCtxKey()
|
|
|
|
if authnMiddlewareCtx := ctx.Value(authnMiddlewareCtxKey); authnMiddlewareCtx != nil {
|
|
|
|
amCtx, ok := authnMiddlewareCtx.(AuthnMiddlewareContext)
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.ErrBadType
|
|
|
|
}
|
|
|
|
|
|
|
|
return &amCtx, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil //nolint: nilnil
|
|
|
|
}
|