upd: improve user api (#878)

This commit is contained in:
jbe-dw 2021-03-16 19:39:53 +01:00 committed by GitHub
parent 46993e08c0
commit 86700f8fd7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 2150 additions and 42 deletions

View File

@ -93,6 +93,15 @@ class AccountCreateFail(StructuredException):
self.name = name self.name = name
class AccountCreateDuplicate(StructuredException):
status_code = 409
def __init__(self, name=None, message="Creation of account failed"):
StructuredException.__init__(self)
self.message = message
self.name = name
class AccountUpdateFail(StructuredException): class AccountUpdateFail(StructuredException):
status_code = 500 status_code = 500
@ -120,6 +129,14 @@ class UserCreateFail(StructuredException):
self.name = name self.name = name
class UserCreateDuplicate(StructuredException):
status_code = 409
def __init__(self, name=None, message="Creation of user failed"):
StructuredException.__init__(self)
self.message = message
self.name = name
class UserUpdateFail(StructuredException): class UserUpdateFail(StructuredException):
status_code = 500 status_code = 500
@ -128,6 +145,14 @@ class UserUpdateFail(StructuredException):
self.message = message self.message = message
self.name = name self.name = name
class UserUpdateFailEmail(StructuredException):
status_code = 409
def __init__(self, name=None, message="Update of user failed"):
StructuredException.__init__(self)
self.message = message
self.name = name
class UserDeleteFail(StructuredException): class UserDeleteFail(StructuredException):
status_code = 500 status_code = 500

View File

@ -27,6 +27,11 @@ class ApiPlainKeySchema(Schema):
plain_key = fields.String() plain_key = fields.String()
class AccountSummarySchema(Schema):
id = fields.Integer()
name = fields.String()
class UserSchema(Schema): class UserSchema(Schema):
id = fields.Integer() id = fields.Integer()
username = fields.String() username = fields.String()
@ -35,6 +40,14 @@ class UserSchema(Schema):
email = fields.String() email = fields.String()
role = fields.Embed(schema=RoleSchema) role = fields.Embed(schema=RoleSchema)
class UserDetailedSchema(Schema):
id = fields.Integer()
username = fields.String()
firstname = fields.String()
lastname = fields.String()
email = fields.String()
role = fields.Embed(schema=RoleSchema)
accounts = fields.Embed(schema=AccountSummarySchema)
class AccountSchema(Schema): class AccountSchema(Schema):
id = fields.Integer() id = fields.Integer()

View File

@ -7,6 +7,7 @@ import ldap
import ldap.filter import ldap.filter
from flask import current_app from flask import current_app
from flask_login import AnonymousUserMixin from flask_login import AnonymousUserMixin
from sqlalchemy import orm
from .base import db from .base import db
from .role import Role from .role import Role
@ -29,6 +30,7 @@ class User(db.Model):
otp_secret = db.Column(db.String(16)) otp_secret = db.Column(db.String(16))
confirmed = db.Column(db.SmallInteger, nullable=False, default=0) confirmed = db.Column(db.SmallInteger, nullable=False, default=0)
role_id = db.Column(db.Integer, db.ForeignKey('role.id')) role_id = db.Column(db.Integer, db.ForeignKey('role.id'))
accounts = None
def __init__(self, def __init__(self,
id=None, id=None,
@ -591,6 +593,10 @@ class User(db.Model):
else: else:
return {'status': False, 'msg': 'Role does not exist'} return {'status': False, 'msg': 'Role does not exist'}
@orm.reconstructor
def set_account(self):
self.accounts = self.get_accounts()
def get_accounts(self): def get_accounts(self):
""" """
Get accounts associated with this user Get accounts associated with this user
@ -602,7 +608,7 @@ class User(db.Model):
.query( .query(
AccountUser, AccountUser,
Account)\ Account)\
.filter(User.id == AccountUser.user_id)\ .filter(self.id == AccountUser.user_id)\
.filter(Account.id == AccountUser.account_id)\ .filter(Account.id == AccountUser.account_id)\
.all() .all()
for q in query: for q in query:

View File

@ -14,13 +14,16 @@ from ..models import (
from ..lib import utils, helper from ..lib import utils, helper
from ..lib.schema import ( from ..lib.schema import (
ApiKeySchema, DomainSchema, ApiPlainKeySchema, UserSchema, AccountSchema, ApiKeySchema, DomainSchema, ApiPlainKeySchema, UserSchema, AccountSchema,
UserDetailedSchema,
) )
from ..lib.errors import ( from ..lib.errors import (
StructuredException, StructuredException,
DomainNotExists, DomainAlreadyExists, DomainAccessForbidden, DomainNotExists, DomainAlreadyExists, DomainAccessForbidden,
RequestIsNotJSON, ApiKeyCreateFail, ApiKeyNotUsable, NotEnoughPrivileges, RequestIsNotJSON, ApiKeyCreateFail, ApiKeyNotUsable, NotEnoughPrivileges,
AccountCreateFail, AccountUpdateFail, AccountDeleteFail, AccountCreateFail, AccountUpdateFail, AccountDeleteFail,
UserCreateFail, UserUpdateFail, UserDeleteFail, AccountCreateDuplicate,
UserCreateFail, UserCreateDuplicate, UserUpdateFail, UserDeleteFail,
UserUpdateFailEmail,
) )
from ..decorators import ( from ..decorators import (
api_basic_auth, api_can_create_domain, is_json, apikey_auth, api_basic_auth, api_can_create_domain, is_json, apikey_auth,
@ -33,11 +36,14 @@ import string
api_bp = Blueprint('api', __name__, url_prefix='/api/v1') api_bp = Blueprint('api', __name__, url_prefix='/api/v1')
apikey_schema = ApiKeySchema(many=True) apikey_schema = ApiKeySchema(many=True)
apikey_single_schema = ApiKeySchema()
domain_schema = DomainSchema(many=True) domain_schema = DomainSchema(many=True)
apikey_plain_schema = ApiPlainKeySchema(many=True) apikey_plain_schema = ApiPlainKeySchema()
user_schema = UserSchema(many=True) user_schema = UserSchema(many=True)
user_single_schema = UserSchema()
user_detailed_schema = UserDetailedSchema()
account_schema = AccountSchema(many=True) account_schema = AccountSchema(many=True)
account_single_schema = AccountSchema()
def get_user_domains(): def get_user_domains():
domains = db.session.query(Domain) \ domains = db.session.query(Domain) \
@ -360,7 +366,7 @@ def api_generate_apikey():
raise ApiKeyCreateFail(message='Api key create failed') raise ApiKeyCreateFail(message='Api key create failed')
apikey.plain_key = b64encode(apikey.plain_key.encode('utf-8')).decode('utf-8') apikey.plain_key = b64encode(apikey.plain_key.encode('utf-8')).decode('utf-8')
return jsonify(apikey_plain_schema.dump([apikey])[0]), 201 return jsonify(apikey_plain_schema.dump(apikey)), 201
@api_bp.route('/pdnsadmin/apikeys', defaults={'domain_name': None}) @api_bp.route('/pdnsadmin/apikeys', defaults={'domain_name': None})
@ -418,7 +424,7 @@ def api_get_apikey(apikey_id):
if apikey_id not in [a.id for a in get_user_apikeys()]: if apikey_id not in [a.id for a in get_user_apikeys()]:
raise DomainAccessForbidden() raise DomainAccessForbidden()
return jsonify(apikey_schema.dump([apikey])[0]), 200 return jsonify(apikey_single_schema.dump(apikey)), 200
@api_bp.route('/pdnsadmin/apikeys/<int:apikey_id>', methods=['DELETE']) @api_bp.route('/pdnsadmin/apikeys/<int:apikey_id>', methods=['DELETE'])
@ -566,12 +572,12 @@ def api_update_apikey(apikey_id):
def api_list_users(username=None): def api_list_users(username=None):
if username is None: if username is None:
user_list = [] or User.query.all() user_list = [] or User.query.all()
return jsonify(user_schema.dump(user_list)), 200
else: else:
user_list = [] or User.query.filter(User.username == username).all() user = User.query.filter(User.username == username).first()
if not user_list: if user is None:
abort(404) abort(404)
return jsonify(user_detailed_schema.dump(user)), 200
return jsonify(user_schema.dump(user_list)), 200
@api_bp.route('/pdnsadmin/users', methods=['POST']) @api_bp.route('/pdnsadmin/users', methods=['POST'])
@ -639,12 +645,12 @@ def api_create_user():
if not result['status']: if not result['status']:
current_app.logger.warning('Create user ({}, {}) error: {}'.format( current_app.logger.warning('Create user ({}, {}) error: {}'.format(
username, email, result['msg'])) username, email, result['msg']))
raise UserCreateFail(message=result['msg']) raise UserCreateDuplicate(message=result['msg'])
history = History(msg='Created user {0}'.format(user.username), history = History(msg='Created user {0}'.format(user.username),
created_by=current_user.username) created_by=current_user.username)
history.add() history.add()
return jsonify(user_schema.dump([user])), 201 return jsonify(user_single_schema.dump(user)), 201
@api_bp.route('/pdnsadmin/users/<int:user_id>', methods=['PUT']) @api_bp.route('/pdnsadmin/users/<int:user_id>', methods=['PUT'])
@ -708,7 +714,10 @@ def api_update_user(user_id):
if not result['status']: if not result['status']:
current_app.logger.warning('Update user ({}, {}) error: {}'.format( current_app.logger.warning('Update user ({}, {}) error: {}'.format(
username, email, result['msg'])) username, email, result['msg']))
raise UserCreateFail(message=result['msg']) if result['msg'].startswith('New email'):
raise UserUpdateFailEmail(message=result['msg'])
else:
raise UserCreateFail(message=result['msg'])
history = History(msg='Updated user {0}'.format(user.username), history = History(msg='Updated user {0}'.format(user.username),
created_by=current_user.username) created_by=current_user.username)
@ -759,25 +768,18 @@ def api_list_accounts(account_name):
else: else:
if account_name is None: if account_name is None:
account_list = [] or Account.query.all() account_list = [] or Account.query.all()
return jsonify(account_schema.dump(account_list)), 200
else: else:
account_list = [] or Account.query.filter( account = Account.query.filter(
Account.name == account_name).all() Account.name == account_name).first()
if not account_list: if account is None:
abort(404) abort(404)
if account_name is None: return jsonify(account_single_schema.dump(account)), 200
return jsonify(account_schema.dump(account_list)), 200
else:
return jsonify(account_schema.dump(account_list)[0]), 200
@api_bp.route('/pdnsadmin/accounts', methods=['POST']) @api_bp.route('/pdnsadmin/accounts', methods=['POST'])
@api_basic_auth @api_basic_auth
def api_create_account(): def api_create_account():
account_exists = [] or Account.query.filter(Account.name == account_name).all()
if len(account_exists) > 0:
msg = "Account name already exists"
current_app.logger.debug(msg)
raise AccountCreateFail(message=msg)
if current_user.role.name not in ['Administrator', 'Operator']: if current_user.role.name not in ['Administrator', 'Operator']:
msg = "{} role cannot create accounts".format(current_user.role.name) msg = "{} role cannot create accounts".format(current_user.role.name)
raise NotEnoughPrivileges(message=msg) raise NotEnoughPrivileges(message=msg)
@ -790,6 +792,12 @@ def api_create_account():
current_app.logger.debug("Account name missing") current_app.logger.debug("Account name missing")
abort(400) abort(400)
account_exists = [] or Account.query.filter(Account.name == name).all()
if len(account_exists) > 0:
msg = "Account {} already exists".format(name)
current_app.logger.debug(msg)
raise AccountCreateDuplicate(message=msg)
account = Account(name=name, account = Account(name=name,
description=description, description=description,
contact=contact, contact=contact,
@ -806,7 +814,7 @@ def api_create_account():
history = History(msg='Create account {0}'.format(account.name), history = History(msg='Create account {0}'.format(account.name),
created_by=current_user.username) created_by=current_user.username)
history.add() history.add()
return jsonify(account_schema.dump([account])[0]), 201 return jsonify(account_single_schema.dump(account)), 201
@api_bp.route('/pdnsadmin/accounts/<int:account_id>', methods=['PUT']) @api_bp.route('/pdnsadmin/accounts/<int:account_id>', methods=['PUT'])
@ -871,6 +879,7 @@ def api_delete_account(account_id):
@api_bp.route('/pdnsadmin/accounts/users/<int:account_id>', methods=['GET']) @api_bp.route('/pdnsadmin/accounts/users/<int:account_id>', methods=['GET'])
@api_bp.route('/pdnsadmin/accounts/<int:account_id>/users', methods=['GET'])
@api_basic_auth @api_basic_auth
@api_role_can('list account users') @api_role_can('list account users')
def api_list_account_users(account_id): def api_list_account_users(account_id):
@ -885,6 +894,9 @@ def api_list_account_users(account_id):
@api_bp.route( @api_bp.route(
'/pdnsadmin/accounts/users/<int:account_id>/<int:user_id>', '/pdnsadmin/accounts/users/<int:account_id>/<int:user_id>',
methods=['PUT']) methods=['PUT'])
@api_bp.route(
'/pdnsadmin/accounts/<int:account_id>/users/<int:user_id>',
methods=['PUT'])
@api_basic_auth @api_basic_auth
@api_role_can('add user to account') @api_role_can('add user to account')
def api_add_account_user(account_id, user_id): def api_add_account_user(account_id, user_id):
@ -909,6 +921,9 @@ def api_add_account_user(account_id, user_id):
@api_bp.route( @api_bp.route(
'/pdnsadmin/accounts/users/<int:account_id>/<int:user_id>', '/pdnsadmin/accounts/users/<int:account_id>/<int:user_id>',
methods=['DELETE']) methods=['DELETE'])
@api_bp.route(
'/pdnsadmin/accounts/<int:account_id>/users/<int:user_id>',
methods=['DELETE'])
@api_basic_auth @api_basic_auth
@api_role_can('remove user from account') @api_role_can('remove user from account')
def api_remove_account_user(account_id, user_id): def api_remove_account_user(account_id, user_id):

View File

@ -1,6 +1,6 @@
swagger: '2.0' swagger: '2.0'
info: info:
version: "0.0.13" version: "0.0.14"
title: PowerDNS Admin Authoritative HTTP API title: PowerDNS Admin Authoritative HTTP API
license: license:
name: MIT name: MIT
@ -1041,6 +1041,10 @@ paths:
description: Unprocessable Entry, the User data provided has issues description: Unprocessable Entry, the User data provided has issues
schema: schema:
$ref: '#/definitions/Error' $ref: '#/definitions/Error'
'409':
description: Duplicate Entry, either the Name or the Email is already in use
schema:
$ref: '#/definitions/Error'
'500': '500':
description: Internal Server Error. There was a problem creating the user description: Internal Server Error. There was a problem creating the user
schema: schema:
@ -1063,7 +1067,7 @@ paths:
'200': '200':
description: Retrieve a specific User description: Retrieve a specific User
schema: schema:
$ref: '#/definitions/User' $ref: '#/definitions/UserDetailed'
'404': '404':
description: Not found. The User with the specified username does not exist description: Not found. The User with the specified username does not exist
schema: schema:
@ -1206,6 +1210,10 @@ paths:
description: Unprocessable Entry, the Account data provided has issues. description: Unprocessable Entry, the Account data provided has issues.
schema: schema:
$ref: '#/definitions/Error' $ref: '#/definitions/Error'
'409':
description: Duplicate Entry, the Name is already in use
schema:
$ref: '#/definitions/Error'
'500': '500':
description: Internal Server Error. There was a problem creating the account description: Internal Server Error. There was a problem creating the account
schema: schema:
@ -1299,7 +1307,7 @@ paths:
description: Internal Server Error. Contains error message description: Internal Server Error. Contains error message
schema: schema:
$ref: '#/definitions/Error' $ref: '#/definitions/Error'
'/pdnsadmin/accounts/users/{account_id}': '/pdnsadmin/accounts/{account_id}/users':
parameters: parameters:
- name: account_id - name: account_id
type: integer type: integer
@ -1316,7 +1324,7 @@ paths:
- user - user
responses: responses:
'200': '200':
description: List of User objects description: List of Summarized User objects
schema: schema:
type: array type: array
items: items:
@ -1329,7 +1337,7 @@ paths:
description: Internal Server Error, accounts could not be retrieved. Contains error message description: Internal Server Error, accounts could not be retrieved. Contains error message
schema: schema:
$ref: '#/definitions/Error' $ref: '#/definitions/Error'
'/pdnsadmin/accounts/users/{account_id}/{user_id}': '/pdnsadmin/accounts/{account_id}/users/{user_id}':
parameters: parameters:
- name: account_id - name: account_id
type: integer type: integer
@ -1380,7 +1388,6 @@ paths:
schema: schema:
$ref: '#/definitions/Error' $ref: '#/definitions/Error'
definitions: definitions:
Server: Server:
title: Server title: Server
@ -1603,9 +1610,9 @@ definitions:
type: string type: string
description: 'Name of the zone' description: 'Name of the zone'
PDNSAdminApiKeyRole: PDNSAdminRole:
title: PDNSAdminApiKeyRole title: PDNSAdminRole
description: Role of ApiKey, defines privileges on domains description: Roles of PowerDNS Admin
properties: properties:
id: id:
type: integer type: integer
@ -1613,7 +1620,7 @@ definitions:
readOnly: true readOnly: true
name: name:
type: string type: string
description: 'Name of role' description: 'The Name of PDNSAdmin role'
ApiKey: ApiKey:
title: ApiKey title: ApiKey
@ -1630,12 +1637,10 @@ definitions:
type: string type: string
description: 'not used on POST, POSTing to server generates the key material' description: 'not used on POST, POSTing to server generates the key material'
domains: domains:
type: array $ref: '#/definitions/PDNSAdminZones'
items:
$ref: '#/definitions/PDNSAdminZones'
description: 'domains to which this apikey has access' description: 'domains to which this apikey has access'
role: role:
$ref: '#/definitions/PDNSAdminApiKeyRole' $ref: '#/definitions/PDNSAdminRole'
description: description:
type: string type: string
description: 'Some user defined description' description: 'Some user defined description'
@ -1676,10 +1681,51 @@ definitions:
type: boolean type: boolean
description: The confirmed status description: The confirmed status
readOnly: false readOnly: false
role_id: role:
$ref: '#/definitions/PDNSAdminRole'
UserDetailed:
title: User
description: User that can access the gui/api
properties:
id:
type: integer type: integer
description: The ID of the role description: The ID for this user (unique)
readOnly: true
username:
type: string
description: The username for this user (unique, immutable)
readOnly: false readOnly: false
password:
type: string
description: The hashed password for this user
readOnly: false
firstname:
type: string
description: The firstname of this user
readOnly: false
lastname:
type: string
description: The lastname of this user
readOnly: false
email:
type: string
description: Email addres for this user
readOnly: false
otp_secret:
type: string
description: OTP secret
readOnly: false
confirmed:
type: boolean
description: The confirmed status
readOnly: false
role:
$ref: '#/definitions/PDNSAdminRole'
accounts:
type: array
items:
$ref: '#/definitions/AccountSummary'
Account: Account:
title: Account title: Account
@ -1706,6 +1752,19 @@ definitions:
description: The email address of the contact for this account description: The email address of the contact for this account
readOnly: false readOnly: false
AccountSummary:
title: AccountSummry
description: Summary of an Account that 'owns' zones
properties:
id:
type: integer
description: The ID for this account (unique)
readOnly: true
name:
type: string
description: The name for this account (unique, immutable)
readOnly: false
ConfigSetting: ConfigSetting:
title: ConfigSetting title: ConfigSetting
properties: properties:

1990
swagger-specv2.yaml Normal file

File diff suppressed because it is too large Load Diff