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}}
|
||||||
|
|
Reference in a new issue