Commit e771d8e9 by Torkel Ödegaard

Organization: You can now update the organization user role directly (without…

Organization: You can now update the organization user role directly (without removing and readding the organization user). Closes #1899
parent b48b11e9
...@@ -4,6 +4,9 @@ ...@@ -4,6 +4,9 @@
- [Issue #1144](https://github.com/grafana/grafana/issues/1144). Templating: You can now select multiple template variables values at the same time. - [Issue #1144](https://github.com/grafana/grafana/issues/1144). Templating: You can now select multiple template variables values at the same time.
- [Issue #1888](https://github.com/grafana/grafana/issues/1144). Templating: Repeat panel or row for each selected template variable value - [Issue #1888](https://github.com/grafana/grafana/issues/1144). Templating: Repeat panel or row for each selected template variable value
**User or Organization admin**
- [Issue #1899](https://github.com/grafana/grafana/issues/1899). Organization: You can now update the organization user role directly (without removing and readding the organization user).
**Backend** **Backend**
- [Issue #1905](https://github.com/grafana/grafana/issues/1905). Github OAuth: You can now configure a Github team membership requirement, thx @dewski - [Issue #1905](https://github.com/grafana/grafana/issues/1905). Github OAuth: You can now configure a Github team membership requirement, thx @dewski
- [Issue #1891](https://github.com/grafana/grafana/issues/1891). Security: New config option to disable the use of gravatar for profile images - [Issue #1891](https://github.com/grafana/grafana/issues/1891). Security: New config option to disable the use of gravatar for profile images
......
...@@ -71,6 +71,7 @@ func Register(r *macaron.Macaron) { ...@@ -71,6 +71,7 @@ func Register(r *macaron.Macaron) {
r.Put("/", bind(m.UpdateOrgCommand{}), UpdateOrg) r.Put("/", bind(m.UpdateOrgCommand{}), UpdateOrg)
r.Post("/users", bind(m.AddOrgUserCommand{}), AddOrgUser) r.Post("/users", bind(m.AddOrgUserCommand{}), AddOrgUser)
r.Get("/users", GetOrgUsers) r.Get("/users", GetOrgUsers)
r.Patch("/users/:id", bind(m.UpdateOrgUserCommand{}), UpdateOrgUser)
r.Delete("/users/:id", RemoveOrgUser) r.Delete("/users/:id", RemoveOrgUser)
}, reqAccountAdmin) }, reqAccountAdmin)
......
...@@ -48,6 +48,23 @@ func GetOrgUsers(c *middleware.Context) { ...@@ -48,6 +48,23 @@ func GetOrgUsers(c *middleware.Context) {
c.JSON(200, query.Result) c.JSON(200, query.Result)
} }
func UpdateOrgUser(c *middleware.Context, cmd m.UpdateOrgUserCommand) {
if !cmd.Role.IsValid() {
c.JsonApiErr(400, "Invalid role specified", nil)
return
}
cmd.UserId = c.ParamsInt64(":id")
cmd.OrgId = c.OrgId
if err := bus.Dispatch(&cmd); err != nil {
c.JsonApiErr(500, "Failed update org user", err)
return
}
c.JsonOK("Organization user updated")
}
func RemoveOrgUser(c *middleware.Context) { func RemoveOrgUser(c *middleware.Context) {
userId := c.ParamsInt64(":id") userId := c.ParamsInt64(":id")
......
...@@ -9,6 +9,7 @@ import ( ...@@ -9,6 +9,7 @@ import (
var ( var (
ErrInvalidRoleType = errors.New("Invalid role type") ErrInvalidRoleType = errors.New("Invalid role type")
ErrLastOrgAdmin = errors.New("Cannot remove last organization admin") ErrLastOrgAdmin = errors.New("Cannot remove last organization admin")
ErrOrgUserNotFound = errors.New("Cannot find the organization user")
) )
type RoleType string type RoleType string
...@@ -24,6 +25,7 @@ func (r RoleType) IsValid() bool { ...@@ -24,6 +25,7 @@ func (r RoleType) IsValid() bool {
} }
type OrgUser struct { type OrgUser struct {
Id int64
OrgId int64 OrgId int64
UserId int64 UserId int64
Role RoleType Role RoleType
...@@ -47,6 +49,13 @@ type AddOrgUserCommand struct { ...@@ -47,6 +49,13 @@ type AddOrgUserCommand struct {
UserId int64 `json:"-"` UserId int64 `json:"-"`
} }
type UpdateOrgUserCommand struct {
Role RoleType `json:"role" binding:"Required"`
OrgId int64 `json:"-"`
UserId int64 `json:"-"`
}
// ---------------------- // ----------------------
// QUERIES // QUERIES
......
...@@ -80,6 +80,19 @@ func TestAccountDataAccess(t *testing.T) { ...@@ -80,6 +80,19 @@ func TestAccountDataAccess(t *testing.T) {
So(err, ShouldBeNil) So(err, ShouldBeNil)
}) })
Convey("Can update org user role", func() {
updateCmd := m.UpdateOrgUserCommand{OrgId: ac1.OrgId, UserId: ac2.Id, Role: m.ROLE_ADMIN}
err = UpdateOrgUser(&updateCmd)
So(err, ShouldBeNil)
orgUsersQuery := m.GetOrgUsersQuery{OrgId: ac1.OrgId}
err = GetOrgUsers(&orgUsersQuery)
So(err, ShouldBeNil)
So(orgUsersQuery.Result[1].Role, ShouldEqual, m.ROLE_ADMIN)
})
Convey("Can get logged in user projection", func() { Convey("Can get logged in user projection", func() {
query := m.GetSignedInUserQuery{UserId: ac2.Id} query := m.GetSignedInUserQuery{UserId: ac2.Id}
err := GetSignedInUser(&query) err := GetSignedInUser(&query)
......
...@@ -14,6 +14,7 @@ func init() { ...@@ -14,6 +14,7 @@ func init() {
bus.AddHandler("sql", AddOrgUser) bus.AddHandler("sql", AddOrgUser)
bus.AddHandler("sql", RemoveOrgUser) bus.AddHandler("sql", RemoveOrgUser)
bus.AddHandler("sql", GetOrgUsers) bus.AddHandler("sql", GetOrgUsers)
bus.AddHandler("sql", UpdateOrgUser)
} }
func AddOrgUser(cmd *m.AddOrgUserCommand) error { func AddOrgUser(cmd *m.AddOrgUserCommand) error {
...@@ -32,6 +33,25 @@ func AddOrgUser(cmd *m.AddOrgUserCommand) error { ...@@ -32,6 +33,25 @@ func AddOrgUser(cmd *m.AddOrgUserCommand) error {
}) })
} }
func UpdateOrgUser(cmd *m.UpdateOrgUserCommand) error {
return inTransaction(func(sess *xorm.Session) error {
var orgUser m.OrgUser
exists, err := sess.Where("org_id=? AND user_id=?", cmd.OrgId, cmd.UserId).Get(&orgUser)
if err != nil {
return err
}
if !exists {
return m.ErrOrgUserNotFound
}
orgUser.Role = cmd.Role
orgUser.Updated = time.Now()
_, err = sess.Id(orgUser.Id).Update(&orgUser)
return err
})
}
func GetOrgUsers(query *m.GetOrgUsersQuery) error { func GetOrgUsers(query *m.GetOrgUsersQuery) error {
query.Result = make([]*m.OrgUserDTO, 0) query.Result = make([]*m.OrgUserDTO, 0)
sess := x.Table("org_user") sess := x.Table("org_user")
......
...@@ -23,6 +23,10 @@ function (angular) { ...@@ -23,6 +23,10 @@ function (angular) {
}); });
}; };
$scope.updateOrgUser = function(user) {
backendSrv.patch('/api/org/users/' + user.userId, user);
};
$scope.removeUser = function(user) { $scope.removeUser = function(user) {
backendSrv.delete('/api/org/users/' + user.userId).then($scope.get); backendSrv.delete('/api/org/users/' + user.userId).then($scope.get);
}; };
......
...@@ -35,7 +35,7 @@ ...@@ -35,7 +35,7 @@
<br> <br>
<table class="grafana-options-table"> <table class="grafana-options-table form-inline">
<tr> <tr>
<th>Login</th> <th>Login</th>
<th>Email</th> <th>Email</th>
...@@ -46,7 +46,8 @@ ...@@ -46,7 +46,8 @@
<td>{{user.login}}</td> <td>{{user.login}}</td>
<td>{{user.email}}</td> <td>{{user.email}}</td>
<td> <td>
{{user.role}} <select type="text" ng-model="user.role" class="input-small" ng-options="f for f in ['Viewer', 'Editor', 'Admin']" ng-change="updateOrgUser(user)">
</select>
</td> </td>
<td style="width: 1%"> <td style="width: 1%">
<a ng-click="removeUser(user)" class="btn btn-danger btn-mini"> <a ng-click="removeUser(user)" class="btn btn-danger btn-mini">
......
...@@ -23,6 +23,10 @@ function (angular, _, config) { ...@@ -23,6 +23,10 @@ function (angular, _, config) {
return this.request({ method: 'POST', url: url, data: data }); return this.request({ method: 'POST', url: url, data: data });
}; };
this.patch = function(url, data) {
return this.request({ method: 'PATCH', url: url, data: data });
};
this.put = function(url, data) { this.put = function(url, data) {
return this.request({ method: 'PUT', url: url, data: data }); return this.request({ method: 'PUT', url: url, data: data });
}; };
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment