2022-10-19 14:40:28 +02:00
// Copyright 2022 The Gitea Authors. All rights reserved.
2022-11-27 13:20:29 -05:00
// SPDX-License-Identifier: MIT
2022-10-19 14:40:28 +02:00
package integration
import (
"fmt"
"net/http"
2023-08-31 12:26:13 -04:00
"net/url"
"strings"
2022-10-19 14:40:28 +02:00
"testing"
"code.gitea.io/gitea/models/db"
"code.gitea.io/gitea/models/organization"
"code.gitea.io/gitea/models/unittest"
user_model "code.gitea.io/gitea/models/user"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/test"
"code.gitea.io/gitea/tests"
"github.com/stretchr/testify/assert"
)
func TestOrgTeamEmailInvite ( t * testing . T ) {
if setting . MailService == nil {
t . Skip ( )
return
}
defer tests . PrepareTestEnv ( t ) ( )
org := unittest . AssertExistsAndLoadBean ( t , & organization . Organization { ID : 3 } )
team := unittest . AssertExistsAndLoadBean ( t , & organization . Team { ID : 2 } )
user := unittest . AssertExistsAndLoadBean ( t , & user_model . User { ID : 5 } )
isMember , err := organization . IsTeamMember ( db . DefaultContext , team . OrgID , team . ID , user . ID )
assert . NoError ( t , err )
assert . False ( t , isMember )
session := loginUser ( t , "user1" )
2023-08-31 12:26:13 -04:00
teamURL := fmt . Sprintf ( "/org/%s/teams/%s" , org . Name , team . Name )
2024-10-10 11:48:21 +08:00
csrf := GetUserCSRFToken ( t , session )
2023-08-31 12:26:13 -04:00
req := NewRequestWithValues ( t , "POST" , teamURL + "/action/add" , map [ string ] string {
2022-10-19 14:40:28 +02:00
"_csrf" : csrf ,
"uid" : "1" ,
"uname" : user . Email ,
} )
resp := session . MakeRequest ( t , req , http . StatusSeeOther )
req = NewRequest ( t , "GET" , test . RedirectURL ( resp ) )
session . MakeRequest ( t , req , http . StatusOK )
// get the invite token
invites , err := organization . GetInvitesByTeamID ( db . DefaultContext , team . ID )
assert . NoError ( t , err )
assert . Len ( t , invites , 1 )
session = loginUser ( t , user . Name )
// join the team
2023-08-31 12:26:13 -04:00
inviteURL := fmt . Sprintf ( "/org/invite/%s" , invites [ 0 ] . Token )
2024-10-10 11:48:21 +08:00
csrf = GetUserCSRFToken ( t , session )
2023-08-31 12:26:13 -04:00
req = NewRequestWithValues ( t , "POST" , inviteURL , map [ string ] string {
2022-10-19 14:40:28 +02:00
"_csrf" : csrf ,
} )
resp = session . MakeRequest ( t , req , http . StatusSeeOther )
req = NewRequest ( t , "GET" , test . RedirectURL ( resp ) )
session . MakeRequest ( t , req , http . StatusOK )
isMember , err = organization . IsTeamMember ( db . DefaultContext , team . OrgID , team . ID , user . ID )
assert . NoError ( t , err )
assert . True ( t , isMember )
}
2023-08-31 12:26:13 -04:00
// Check that users are redirected to accept the invitation correctly after login
func TestOrgTeamEmailInviteRedirectsExistingUser ( t * testing . T ) {
if setting . MailService == nil {
t . Skip ( )
return
}
defer tests . PrepareTestEnv ( t ) ( )
org := unittest . AssertExistsAndLoadBean ( t , & organization . Organization { ID : 3 } )
team := unittest . AssertExistsAndLoadBean ( t , & organization . Team { ID : 2 } )
user := unittest . AssertExistsAndLoadBean ( t , & user_model . User { ID : 5 } )
isMember , err := organization . IsTeamMember ( db . DefaultContext , team . OrgID , team . ID , user . ID )
assert . NoError ( t , err )
assert . False ( t , isMember )
// create the invite
session := loginUser ( t , "user1" )
teamURL := fmt . Sprintf ( "/org/%s/teams/%s" , org . Name , team . Name )
req := NewRequestWithValues ( t , "POST" , teamURL + "/action/add" , map [ string ] string {
2024-10-10 11:48:21 +08:00
"_csrf" : GetUserCSRFToken ( t , session ) ,
2023-08-31 12:26:13 -04:00
"uid" : "1" ,
"uname" : user . Email ,
} )
resp := session . MakeRequest ( t , req , http . StatusSeeOther )
req = NewRequest ( t , "GET" , test . RedirectURL ( resp ) )
session . MakeRequest ( t , req , http . StatusOK )
// get the invite token
invites , err := organization . GetInvitesByTeamID ( db . DefaultContext , team . ID )
assert . NoError ( t , err )
assert . Len ( t , invites , 1 )
// accept the invite
inviteURL := fmt . Sprintf ( "/org/invite/%s" , invites [ 0 ] . Token )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "/user/login?redirect_to=%s" , url . QueryEscape ( inviteURL ) ) )
resp = MakeRequest ( t , req , http . StatusOK )
doc := NewHTMLParser ( t , resp . Body )
req = NewRequestWithValues ( t , "POST" , "/user/login" , map [ string ] string {
"_csrf" : doc . GetCSRF ( ) ,
"user_name" : "user5" ,
"password" : "password" ,
} )
for _ , c := range resp . Result ( ) . Cookies ( ) {
req . AddCookie ( c )
}
resp = MakeRequest ( t , req , http . StatusSeeOther )
assert . Equal ( t , inviteURL , test . RedirectURL ( resp ) )
// complete the login process
ch := http . Header { }
ch . Add ( "Cookie" , strings . Join ( resp . Header ( ) [ "Set-Cookie" ] , ";" ) )
cr := http . Request { Header : ch }
session = emptyTestSession ( t )
baseURL , err := url . Parse ( setting . AppURL )
assert . NoError ( t , err )
session . jar . SetCookies ( baseURL , cr . Cookies ( ) )
// make the request
req = NewRequestWithValues ( t , "POST" , test . RedirectURL ( resp ) , map [ string ] string {
2024-10-10 11:48:21 +08:00
"_csrf" : GetUserCSRFToken ( t , session ) ,
2023-08-31 12:26:13 -04:00
} )
resp = session . MakeRequest ( t , req , http . StatusSeeOther )
req = NewRequest ( t , "GET" , test . RedirectURL ( resp ) )
session . MakeRequest ( t , req , http . StatusOK )
isMember , err = organization . IsTeamMember ( db . DefaultContext , team . OrgID , team . ID , user . ID )
assert . NoError ( t , err )
assert . True ( t , isMember )
}
// Check that newly signed up users are redirected to accept the invitation correctly
func TestOrgTeamEmailInviteRedirectsNewUser ( t * testing . T ) {
if setting . MailService == nil {
t . Skip ( )
return
}
defer tests . PrepareTestEnv ( t ) ( )
org := unittest . AssertExistsAndLoadBean ( t , & organization . Organization { ID : 3 } )
team := unittest . AssertExistsAndLoadBean ( t , & organization . Team { ID : 2 } )
// create the invite
session := loginUser ( t , "user1" )
teamURL := fmt . Sprintf ( "/org/%s/teams/%s" , org . Name , team . Name )
req := NewRequestWithValues ( t , "POST" , teamURL + "/action/add" , map [ string ] string {
2024-10-10 11:48:21 +08:00
"_csrf" : GetUserCSRFToken ( t , session ) ,
2023-08-31 12:26:13 -04:00
"uid" : "1" ,
"uname" : "doesnotexist@example.com" ,
} )
resp := session . MakeRequest ( t , req , http . StatusSeeOther )
req = NewRequest ( t , "GET" , test . RedirectURL ( resp ) )
session . MakeRequest ( t , req , http . StatusOK )
// get the invite token
invites , err := organization . GetInvitesByTeamID ( db . DefaultContext , team . ID )
assert . NoError ( t , err )
assert . Len ( t , invites , 1 )
// accept the invite
inviteURL := fmt . Sprintf ( "/org/invite/%s" , invites [ 0 ] . Token )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "/user/sign_up?redirect_to=%s" , url . QueryEscape ( inviteURL ) ) )
resp = MakeRequest ( t , req , http . StatusOK )
doc := NewHTMLParser ( t , resp . Body )
req = NewRequestWithValues ( t , "POST" , "/user/sign_up" , map [ string ] string {
"_csrf" : doc . GetCSRF ( ) ,
"user_name" : "doesnotexist" ,
"email" : "doesnotexist@example.com" ,
"password" : "examplePassword!1" ,
"retype" : "examplePassword!1" ,
} )
for _ , c := range resp . Result ( ) . Cookies ( ) {
req . AddCookie ( c )
}
resp = MakeRequest ( t , req , http . StatusSeeOther )
assert . Equal ( t , inviteURL , test . RedirectURL ( resp ) )
// complete the signup process
ch := http . Header { }
ch . Add ( "Cookie" , strings . Join ( resp . Header ( ) [ "Set-Cookie" ] , ";" ) )
cr := http . Request { Header : ch }
session = emptyTestSession ( t )
baseURL , err := url . Parse ( setting . AppURL )
assert . NoError ( t , err )
session . jar . SetCookies ( baseURL , cr . Cookies ( ) )
// make the redirected request
req = NewRequestWithValues ( t , "POST" , test . RedirectURL ( resp ) , map [ string ] string {
2024-10-10 11:48:21 +08:00
"_csrf" : GetUserCSRFToken ( t , session ) ,
2023-08-31 12:26:13 -04:00
} )
resp = session . MakeRequest ( t , req , http . StatusSeeOther )
req = NewRequest ( t , "GET" , test . RedirectURL ( resp ) )
session . MakeRequest ( t , req , http . StatusOK )
// get the new user
newUser , err := user_model . GetUserByName ( db . DefaultContext , "doesnotexist" )
assert . NoError ( t , err )
isMember , err := organization . IsTeamMember ( db . DefaultContext , team . OrgID , team . ID , newUser . ID )
assert . NoError ( t , err )
assert . True ( t , isMember )
}
// Check that users are redirected correctly after confirming their email
func TestOrgTeamEmailInviteRedirectsNewUserWithActivation ( t * testing . T ) {
if setting . MailService == nil {
t . Skip ( )
return
}
// enable email confirmation temporarily
2024-10-10 11:48:21 +08:00
defer test . MockVariableValue ( & setting . Service . RegisterEmailConfirm , true ) ( )
2023-08-31 12:26:13 -04:00
defer tests . PrepareTestEnv ( t ) ( )
org := unittest . AssertExistsAndLoadBean ( t , & organization . Organization { ID : 3 } )
team := unittest . AssertExistsAndLoadBean ( t , & organization . Team { ID : 2 } )
2024-10-10 11:48:21 +08:00
// user1: create the invite
2023-08-31 12:26:13 -04:00
session := loginUser ( t , "user1" )
teamURL := fmt . Sprintf ( "/org/%s/teams/%s" , org . Name , team . Name )
req := NewRequestWithValues ( t , "POST" , teamURL + "/action/add" , map [ string ] string {
2024-10-10 11:48:21 +08:00
"_csrf" : GetUserCSRFToken ( t , session ) ,
2023-08-31 12:26:13 -04:00
"uid" : "1" ,
"uname" : "doesnotexist@example.com" ,
} )
resp := session . MakeRequest ( t , req , http . StatusSeeOther )
req = NewRequest ( t , "GET" , test . RedirectURL ( resp ) )
session . MakeRequest ( t , req , http . StatusOK )
// get the invite token
invites , err := organization . GetInvitesByTeamID ( db . DefaultContext , team . ID )
assert . NoError ( t , err )
assert . Len ( t , invites , 1 )
2024-10-10 11:48:21 +08:00
// new user: accept the invite
session = emptyTestSession ( t )
2023-08-31 12:26:13 -04:00
inviteURL := fmt . Sprintf ( "/org/invite/%s" , invites [ 0 ] . Token )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "/user/sign_up?redirect_to=%s" , url . QueryEscape ( inviteURL ) ) )
2024-10-10 11:48:21 +08:00
session . MakeRequest ( t , req , http . StatusOK )
2023-08-31 12:26:13 -04:00
req = NewRequestWithValues ( t , "POST" , "/user/sign_up" , map [ string ] string {
"user_name" : "doesnotexist" ,
"email" : "doesnotexist@example.com" ,
"password" : "examplePassword!1" ,
"retype" : "examplePassword!1" ,
} )
2024-10-10 11:48:21 +08:00
session . MakeRequest ( t , req , http . StatusOK )
2023-08-31 12:26:13 -04:00
user , err := user_model . GetUserByName ( db . DefaultContext , "doesnotexist" )
assert . NoError ( t , err )
activateURL := fmt . Sprintf ( "/user/activate?code=%s" , user . GenerateEmailActivateCode ( "doesnotexist@example.com" ) )
req = NewRequestWithValues ( t , "POST" , activateURL , map [ string ] string {
"password" : "examplePassword!1" ,
} )
resp = session . MakeRequest ( t , req , http . StatusSeeOther )
// should be redirected to accept the invite
assert . Equal ( t , inviteURL , test . RedirectURL ( resp ) )
req = NewRequestWithValues ( t , "POST" , test . RedirectURL ( resp ) , map [ string ] string {
2024-10-10 11:48:21 +08:00
"_csrf" : GetUserCSRFToken ( t , session ) ,
2023-08-31 12:26:13 -04:00
} )
resp = session . MakeRequest ( t , req , http . StatusSeeOther )
req = NewRequest ( t , "GET" , test . RedirectURL ( resp ) )
session . MakeRequest ( t , req , http . StatusOK )
isMember , err := organization . IsTeamMember ( db . DefaultContext , team . OrgID , team . ID , user . ID )
assert . NoError ( t , err )
assert . True ( t , isMember )
}
// Test that a logged-in user who navigates to the sign-up link is then redirected using redirect_to
// For example: an invite may have been created before the user account was created, but they may be
// accepting the invite after having created an account separately
func TestOrgTeamEmailInviteRedirectsExistingUserWithLogin ( t * testing . T ) {
if setting . MailService == nil {
t . Skip ( )
return
}
defer tests . PrepareTestEnv ( t ) ( )
org := unittest . AssertExistsAndLoadBean ( t , & organization . Organization { ID : 3 } )
team := unittest . AssertExistsAndLoadBean ( t , & organization . Team { ID : 2 } )
user := unittest . AssertExistsAndLoadBean ( t , & user_model . User { ID : 5 } )
isMember , err := organization . IsTeamMember ( db . DefaultContext , team . OrgID , team . ID , user . ID )
assert . NoError ( t , err )
assert . False ( t , isMember )
// create the invite
session := loginUser ( t , "user1" )
teamURL := fmt . Sprintf ( "/org/%s/teams/%s" , org . Name , team . Name )
req := NewRequestWithValues ( t , "POST" , teamURL + "/action/add" , map [ string ] string {
2024-10-10 11:48:21 +08:00
"_csrf" : GetUserCSRFToken ( t , session ) ,
2023-08-31 12:26:13 -04:00
"uid" : "1" ,
"uname" : user . Email ,
} )
resp := session . MakeRequest ( t , req , http . StatusSeeOther )
req = NewRequest ( t , "GET" , test . RedirectURL ( resp ) )
session . MakeRequest ( t , req , http . StatusOK )
// get the invite token
invites , err := organization . GetInvitesByTeamID ( db . DefaultContext , team . ID )
assert . NoError ( t , err )
assert . Len ( t , invites , 1 )
// note: the invited user has logged in
session = loginUser ( t , "user5" )
// accept the invite (note: this uses the sign_up url)
inviteURL := fmt . Sprintf ( "/org/invite/%s" , invites [ 0 ] . Token )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "/user/sign_up?redirect_to=%s" , url . QueryEscape ( inviteURL ) ) )
resp = session . MakeRequest ( t , req , http . StatusSeeOther )
assert . Equal ( t , inviteURL , test . RedirectURL ( resp ) )
// make the request
req = NewRequestWithValues ( t , "POST" , test . RedirectURL ( resp ) , map [ string ] string {
2024-10-10 11:48:21 +08:00
"_csrf" : GetUserCSRFToken ( t , session ) ,
2023-08-31 12:26:13 -04:00
} )
resp = session . MakeRequest ( t , req , http . StatusSeeOther )
req = NewRequest ( t , "GET" , test . RedirectURL ( resp ) )
session . MakeRequest ( t , req , http . StatusOK )
isMember , err = organization . IsTeamMember ( db . DefaultContext , team . OrgID , team . ID , user . ID )
assert . NoError ( t , err )
assert . True ( t , isMember )
}