Show user OpenID URIs in their profile (#1314)
This commit is contained in:
		
							parent
							
								
									e1586898b2
								
							
						
					
					
						commit
						9182a35f18
					
				
					 11 changed files with 180 additions and 0 deletions
				
			
		|  | @ -248,6 +248,7 @@ func runWeb(ctx *cli.Context) error { | ||||||
| 				m.Combo("").Get(user.SettingsOpenID). | 				m.Combo("").Get(user.SettingsOpenID). | ||||||
| 					Post(bindIgnErr(auth.AddOpenIDForm{}), user.SettingsOpenIDPost) | 					Post(bindIgnErr(auth.AddOpenIDForm{}), user.SettingsOpenIDPost) | ||||||
| 				m.Post("/delete", user.DeleteOpenID) | 				m.Post("/delete", user.DeleteOpenID) | ||||||
|  | 				m.Post("/toggle_visibility", user.ToggleOpenIDVisibility) | ||||||
| 			}) | 			}) | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										17
									
								
								models/fixtures/user_open_id.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								models/fixtures/user_open_id.yml
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,17 @@ | ||||||
|  | - | ||||||
|  |   id: 1 | ||||||
|  |   uid: 1 | ||||||
|  |   uri: https://user1.domain1.tld/ | ||||||
|  |   show: false | ||||||
|  | 
 | ||||||
|  | - | ||||||
|  |   id: 2 | ||||||
|  |   uid: 1 | ||||||
|  |   uri: http://user1.domain2.tld/ | ||||||
|  |   show: true | ||||||
|  | 
 | ||||||
|  | - | ||||||
|  |   id: 3 | ||||||
|  |   uid: 2 | ||||||
|  |   uri: https://domain1.tld/user2/ | ||||||
|  |   show: true | ||||||
|  | @ -98,6 +98,8 @@ var migrations = []Migration{ | ||||||
| 	NewMigration("add user openid table", addUserOpenID), | 	NewMigration("add user openid table", addUserOpenID), | ||||||
| 	// v24 -> v25
 | 	// v24 -> v25
 | ||||||
| 	NewMigration("change the key_id and primary_key_id type", changeGPGKeysColumns), | 	NewMigration("change the key_id and primary_key_id type", changeGPGKeysColumns), | ||||||
|  | 	// v25 -> v26
 | ||||||
|  | 	NewMigration("add show field in user openid table", addUserOpenIDShow), | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Migrate database to current version
 | // Migrate database to current version
 | ||||||
|  |  | ||||||
							
								
								
									
										18
									
								
								models/migrations/v25.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								models/migrations/v25.go
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,18 @@ | ||||||
|  | // Copyright 2017 Gitea. All rights reserved.
 | ||||||
|  | // Use of this source code is governed by a MIT-style
 | ||||||
|  | // license that can be found in the LICENSE file.
 | ||||||
|  | 
 | ||||||
|  | package migrations | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 
 | ||||||
|  | 	"github.com/go-xorm/xorm" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func addUserOpenIDShow(x *xorm.Engine) error { | ||||||
|  | 	if err := x.Sync2(new(UserOpenID)); err != nil { | ||||||
|  | 		return fmt.Errorf("Sync2: %v", err) | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | @ -21,6 +21,7 @@ type UserOpenID struct { | ||||||
| 	ID          int64  `xorm:"pk autoincr"` | 	ID          int64  `xorm:"pk autoincr"` | ||||||
| 	UID         int64  `xorm:"INDEX NOT NULL"` | 	UID         int64  `xorm:"INDEX NOT NULL"` | ||||||
| 	URI         string `xorm:"UNIQUE NOT NULL"` | 	URI         string `xorm:"UNIQUE NOT NULL"` | ||||||
|  | 	Show        bool   `xorm:"DEFAULT false"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetUserOpenIDs returns all openid addresses that belongs to given user.
 | // GetUserOpenIDs returns all openid addresses that belongs to given user.
 | ||||||
|  | @ -28,6 +29,7 @@ func GetUserOpenIDs(uid int64) ([]*UserOpenID, error) { | ||||||
| 	openids := make([]*UserOpenID, 0, 5) | 	openids := make([]*UserOpenID, 0, 5) | ||||||
| 	if err := x. | 	if err := x. | ||||||
| 		Where("uid=?", uid). | 		Where("uid=?", uid). | ||||||
|  | 		Asc("id"). | ||||||
| 		Find(&openids); err != nil { | 		Find(&openids); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | @ -89,6 +91,12 @@ func DeleteUserOpenID(openid *UserOpenID) (err error) { | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // ToggleUserOpenIDVisibility toggles visibility of an openid address of given user.
 | ||||||
|  | func ToggleUserOpenIDVisibility(id int64) (err error) { | ||||||
|  | 	_, err = x.Exec("update user_open_id set show = not show where id = ?", id) | ||||||
|  | 	return err | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // GetUserByOpenID returns the user object by given OpenID if exists.
 | // GetUserByOpenID returns the user object by given OpenID if exists.
 | ||||||
| func GetUserByOpenID(uri string) (*User, error) { | func GetUserByOpenID(uri string) (*User, error) { | ||||||
| 	if len(uri) == 0 { | 	if len(uri) == 0 { | ||||||
|  |  | ||||||
							
								
								
									
										82
									
								
								models/user_openid_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								models/user_openid_test.go
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,82 @@ | ||||||
|  | // Copyright 2017 The Gitea Authors. All rights reserved.
 | ||||||
|  | // Use of this source code is governed by a MIT-style
 | ||||||
|  | // license that can be found in the LICENSE file.
 | ||||||
|  | 
 | ||||||
|  | package models | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"testing" | ||||||
|  | 
 | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func TestGetUserOpenIDs(t *testing.T) { | ||||||
|  | 	assert.NoError(t, PrepareTestDatabase()) | ||||||
|  | 
 | ||||||
|  | 	oids, err := GetUserOpenIDs(int64(1)) | ||||||
|  | 	if assert.NoError(t, err) { | ||||||
|  | 		assert.Len(t, oids, 2) | ||||||
|  | 		assert.Equal(t, oids[0].URI, "https://user1.domain1.tld/") | ||||||
|  | 		assert.False(t, oids[0].Show) | ||||||
|  | 		assert.Equal(t, oids[1].URI, "http://user1.domain2.tld/") | ||||||
|  | 		assert.True(t, oids[1].Show) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	oids, err = GetUserOpenIDs(int64(2)) | ||||||
|  | 	if assert.NoError(t, err) { | ||||||
|  | 		assert.Len(t, oids, 1) | ||||||
|  | 		assert.Equal(t, oids[0].URI, "https://domain1.tld/user2/") | ||||||
|  | 		assert.True(t, oids[0].Show) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestGetUserByOpenID(t *testing.T) { | ||||||
|  | 	assert.NoError(t, PrepareTestDatabase()) | ||||||
|  | 
 | ||||||
|  | 	user, err := GetUserByOpenID("https://unknown") | ||||||
|  | 	if assert.Error(t, err) { | ||||||
|  | 		assert.True(t, IsErrUserNotExist(err)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	user, err = GetUserByOpenID("https://user1.domain1.tld") | ||||||
|  | 	if assert.NoError(t, err) { | ||||||
|  | 		assert.Equal(t, user.ID, int64(1)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	user, err = GetUserByOpenID("https://domain1.tld/user2/") | ||||||
|  | 	if assert.NoError(t, err) { | ||||||
|  | 		assert.Equal(t, user.ID, int64(2)) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestToggleUserOpenIDVisibility(t *testing.T) { | ||||||
|  | 	assert.NoError(t, PrepareTestDatabase()) | ||||||
|  | 	oids, err := GetUserOpenIDs(int64(2)) | ||||||
|  | 	if ! assert.NoError(t, err) { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	assert.Len(t, oids, 1) | ||||||
|  | 	assert.True(t, oids[0].Show) | ||||||
|  | 
 | ||||||
|  | 	err = ToggleUserOpenIDVisibility(oids[0].ID) | ||||||
|  | 	if ! assert.NoError(t, err) { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	oids, err = GetUserOpenIDs(int64(2)) | ||||||
|  | 	if assert.NoError(t, err) { | ||||||
|  | 		assert.Len(t, oids, 1) | ||||||
|  | 		assert.False(t, oids[0].Show) | ||||||
|  | 	} | ||||||
|  | 	err = ToggleUserOpenIDVisibility(oids[0].ID) | ||||||
|  | 	if ! assert.NoError(t, err) { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	oids, err = GetUserOpenIDs(int64(2)) | ||||||
|  | 	if ! assert.NoError(t, err) { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	assert.Len(t, oids, 1) | ||||||
|  | 	assert.True(t, oids[0].Show) | ||||||
|  | } | ||||||
|  | @ -365,6 +365,8 @@ last_used = Last used on | ||||||
| no_activity = No recent activity | no_activity = No recent activity | ||||||
| key_state_desc = This key is used in last 7 days | key_state_desc = This key is used in last 7 days | ||||||
| token_state_desc = This token is used in last 7 days | token_state_desc = This token is used in last 7 days | ||||||
|  | show_openid = Show on profile | ||||||
|  | hide_openid = Hide from profile | ||||||
| 
 | 
 | ||||||
| manage_social = Manage Associated Social Accounts | manage_social = Manage Associated Social Accounts | ||||||
| social_desc = This is a list of associated social accounts. For security reasons, please make sure you recognize all of these entries, as they can be used to log in to your account. | social_desc = This is a list of associated social accounts. For security reasons, please make sure you recognize all of these entries, as they can be used to log in to your account. | ||||||
|  |  | ||||||
|  | @ -75,9 +75,17 @@ func Profile(ctx *context.Context) { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	// Show OpenID URIs
 | ||||||
|  | 	openIDs, err := models.GetUserOpenIDs(ctxUser.ID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		ctx.Handle(500, "GetUserOpenIDs", err) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	ctx.Data["Title"] = ctxUser.DisplayName() | 	ctx.Data["Title"] = ctxUser.DisplayName() | ||||||
| 	ctx.Data["PageIsUserProfile"] = true | 	ctx.Data["PageIsUserProfile"] = true | ||||||
| 	ctx.Data["Owner"] = ctxUser | 	ctx.Data["Owner"] = ctxUser | ||||||
|  | 	ctx.Data["OpenIDs"] = openIDs | ||||||
| 	showPrivate := ctx.IsSigned && (ctx.User.IsAdmin || ctx.User.ID == ctxUser.ID) | 	showPrivate := ctx.IsSigned && (ctx.User.IsAdmin || ctx.User.ID == ctxUser.ID) | ||||||
| 
 | 
 | ||||||
| 	orgs, err := models.GetOrgsByUserID(ctxUser.ID, showPrivate) | 	orgs, err := models.GetOrgsByUserID(ctxUser.ID, showPrivate) | ||||||
|  |  | ||||||
|  | @ -45,6 +45,12 @@ func SettingsOpenIDPost(ctx *context.Context, form auth.AddOpenIDForm) { | ||||||
| 	ctx.Data["PageIsSettingsOpenID"] = true | 	ctx.Data["PageIsSettingsOpenID"] = true | ||||||
| 
 | 
 | ||||||
| 	if ctx.HasError() { | 	if ctx.HasError() { | ||||||
|  | 		openid, err := models.GetUserOpenIDs(ctx.User.ID) | ||||||
|  | 		if err != nil { | ||||||
|  | 			ctx.Handle(500, "GetUserOpenIDs", err) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		ctx.Data["OpenIDs"] = openid | ||||||
| 		ctx.HTML(200, tplSettingsOpenID) | 		ctx.HTML(200, tplSettingsOpenID) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  | @ -140,3 +146,13 @@ func DeleteOpenID(ctx *context.Context) { | ||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // ToggleOpenIDVisibility response for toggle visibility of user's openid
 | ||||||
|  | func ToggleOpenIDVisibility(ctx *context.Context) { | ||||||
|  | 	if err := models.ToggleUserOpenIDVisibility(ctx.QueryInt64("id")); err != nil { | ||||||
|  | 		ctx.Handle(500, "ToggleUserOpenIDVisibility", err) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	ctx.Redirect(setting.AppSubURL + "/user/settings/openid") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  |  | ||||||
|  | @ -34,6 +34,14 @@ | ||||||
| 									<a target="_blank" rel="noopener" href="{{.Owner.Website}}">{{.Owner.Website}}</a> | 									<a target="_blank" rel="noopener" href="{{.Owner.Website}}">{{.Owner.Website}}</a> | ||||||
| 								</li> | 								</li> | ||||||
| 							{{end}} | 							{{end}} | ||||||
|  | 							{{range .OpenIDs}} | ||||||
|  | 								{{if .Show}} | ||||||
|  | 									<li> | ||||||
|  | 										<i class="fa fa-openid"></i> | ||||||
|  | 										<a target="_blank" rel="noopener" href="{{.URI}}">{{.URI}}</a> | ||||||
|  | 									</li> | ||||||
|  | 								{{end}} | ||||||
|  | 							{{end}} | ||||||
| 							<li><i class="octicon octicon-clock"></i> {{.i18n.Tr "user.join_on"}} {{DateFmtShort .Owner.Created}}</li> | 							<li><i class="octicon octicon-clock"></i> {{.i18n.Tr "user.join_on"}} {{DateFmtShort .Owner.Created}}</li> | ||||||
| 							<li> | 							<li> | ||||||
| 								<i class="octicon octicon-person"></i> | 								<i class="octicon octicon-person"></i> | ||||||
|  |  | ||||||
|  | @ -20,6 +20,24 @@ | ||||||
| 									{{$.i18n.Tr "settings.delete_key"}} | 									{{$.i18n.Tr "settings.delete_key"}} | ||||||
| 								</button> | 								</button> | ||||||
| 							</div> | 							</div> | ||||||
|  | 							<div class="ui right"> | ||||||
|  | 								<form action="{{$.Link}}/toggle_visibility" method="post"> | ||||||
|  | 								{{$.CsrfTokenHtml}} | ||||||
|  | 								<input name="id" type="hidden" value="{{.ID}}"> | ||||||
|  | 								{{if .Show}} | ||||||
|  | 									<button class="ui tiny button"> | ||||||
|  | 									<i class="icon fa-eye"></i> | ||||||
|  | 									{{$.i18n.Tr "settings.hide_openid"}} | ||||||
|  | 									</button> | ||||||
|  | 								{{else}} | ||||||
|  | 									<button class="ui tiny button"> | ||||||
|  | 									<i class="icon fa-eye-slash"></i> | ||||||
|  | 									{{$.i18n.Tr "settings.show_openid"}} | ||||||
|  | 									</button> | ||||||
|  | 								{{end}} | ||||||
|  | 								</button> | ||||||
|  | 								</form> | ||||||
|  | 							</div> | ||||||
| 						</div> | 						</div> | ||||||
| 					</div> | 					</div> | ||||||
| 				{{end}} | 				{{end}} | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue