2016-02-17 14:19:52 +02:00
|
|
|
package providers
|
|
|
|
|
|
|
|
import (
|
2020-05-05 17:53:33 +02:00
|
|
|
"context"
|
2020-12-05 20:57:33 +02:00
|
|
|
"errors"
|
2016-02-17 14:19:52 +02:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"net/url"
|
|
|
|
|
2020-09-29 18:44:42 +02:00
|
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/apis/sessions"
|
2020-12-05 20:57:33 +02:00
|
|
|
. "github.com/onsi/ginkgo"
|
|
|
|
. "github.com/onsi/ginkgo/extensions/table"
|
|
|
|
. "github.com/onsi/gomega"
|
2016-02-17 14:19:52 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func testGitLabProvider(hostname string) *GitLabProvider {
|
|
|
|
p := NewGitLabProvider(
|
|
|
|
&ProviderData{
|
|
|
|
ProviderName: "",
|
|
|
|
LoginURL: &url.URL{},
|
|
|
|
RedeemURL: &url.URL{},
|
|
|
|
ProfileURL: &url.URL{},
|
|
|
|
ValidateURL: &url.URL{},
|
|
|
|
Scope: ""})
|
|
|
|
if hostname != "" {
|
|
|
|
updateURL(p.Data().LoginURL, hostname)
|
|
|
|
updateURL(p.Data().RedeemURL, hostname)
|
|
|
|
updateURL(p.Data().ProfileURL, hostname)
|
|
|
|
updateURL(p.Data().ValidateURL, hostname)
|
|
|
|
}
|
2019-08-06 13:20:54 +02:00
|
|
|
|
2016-02-17 14:19:52 +02:00
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
2019-08-06 13:20:54 +02:00
|
|
|
func testGitLabBackend() *httptest.Server {
|
|
|
|
userInfo := `
|
|
|
|
{
|
|
|
|
"nickname": "FooBar",
|
|
|
|
"email": "foo@bar.com",
|
|
|
|
"email_verified": false,
|
|
|
|
"groups": ["foo", "bar"]
|
|
|
|
}
|
|
|
|
`
|
2021-03-25 17:48:20 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
projectInfo := `
|
|
|
|
{
|
|
|
|
"name": "MyProject",
|
|
|
|
"archived": false,
|
|
|
|
"path_with_namespace": "my_group/my_project",
|
|
|
|
"permissions": {
|
|
|
|
"project_access": null,
|
|
|
|
"group_access": {
|
|
|
|
"access_level": 30,
|
|
|
|
"notification_level": 3
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`
|
|
|
|
|
2021-03-25 17:48:20 +02:00
|
|
|
noAccessProjectInfo := `
|
|
|
|
{
|
|
|
|
"name": "NoAccessProject",
|
|
|
|
"archived": false,
|
|
|
|
"path_with_namespace": "no_access_group/no_access_project",
|
|
|
|
"permissions": {
|
|
|
|
"project_access": null,
|
|
|
|
"group_access": null,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`
|
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
personalProjectInfo := `
|
|
|
|
{
|
|
|
|
"name": "MyPersonalProject",
|
|
|
|
"archived": false,
|
|
|
|
"path_with_namespace": "my_profile/my_personal_project",
|
|
|
|
"permissions": {
|
|
|
|
"project_access": {
|
|
|
|
"access_level": 30,
|
|
|
|
"notification_level": 3
|
|
|
|
},
|
|
|
|
"group_access": null
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`
|
|
|
|
|
|
|
|
archivedProjectInfo := `
|
|
|
|
{
|
|
|
|
"name": "MyArchivedProject",
|
|
|
|
"archived": true,
|
|
|
|
"path_with_namespace": "my_group/my_archived_project",
|
|
|
|
"permissions": {
|
|
|
|
"project_access": {
|
|
|
|
"access_level": 30,
|
|
|
|
"notification_level": 3
|
|
|
|
},
|
|
|
|
"group_access": null
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`
|
|
|
|
|
2019-08-06 13:20:54 +02:00
|
|
|
authHeader := "Bearer gitlab_access_token"
|
2016-02-17 14:19:52 +02:00
|
|
|
|
|
|
|
return httptest.NewServer(http.HandlerFunc(
|
|
|
|
func(w http.ResponseWriter, r *http.Request) {
|
2020-12-05 20:57:33 +02:00
|
|
|
switch r.URL.Path {
|
|
|
|
case "/oauth/userinfo":
|
2019-08-06 13:20:54 +02:00
|
|
|
if r.Header["Authorization"][0] == authHeader {
|
|
|
|
w.WriteHeader(200)
|
|
|
|
w.Write([]byte(userInfo))
|
|
|
|
} else {
|
|
|
|
w.WriteHeader(401)
|
|
|
|
}
|
2020-12-05 20:57:33 +02:00
|
|
|
case "/api/v4/projects/my_group/my_project":
|
|
|
|
if r.Header["Authorization"][0] == authHeader {
|
|
|
|
w.WriteHeader(200)
|
|
|
|
w.Write([]byte(projectInfo))
|
|
|
|
} else {
|
|
|
|
w.WriteHeader(401)
|
|
|
|
}
|
2021-03-25 17:48:20 +02:00
|
|
|
case "/api/v4/projects/no_access_group/no_access_project":
|
|
|
|
if r.Header["Authorization"][0] == authHeader {
|
|
|
|
w.WriteHeader(200)
|
|
|
|
w.Write([]byte(noAccessProjectInfo))
|
|
|
|
} else {
|
|
|
|
w.WriteHeader(401)
|
|
|
|
}
|
2020-12-05 20:57:33 +02:00
|
|
|
case "/api/v4/projects/my_group/my_archived_project":
|
|
|
|
if r.Header["Authorization"][0] == authHeader {
|
|
|
|
w.WriteHeader(200)
|
|
|
|
w.Write([]byte(archivedProjectInfo))
|
|
|
|
} else {
|
|
|
|
w.WriteHeader(401)
|
|
|
|
}
|
|
|
|
case "/api/v4/projects/my_profile/my_personal_project":
|
|
|
|
if r.Header["Authorization"][0] == authHeader {
|
|
|
|
w.WriteHeader(200)
|
|
|
|
w.Write([]byte(personalProjectInfo))
|
|
|
|
} else {
|
|
|
|
w.WriteHeader(401)
|
|
|
|
}
|
|
|
|
case "/api/v4/projects/my_group/my_bad_project":
|
|
|
|
w.WriteHeader(403)
|
|
|
|
default:
|
2019-08-06 13:20:54 +02:00
|
|
|
w.WriteHeader(404)
|
2016-02-17 14:19:52 +02:00
|
|
|
}
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
var _ = Describe("Gitlab Provider Tests", func() {
|
|
|
|
var p *GitLabProvider
|
|
|
|
var b *httptest.Server
|
2019-08-06 13:20:54 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
BeforeEach(func() {
|
|
|
|
b = testGitLabBackend()
|
2019-08-06 13:20:54 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
bURL, err := url.Parse(b.URL)
|
|
|
|
Expect(err).To(BeNil())
|
2016-02-17 14:19:52 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
p = testGitLabProvider(bURL.Host)
|
|
|
|
})
|
2019-08-06 13:20:54 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
AfterEach(func() {
|
|
|
|
b.Close()
|
|
|
|
})
|
2019-08-06 13:20:54 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
Context("with bad token", func() {
|
|
|
|
It("should trigger an error", func() {
|
|
|
|
p.AllowUnverifiedEmail = false
|
|
|
|
session := &sessions.SessionState{AccessToken: "unexpected_gitlab_access_token"}
|
|
|
|
err := p.EnrichSession(context.Background(), session)
|
|
|
|
Expect(err).To(MatchError(errors.New("failed to retrieve user info: error getting user info: unexpected status \"401\": ")))
|
|
|
|
})
|
|
|
|
})
|
2016-02-17 14:19:52 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
Context("when filtering on email", func() {
|
|
|
|
type emailsTableInput struct {
|
|
|
|
expectedError error
|
|
|
|
expectedValue string
|
|
|
|
allowUnverifiedEmail bool
|
|
|
|
}
|
2016-02-17 14:19:52 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
DescribeTable("should return expected results",
|
|
|
|
func(in emailsTableInput) {
|
|
|
|
p.AllowUnverifiedEmail = in.allowUnverifiedEmail
|
|
|
|
session := &sessions.SessionState{AccessToken: "gitlab_access_token"}
|
2016-02-17 14:19:52 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
err := p.EnrichSession(context.Background(), session)
|
2016-02-17 14:19:52 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
if in.expectedError != nil {
|
2021-06-13 21:41:57 +02:00
|
|
|
Expect(err).To(MatchError(in.expectedError))
|
2020-12-05 20:57:33 +02:00
|
|
|
} else {
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
Expect(session.Email).To(Equal(in.expectedValue))
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Entry("unverified email denied", emailsTableInput{
|
|
|
|
expectedError: errors.New("user email is not verified"),
|
|
|
|
allowUnverifiedEmail: false,
|
|
|
|
}),
|
|
|
|
Entry("unverified email allowed", emailsTableInput{
|
|
|
|
expectedError: nil,
|
|
|
|
expectedValue: "foo@bar.com",
|
|
|
|
allowUnverifiedEmail: true,
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
})
|
2016-02-17 14:19:52 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
Context("when filtering on gitlab entities (groups and projects)", func() {
|
|
|
|
type entitiesTableInput struct {
|
2021-06-13 21:41:57 +02:00
|
|
|
allowedProjects []string
|
|
|
|
allowedGroups []string
|
|
|
|
scope string
|
|
|
|
expectedAuthz bool
|
|
|
|
expectedError error
|
|
|
|
expectedGroups []string
|
|
|
|
expectedScope string
|
2020-12-05 20:57:33 +02:00
|
|
|
}
|
2016-02-17 14:19:52 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
DescribeTable("should return expected results",
|
|
|
|
func(in entitiesTableInput) {
|
|
|
|
p.AllowUnverifiedEmail = true
|
2021-06-13 21:41:57 +02:00
|
|
|
if in.scope != "" {
|
|
|
|
p.Scope = in.scope
|
|
|
|
}
|
2020-12-05 20:57:33 +02:00
|
|
|
session := &sessions.SessionState{AccessToken: "gitlab_access_token"}
|
2019-08-06 13:20:54 +02:00
|
|
|
|
2021-06-13 21:41:57 +02:00
|
|
|
p.SetAllowedGroups(in.allowedGroups)
|
2019-08-06 13:20:54 +02:00
|
|
|
|
2021-06-13 21:41:57 +02:00
|
|
|
err := p.SetAllowedProjects(in.allowedProjects)
|
2021-06-13 23:29:56 +02:00
|
|
|
if in.expectedError == nil {
|
2021-06-13 21:41:57 +02:00
|
|
|
Expect(err).To(BeNil())
|
|
|
|
} else {
|
|
|
|
Expect(err).To(MatchError(in.expectedError))
|
|
|
|
return
|
2020-12-05 20:57:33 +02:00
|
|
|
}
|
2021-06-13 21:41:57 +02:00
|
|
|
Expect(p.Scope).To(Equal(in.expectedScope))
|
2019-08-06 13:20:54 +02:00
|
|
|
|
2020-12-05 20:57:33 +02:00
|
|
|
err = p.EnrichSession(context.Background(), session)
|
2021-06-13 21:41:57 +02:00
|
|
|
Expect(err).To(BeNil())
|
|
|
|
Expect(session.Groups).To(Equal(in.expectedGroups))
|
2019-08-06 13:20:54 +02:00
|
|
|
|
2021-06-13 21:41:57 +02:00
|
|
|
authorized, err := p.Authorize(context.Background(), session)
|
2020-12-05 20:57:33 +02:00
|
|
|
Expect(err).To(BeNil())
|
2021-06-13 21:41:57 +02:00
|
|
|
Expect(authorized).To(Equal(in.expectedAuthz))
|
2020-12-05 20:57:33 +02:00
|
|
|
},
|
|
|
|
Entry("project membership valid on group project", entitiesTableInput{
|
2021-06-13 21:41:57 +02:00
|
|
|
allowedProjects: []string{"my_group/my_project"},
|
|
|
|
expectedAuthz: true,
|
|
|
|
expectedGroups: []string{"foo", "bar", "project:my_group/my_project"},
|
|
|
|
expectedScope: "openid email read_api",
|
2020-12-05 20:57:33 +02:00
|
|
|
}),
|
|
|
|
Entry("project membership invalid on group project, insufficient access level level", entitiesTableInput{
|
2021-06-13 21:41:57 +02:00
|
|
|
allowedProjects: []string{"my_group/my_project=40"},
|
|
|
|
expectedAuthz: false,
|
|
|
|
expectedGroups: []string{"foo", "bar"},
|
|
|
|
expectedScope: "openid email read_api",
|
2020-12-05 20:57:33 +02:00
|
|
|
}),
|
2021-03-25 17:48:20 +02:00
|
|
|
Entry("project membership invalid on group project, no access at all", entitiesTableInput{
|
2021-06-13 21:41:57 +02:00
|
|
|
allowedProjects: []string{"no_access_group/no_access_project=30"},
|
|
|
|
expectedAuthz: false,
|
|
|
|
expectedGroups: []string{"foo", "bar"},
|
|
|
|
expectedScope: "openid email read_api",
|
2021-03-25 17:48:20 +02:00
|
|
|
}),
|
2020-12-05 20:57:33 +02:00
|
|
|
Entry("project membership valid on personnal project", entitiesTableInput{
|
2021-06-13 21:41:57 +02:00
|
|
|
allowedProjects: []string{"my_profile/my_personal_project"},
|
|
|
|
scope: "openid email read_api profile",
|
|
|
|
expectedAuthz: true,
|
|
|
|
expectedGroups: []string{"foo", "bar", "project:my_profile/my_personal_project"},
|
|
|
|
expectedScope: "openid email read_api profile",
|
2020-12-05 20:57:33 +02:00
|
|
|
}),
|
|
|
|
Entry("project membership invalid on personnal project, insufficient access level", entitiesTableInput{
|
2021-06-13 21:41:57 +02:00
|
|
|
allowedProjects: []string{"my_profile/my_personal_project=40"},
|
|
|
|
expectedAuthz: false,
|
|
|
|
expectedGroups: []string{"foo", "bar"},
|
|
|
|
expectedScope: "openid email read_api",
|
2020-12-05 20:57:33 +02:00
|
|
|
}),
|
|
|
|
Entry("project membership invalid", entitiesTableInput{
|
2021-06-13 21:41:57 +02:00
|
|
|
allowedProjects: []string{"my_group/my_bad_project"},
|
|
|
|
expectedAuthz: false,
|
|
|
|
expectedGroups: []string{"foo", "bar"},
|
|
|
|
expectedScope: "openid email read_api",
|
2020-12-05 20:57:33 +02:00
|
|
|
}),
|
|
|
|
Entry("group membership valid", entitiesTableInput{
|
2021-06-13 21:41:57 +02:00
|
|
|
allowedGroups: []string{"foo"},
|
|
|
|
expectedGroups: []string{"foo", "bar"},
|
|
|
|
expectedAuthz: true,
|
|
|
|
expectedScope: "openid email",
|
2020-12-05 20:57:33 +02:00
|
|
|
}),
|
|
|
|
Entry("groups and projects", entitiesTableInput{
|
2021-06-13 21:41:57 +02:00
|
|
|
allowedGroups: []string{"foo", "baz"},
|
|
|
|
allowedProjects: []string{"my_group/my_project", "my_profile/my_personal_project"},
|
|
|
|
expectedAuthz: true,
|
|
|
|
expectedGroups: []string{"foo", "bar", "project:my_group/my_project", "project:my_profile/my_personal_project"},
|
|
|
|
expectedScope: "openid email read_api",
|
2020-12-05 20:57:33 +02:00
|
|
|
}),
|
|
|
|
Entry("archived projects", entitiesTableInput{
|
2021-06-13 21:41:57 +02:00
|
|
|
allowedProjects: []string{"my_group/my_archived_project"},
|
|
|
|
expectedAuthz: false,
|
|
|
|
expectedGroups: []string{"foo", "bar"},
|
|
|
|
expectedScope: "openid email read_api",
|
2020-12-05 20:57:33 +02:00
|
|
|
}),
|
2021-06-13 21:41:57 +02:00
|
|
|
Entry("invalid project format", entitiesTableInput{
|
|
|
|
allowedProjects: []string{"my_group/my_invalid_project=123"},
|
|
|
|
expectedError: errors.New("invalid gitlab project access level specified (my_group/my_invalid_project)"),
|
|
|
|
expectedScope: "openid email read_api",
|
2020-12-05 20:57:33 +02:00
|
|
|
}),
|
|
|
|
)
|
|
|
|
})
|
|
|
|
})
|