5e68dad636
* chore: bump go to version 1.21 update all depedencies as well * fix linting issues based on golang 1.20 deprecations * cleanup go depedencies * add custom gomega matcher for option intefaces * revert and upgrade golangci-lint to 1.55.2 * fix lint issues for v1.55.2 of golangci-lint * fix toml loading test * remove second runspecs call * update go.sum * revert testutil package
301 lines
8.4 KiB
Go
301 lines
8.4 KiB
Go
package providers
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net/url"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/apis/options"
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/apis/sessions"
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/logger"
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/requests"
|
|
)
|
|
|
|
const (
|
|
gitlabProviderName = "GitLab"
|
|
gitlabDefaultScope = "openid email"
|
|
gitlabProjectPrefix = "project:"
|
|
)
|
|
|
|
// GitLabProvider represents a GitLab based Identity Provider
|
|
type GitLabProvider struct {
|
|
*OIDCProvider
|
|
|
|
allowedProjects []*gitlabProject
|
|
// Expose this for unit testing
|
|
oidcRefreshFunc func(context.Context, *sessions.SessionState) (bool, error)
|
|
}
|
|
|
|
var _ Provider = (*GitLabProvider)(nil)
|
|
|
|
// NewGitLabProvider initiates a new GitLabProvider
|
|
func NewGitLabProvider(p *ProviderData, opts options.Provider) (*GitLabProvider, error) {
|
|
p.setProviderDefaults(providerDefaults{
|
|
name: gitlabProviderName,
|
|
})
|
|
|
|
if p.Scope == "" {
|
|
p.Scope = gitlabDefaultScope
|
|
}
|
|
|
|
oidcProvider := NewOIDCProvider(p, opts.OIDCConfig)
|
|
|
|
provider := &GitLabProvider{
|
|
OIDCProvider: oidcProvider,
|
|
oidcRefreshFunc: oidcProvider.RefreshSession,
|
|
}
|
|
provider.setAllowedGroups(opts.GitLabConfig.Group)
|
|
|
|
if err := provider.setAllowedProjects(opts.GitLabConfig.Projects); err != nil {
|
|
return nil, fmt.Errorf("could not configure allowed projects: %v", err)
|
|
}
|
|
|
|
return provider, nil
|
|
}
|
|
|
|
// setAllowedProjects adds Gitlab projects to the AllowedGroups list
|
|
// and tracks them to do a project API lookup during `EnrichSession`.
|
|
func (p *GitLabProvider) setAllowedProjects(projects []string) error {
|
|
for _, project := range projects {
|
|
gp, err := newGitlabProject(project)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
p.allowedProjects = append(p.allowedProjects, gp)
|
|
p.AllowedGroups[formatProject(gp)] = struct{}{}
|
|
}
|
|
if len(p.allowedProjects) > 0 {
|
|
p.setProjectScope()
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// gitlabProject represents a Gitlab project constraint entity
|
|
type gitlabProject struct {
|
|
Name string
|
|
AccessLevel int
|
|
}
|
|
|
|
// newGitlabProject Creates a new GitlabProject struct from project string
|
|
// formatted as `namespace/project=accesslevel`
|
|
// if no accesslevel provided, use the default one
|
|
func newGitlabProject(project string) (*gitlabProject, error) {
|
|
const defaultAccessLevel = 20
|
|
// see https://docs.gitlab.com/ee/api/members.html#valid-access-levels
|
|
validAccessLevel := [4]int{10, 20, 30, 40}
|
|
|
|
parts := strings.SplitN(project, "=", 2)
|
|
if len(parts) == 2 {
|
|
lvl, err := strconv.Atoi(parts[1])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, valid := range validAccessLevel {
|
|
if lvl == valid {
|
|
return &gitlabProject{
|
|
Name: parts[0],
|
|
AccessLevel: lvl,
|
|
}, nil
|
|
}
|
|
}
|
|
return nil, fmt.Errorf("invalid gitlab project access level specified (%s)", parts[0])
|
|
}
|
|
|
|
return &gitlabProject{
|
|
Name: project,
|
|
AccessLevel: defaultAccessLevel,
|
|
}, nil
|
|
}
|
|
|
|
// setProjectScope ensures read_api is added to scope when filtering on projects
|
|
func (p *GitLabProvider) setProjectScope() {
|
|
for _, val := range strings.Split(p.Scope, " ") {
|
|
if val == "read_api" {
|
|
return
|
|
}
|
|
}
|
|
p.Scope += " read_api"
|
|
}
|
|
|
|
// EnrichSession enriches the session with the response from the userinfo API
|
|
// endpoint & projects API endpoint for allowed projects.
|
|
func (p *GitLabProvider) EnrichSession(ctx context.Context, s *sessions.SessionState) error {
|
|
// Retrieve user info
|
|
userinfo, err := p.getUserinfo(ctx, s)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to retrieve user info: %v", err)
|
|
}
|
|
|
|
// Check if email is verified
|
|
if !p.AllowUnverifiedEmail && !userinfo.EmailVerified {
|
|
return fmt.Errorf("user email is not verified")
|
|
}
|
|
|
|
if userinfo.Nickname != "" {
|
|
s.User = userinfo.Nickname
|
|
}
|
|
if userinfo.Email != "" {
|
|
s.Email = userinfo.Email
|
|
}
|
|
if len(userinfo.Groups) > 0 {
|
|
s.Groups = userinfo.Groups
|
|
}
|
|
|
|
// Add projects as `project:blah` to s.Groups
|
|
p.addProjectsToSession(ctx, s)
|
|
|
|
return nil
|
|
}
|
|
|
|
type gitlabUserinfo struct {
|
|
Nickname string `json:"nickname"`
|
|
Email string `json:"email"`
|
|
EmailVerified bool `json:"email_verified"`
|
|
Groups []string `json:"groups"`
|
|
}
|
|
|
|
func (p *GitLabProvider) getUserinfo(ctx context.Context, s *sessions.SessionState) (*gitlabUserinfo, error) {
|
|
// Retrieve user info JSON
|
|
// https://docs.gitlab.com/ee/integration/openid_connect_provider.html#shared-information
|
|
|
|
// Build user info url from login url of GitLab instance
|
|
userinfoURL := *p.LoginURL
|
|
userinfoURL.Path = "/oauth/userinfo"
|
|
|
|
var userinfo gitlabUserinfo
|
|
err := requests.New(userinfoURL.String()).
|
|
WithContext(ctx).
|
|
SetHeader("Authorization", tokenTypeBearer+" "+s.AccessToken).
|
|
Do().
|
|
UnmarshalInto(&userinfo)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error getting user info: %v", err)
|
|
}
|
|
|
|
return &userinfo, nil
|
|
}
|
|
|
|
// addProjectsToSession adds projects matching user access requirements into
|
|
// the session state groups list.
|
|
// This method prefixes projects names with `project:` to specify group kind.
|
|
func (p *GitLabProvider) addProjectsToSession(ctx context.Context, s *sessions.SessionState) {
|
|
// Iterate over projects, check if oauth2-proxy can get project information on behalf of the user
|
|
for _, project := range p.allowedProjects {
|
|
projectInfo, err := p.getProjectInfo(ctx, s, project.Name)
|
|
if err != nil {
|
|
logger.Errorf("Warning: project info request failed: %v", err)
|
|
continue
|
|
}
|
|
|
|
if projectInfo.Archived {
|
|
logger.Errorf("Warning: project %s is archived", project.Name)
|
|
continue
|
|
}
|
|
|
|
perms := projectInfo.Permissions.ProjectAccess
|
|
if perms == nil {
|
|
// use group project access as fallback
|
|
perms = projectInfo.Permissions.GroupAccess
|
|
// group project access is not set for this user then we give up
|
|
if perms == nil {
|
|
logger.Errorf("Warning: user %q has no project level access to %s",
|
|
s.Email, project.Name)
|
|
continue
|
|
}
|
|
}
|
|
|
|
if perms.AccessLevel < project.AccessLevel {
|
|
logger.Errorf(
|
|
"Warning: user %q does not have the minimum required access level for project %q",
|
|
s.Email,
|
|
project.Name,
|
|
)
|
|
continue
|
|
}
|
|
|
|
s.Groups = append(s.Groups, formatProject(project))
|
|
}
|
|
}
|
|
|
|
type gitlabPermissionAccess struct {
|
|
AccessLevel int `json:"access_level"`
|
|
}
|
|
|
|
type gitlabProjectPermission struct {
|
|
ProjectAccess *gitlabPermissionAccess `json:"project_access"`
|
|
GroupAccess *gitlabPermissionAccess `json:"group_access"`
|
|
}
|
|
|
|
type gitlabProjectInfo struct {
|
|
Name string `json:"name"`
|
|
Archived bool `json:"archived"`
|
|
PathWithNamespace string `json:"path_with_namespace"`
|
|
Permissions gitlabProjectPermission `json:"permissions"`
|
|
}
|
|
|
|
func (p *GitLabProvider) getProjectInfo(ctx context.Context, s *sessions.SessionState, project string) (*gitlabProjectInfo, error) {
|
|
var projectInfo gitlabProjectInfo
|
|
|
|
endpointURL := &url.URL{
|
|
Scheme: p.LoginURL.Scheme,
|
|
Host: p.LoginURL.Host,
|
|
Path: "/api/v4/projects/",
|
|
}
|
|
|
|
err := requests.New(fmt.Sprintf("%s%s", endpointURL.String(), url.QueryEscape(project))).
|
|
WithContext(ctx).
|
|
SetHeader("Authorization", tokenTypeBearer+" "+s.AccessToken).
|
|
Do().
|
|
UnmarshalInto(&projectInfo)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get project info: %v", err)
|
|
}
|
|
|
|
return &projectInfo, nil
|
|
}
|
|
|
|
func formatProject(project *gitlabProject) string {
|
|
return gitlabProjectPrefix + project.Name
|
|
}
|
|
|
|
// RefreshSession refreshes the session with the OIDCProvider implementation
|
|
// but preserves the custom GitLab projects added in the `EnrichSession` stage.
|
|
func (p *GitLabProvider) RefreshSession(ctx context.Context, s *sessions.SessionState) (bool, error) {
|
|
nickname := s.User
|
|
projects := getSessionProjects(s)
|
|
// This will overwrite s.Groups with the new IDToken's `groups` claims
|
|
// and s.User with the `sub` claim.
|
|
refreshed, err := p.oidcRefreshFunc(ctx, s)
|
|
if refreshed && err == nil {
|
|
s.User = nickname
|
|
s.Groups = append(s.Groups, projects...)
|
|
s.Groups = deduplicateGroups(s.Groups)
|
|
}
|
|
return refreshed, err
|
|
}
|
|
|
|
func getSessionProjects(s *sessions.SessionState) []string {
|
|
var projects []string
|
|
for _, group := range s.Groups {
|
|
if strings.HasPrefix(group, gitlabProjectPrefix) {
|
|
projects = append(projects, group)
|
|
}
|
|
}
|
|
return projects
|
|
}
|
|
|
|
func deduplicateGroups(groups []string) []string {
|
|
groupSet := make(map[string]struct{})
|
|
for _, group := range groups {
|
|
groupSet[group] = struct{}{}
|
|
}
|
|
|
|
uniqueGroups := make([]string, 0, len(groupSet))
|
|
for group := range groupSet {
|
|
uniqueGroups = append(uniqueGroups, group)
|
|
}
|
|
return uniqueGroups
|
|
}
|