blob: d2a3735a2b3ad31f249db7c46d48e8956deebf6b [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package etcd
import (
"context"
"encoding/json"
"fmt"
"strconv"
"time"
crbac "github.com/go-chassis/cari/rbac"
"github.com/little-cui/etcdadpt"
"github.com/apache/servicecomb-service-center/datasource"
"github.com/apache/servicecomb-service-center/datasource/etcd/path"
esync "github.com/apache/servicecomb-service-center/datasource/etcd/sync"
"github.com/apache/servicecomb-service-center/datasource/rbac"
"github.com/apache/servicecomb-service-center/pkg/etcdsync"
"github.com/apache/servicecomb-service-center/pkg/log"
"github.com/apache/servicecomb-service-center/pkg/util"
)
func (rm *RbacDAO) CreateRole(ctx context.Context, r *crbac.Role) error {
lock, err := etcdsync.Lock("/role-creating/"+r.Name, -1, false)
if err != nil {
return fmt.Errorf("role %s is creating", r.Name)
}
defer func() {
err := lock.Unlock()
if err != nil {
log.Error("can not release role lock", err)
}
}()
key := path.GenerateRBACRoleKey(r.Name)
exist, err := rm.RoleExist(ctx, r.Name)
if err != nil {
log.Error("can not save role info", err)
return err
}
if exist {
return rbac.ErrRoleDuplicated
}
r.ID = util.GenerateUUID()
r.CreateTime = strconv.FormatInt(time.Now().Unix(), 10)
r.UpdateTime = r.CreateTime
value, err := json.Marshal(r)
if err != nil {
log.Error("role info is invalid", err)
return err
}
opts := []etcdadpt.OpOptions{
etcdadpt.OpPut(etcdadpt.WithStrKey(key), etcdadpt.WithValue(value)),
}
syncOpts, err := esync.GenCreateOpts(ctx, datasource.ResourceRole, r)
if err != nil {
log.Error("fail to create sync opts", err)
return err
}
opts = append(opts, syncOpts...)
err = etcdadpt.Txn(ctx, opts)
if err != nil {
log.Error("can not save account info", err)
return err
}
log.Info("create new role: " + r.ID)
return nil
}
func (rm *RbacDAO) RoleExist(ctx context.Context, name string) (bool, error) {
return etcdadpt.Exist(ctx, path.GenerateRBACRoleKey(name))
}
func (rm *RbacDAO) GetRole(ctx context.Context, name string) (*crbac.Role, error) {
kv, err := etcdadpt.Get(ctx, path.GenerateRBACRoleKey(name))
if err != nil {
return nil, err
}
if kv == nil {
return nil, rbac.ErrRoleNotExist
}
role := &crbac.Role{}
err = json.Unmarshal(kv.Value, role)
if err != nil {
log.Error("role info format invalid", err)
return nil, err
}
return role, nil
}
func (rm *RbacDAO) ListRole(ctx context.Context) ([]*crbac.Role, int64, error) {
kvs, n, err := etcdadpt.List(ctx, path.GenerateRBACRoleKey(""))
if err != nil {
return nil, 0, err
}
roles := make([]*crbac.Role, 0, n)
for _, v := range kvs {
r := &crbac.Role{}
err = json.Unmarshal(v.Value, r)
if err != nil {
log.Error("role info format invalid:", err)
continue //do not fail if some role is invalid
}
roles = append(roles, r)
}
return roles, n, nil
}
func (rm *RbacDAO) DeleteRole(ctx context.Context, name string) (bool, error) {
exists, err := RoleBindingExists(ctx, name)
if err != nil {
log.Error("", err)
return false, err
}
if exists {
return false, rbac.ErrRoleBindingExist
}
opts := []etcdadpt.OpOptions{etcdadpt.OpDel(etcdadpt.WithStrKey(path.GenerateRBACRoleKey(name)))}
syncOpts, err := esync.GenDeleteOpts(ctx, datasource.ResourceRole, name, name)
if err != nil {
log.Error("fail to create sync opts", err)
return false, err
}
opts = append(opts, syncOpts...)
err = etcdadpt.Txn(ctx, opts)
if err != nil {
return false, err
}
return true, nil
}
func RoleBindingExists(ctx context.Context, role string) (bool, error) {
_, total, err := etcdadpt.List(ctx, path.GenRoleAccountPrefixIdxKey(role))
if err != nil {
log.Error("", err)
return false, err
}
return total > 0, nil
}
func (rm *RbacDAO) UpdateRole(ctx context.Context, name string, role *crbac.Role) error {
role.UpdateTime = strconv.FormatInt(time.Now().Unix(), 10)
value, err := json.Marshal(role)
if err != nil {
log.Error("role info is invalid", err)
return err
}
opts := []etcdadpt.OpOptions{
etcdadpt.OpPut(etcdadpt.WithStrKey(path.GenerateRBACRoleKey(name)), etcdadpt.WithValue(value)),
}
syncOpts, err := esync.GenUpdateOpts(ctx, datasource.ResourceRole, role)
if err != nil {
log.Error("fail to create sync opts", err)
return err
}
opts = append(opts, syncOpts...)
return etcdadpt.Txn(ctx, opts)
}