diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index fcb9164401cc2..057396197c606 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -5,9 +5,11 @@ on: branches: - main pull_request: + branches: + - main workflow_dispatch: -permissions: read-all +permissions: {} jobs: package: @@ -41,6 +43,9 @@ jobs: - name: Apply patches run: for i in ./patch/*.patch; do patch -p1 < "$i"; done + - name: Add overlay + run: cp -R ./patch/overlay/* ./ + - name: Build server uses: ./patch/build-server diff --git a/0002-enable-openid-provider.patch b/0002-enable-openid-provider.patch new file mode 100644 index 0000000000000..40d900c059a46 --- /dev/null +++ b/0002-enable-openid-provider.patch @@ -0,0 +1,27 @@ +Import openid oauthprovider and enable in non-enterprise version. + +--- a/server/cmd/mattermost/main.go ++++ b/server/cmd/mattermost/main.go +@@ -11,6 +11,7 @@ import ( + _ "github.com/mattermost/mattermost/server/v8/channels/app/slashcommands" + // Plugins + _ "github.com/mattermost/mattermost/server/v8/channels/app/oauthproviders/gitlab" ++ _ "github.com/mattermost/mattermost/server/v8/channels/app/oauthproviders/openid" + + // Enterprise Imports + _ "github.com/mattermost/mattermost/server/v8/enterprise" +--- a/server/config/client.go ++++ b/server/config/client.go +@@ -326,9 +326,9 @@ func GenerateLimitedClientConfig(c *model.Config, telemetryID string, license *m + props["SamlLoginButtonTextColor"] = "" + props["EnableSignUpWithGoogle"] = "false" + props["EnableSignUpWithOffice365"] = "false" +- props["EnableSignUpWithOpenId"] = "false" +- props["OpenIdButtonText"] = "" +- props["OpenIdButtonColor"] = "" ++ props["EnableSignUpWithOpenId"] = strconv.FormatBool(*c.OpenIdSettings.Enable) ++ props["OpenIdButtonColor"] = *c.OpenIdSettings.ButtonColor ++ props["OpenIdButtonText"] = *c.OpenIdSettings.ButtonText + props["CWSURL"] = "" + props["EnableCustomBrand"] = strconv.FormatBool(*c.TeamSettings.EnableCustomBrand) + props["CustomBrandText"] = *c.TeamSettings.CustomBrandText diff --git a/0002-replace-gitlab-icon-with-opencampus-icon.patch b/0002-replace-gitlab-icon-with-opencampus-icon.patch deleted file mode 100644 index 91630f683b6e0..0000000000000 --- a/0002-replace-gitlab-icon-with-opencampus-icon.patch +++ /dev/null @@ -1,13 +0,0 @@ -replace gitlab icon with opencampus icon. - ---- a/webapp/channels/src/components/common/svg_images_components/gitlab_svg.tsx -+++ b/webapp/channels/src/components/common/svg_images_components/gitlab_svg.tsx -@@ -34,7 +34,7 @@ export default () => ( - id='image0_615_81787' - width='338' - height='311' -- xlinkHref='' -+ xlinkHref='' - /> - - diff --git a/0003-replace-openid-icon-with-opencampus-icon.patch b/0003-replace-openid-icon-with-opencampus-icon.patch new file mode 100644 index 0000000000000..5b5520f10a2d7 --- /dev/null +++ b/0003-replace-openid-icon-with-opencampus-icon.patch @@ -0,0 +1,50 @@ +Replace openid logo with the opencampus icon. + +--- a/webapp/channels/src/components/widgets/icons/login_openid_icon.tsx ++++ b/webapp/channels/src/components/widgets/icons/login_openid_icon.tsx +@@ -12,23 +12,31 @@ export default function LoginOpenIdIcon(props: React.HTMLAttributes +- +- +- ++ ++ ++ ++ + + + diff --git a/README.md b/README.md index db06fad748b1c..fd4922ed9591a 100644 --- a/README.md +++ b/README.md @@ -5,11 +5,12 @@ applied. The upstream mettermost version is specified in the `VERSION` file. All patch files (files ending with .patch) in the root directory of the -repository are applied onto the specified mattermost version. When -adding new patches or changing them please make sure a description -exists and is up to date. +repository are applied onto the specified mattermost version. Files +inside of the `overlay/` directory are copied directly. When adding new +patches or changing them please make sure a description exists and is +up to date. ## License -The code is published under the AGPL version 3.0 only. See the -`LICENSE` file for details. +The code is published under the AGPL version 3.0 only unless otherwise +specified in the file. See the `LICENSE` file for details. diff --git a/overlay/server/channels/app/oauthproviders/openid/openid.go b/overlay/server/channels/app/oauthproviders/openid/openid.go new file mode 100644 index 0000000000000..4c58a5a28e995 --- /dev/null +++ b/overlay/server/channels/app/oauthproviders/openid/openid.go @@ -0,0 +1,246 @@ +// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. +// See [LICENSE.txt] for license information. +// [LICENSE.txt]: https://github.com/mattermost/mattermost/blob/03a5b4a288ba60c52e604e5ef017cf5fc179db51/LICENSE.txt + +package oauthopenid + +import ( + "encoding/base64" + "encoding/json" + "errors" + "io" + "net/http" + "strconv" + "strings" + "time" + + "github.com/mattermost/mattermost/server/public/model" + "github.com/mattermost/mattermost/server/public/shared/mlog" + "github.com/mattermost/mattermost/server/public/shared/request" + "github.com/mattermost/mattermost/server/v8/einterfaces" +) + +type CacheData struct { + Service string + Expires int64 + Settings model.SSOSettings +} + +type OpenIdMetadata struct { + Issuer string `json:"issuer"` + AuthorizationEndpoint string `json:"authorization_endpoint"` + TokenEndpoint string `json:"token_endpoint"` + UserEndpoint string `json:"userinfo_endpoint"` + JwksURI string `json:"jwks_uri"` + Algorithms []string `json:"id_token_signing_alg_values_supported"` +} + +type OpenIdProvider struct { + CacheData *CacheData +} + +type OpenIdUser struct { + Id string `json:"sub"` + Oid string `json:"oid"` //Office 365 only + FirstName string `json:"given_name"` + LastName string `json:"family_name"` + Name string `json:"name"` + Nickname string `json:"nickname"` + Email string `json:"email"` +} + +func init() { + provider := &OpenIdProvider{} + einterfaces.RegisterOAuthProvider(model.ServiceOpenid, provider) +} + +func (o *OpenIdProvider) userFromOpenIdUser(logger mlog.LoggerIFace, u *OpenIdUser) *model.User { + user := &model.User{} + + user.Email = u.Email + user.Username = model.CleanUsername(logger, strings.Split(user.Email, "@")[0]) + if o.CacheData.Service == model.ServiceGitlab && u.Nickname != "" { + user.Username = u.Nickname + } + + user.FirstName = u.FirstName + user.LastName = u.LastName + user.Nickname = u.Nickname + + user.AuthData = new(string) + *user.AuthData = o.getAuthData(u) + + return user +} + +func (o *OpenIdProvider) getAuthData(u *OpenIdUser) string { + if o.CacheData.Service == model.ServiceOffice365 { + if u.Oid != "" { + return u.Oid + } + } + return u.Id +} + +func openIDUserFromJSON(data io.Reader) (*OpenIdUser, error) { + decoder := json.NewDecoder(data) + var u OpenIdUser + err := decoder.Decode(&u) + if err != nil { + return nil, err + } + return &u, nil +} + +func (u *OpenIdUser) IsValid() error { + if u.Id == "" { + return errors.New("invalid id") + } + + if u.Email == "" { + return errors.New("invalid emails") + } + return nil +} + +func (u *OpenIdUser) GetIdentifier() string { + return model.ServiceOpenid +} + +func (o *OpenIdProvider) GetUserFromJSON(c request.CTX, data io.Reader, tokenUser *model.User) (*model.User, error) { + oid, err := openIDUserFromJSON(data) + if err != nil { + return nil, err + } + jsonUser := o.userFromOpenIdUser(c.Logger(), oid) + + if tokenUser != nil { + jsonUser = o.combineUsers(jsonUser, tokenUser) + } + return jsonUser, nil +} + +func (o *OpenIdProvider) combineUsers(jsonUser *model.User, tokenUser *model.User) *model.User { + if o.CacheData.Service == model.ServiceOffice365 { + jsonUser.AuthData = tokenUser.AuthData + } + return jsonUser +} + +func (o *OpenIdProvider) GetAuthDataFromJSON(data io.Reader) (string, error) { + u, err := openIDUserFromJSON(data) + if err != nil { + return "", err + } + + err = u.IsValid() + if err != nil { + return "", err + } + return o.getAuthData(u), nil +} + +// GetSSOSettings returns SSO Settings from Cache or Discovery Document +func (o *OpenIdProvider) GetSSOSettings(_ request.CTX, config *model.Config, service string) (*model.SSOSettings, error) { + settings := config.OpenIdSettings + if service == model.ServiceOffice365 { + settings = *config.Office365Settings.SSOSettings() + } else if service == model.ServiceGoogle { + settings = config.GoogleSettings + } else if service == model.ServiceGitlab { + settings = config.GitLabSettings + } + + if o.CacheData != nil && !settingsChanged(*o.CacheData, service, settings) && o.CacheData.Expires > time.Now().Unix() { + return &o.CacheData.Settings, nil + } + + var age int64 = 0 + if *settings.DiscoveryEndpoint != "" { + response, err := http.Get(*settings.DiscoveryEndpoint) + if err != nil { + return nil, err + } + defer response.Body.Close() + + for _, v := range strings.Split(response.Header.Get("Cache-Control"), ",") { + if strings.Contains(v, "max-age") { + ageValue := strings.Split(v, "=")[1] + age, _ = strconv.ParseInt(ageValue, 10, 64) + } + } + responseData, err := io.ReadAll(response.Body) + if err != nil { + return nil, err + } + + var openIDResponse OpenIdMetadata + err = json.Unmarshal(responseData, &openIDResponse) + if err != nil { + return nil, err + } + + settings.AuthEndpoint = &openIDResponse.AuthorizationEndpoint + settings.TokenEndpoint = &openIDResponse.TokenEndpoint + settings.UserAPIEndpoint = &openIDResponse.UserEndpoint + } + expires := time.Now().Unix() + age + + o.CacheData = &CacheData{ + Service: service, + Expires: expires, + Settings: settings, + } + return &settings, nil +} + +func settingsChanged(cacheData CacheData, service string, configSettings model.SSOSettings) bool { + if cacheData.Service == service && + cacheData.Settings.DiscoveryEndpoint == configSettings.DiscoveryEndpoint && + cacheData.Settings.Secret == configSettings.Secret && + cacheData.Settings.Id == configSettings.Id { + return false + } + return true +} + +func (o *OpenIdProvider) GetUserFromIdToken(c request.CTX, idToken string) (*model.User, error) { + parts := strings.Split(idToken, ".") + if len(parts) != 3 { + return nil, errors.New("invalid Id Token") + } + + b, err := base64.RawURLEncoding.DecodeString(parts[1]) + if err != nil { + return nil, err + } + + claims := &OpenIdUser{} + json.Unmarshal(b, &claims) + + return o.userFromOpenIdUser(c.Logger(), claims), nil +} + +func (o *OpenIdProvider) IsSameUser(_ request.CTX, dbUser, oauthUser *model.User) bool { + // Office365 OAuth would store Ids without dashes. (ie. 0e8fddd450d344999a93a390ee8cb83d) + // Office365 OpenId will return as a formatted GUID (ie. '0e8fddd4-50d3-4499-9a93-a390ee8cb83d') + // If this is a UUID that starts with all zero. (ie. 00000000-0000-0000-be95-fe607df5dbeb) + // For backwards compatibility we store the auth data from OAuth as be95fe607df5dbeb + if dbUser.AuthData == nil || oauthUser.AuthData == nil { + return false + } + dbID := *dbUser.AuthData + oauthID := *oauthUser.AuthData + if dbID == "" || oauthID == "" { + return false + } + parts := strings.Split(oauthID, "-") + for _, part := range parts { + if strings.Count(part, "0") != len(part) { + if !strings.Contains(dbID, part) { + return false + } + } + } + return true +}