Source code for privacyidea.lib.tokens.vascotoken

#  License:  AGPLv3
#  contact:  http://www.privacyidea.org
#
#  2018-01-15   Friedrich Weber <friedrich.weber@netknights.it>
#               Initial version of the VASCO token
#
# This code is free software; you can redistribute it and/or
# modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
# License as published by the Free Software Foundation; either
# version 3 of the License, or any later version.
#
# This code is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU AFFERO GENERAL PUBLIC LICENSE for more details.
#
# You should have received a copy of the GNU Affero General Public
# License along with this program.  If not, see <http://www.gnu.org/licenses/>.
#
#
import binascii

__doc__ = """This is the implementation of the VASCO token"""

import logging
from privacyidea.lib.params import get_optional
from privacyidea.lib.utils import is_true
from privacyidea.lib.decorators import check_token_locked
from privacyidea.lib.error import ParameterError
from privacyidea.lib.log import log_with
from privacyidea.lib.tokenclass import TokenClass
from privacyidea.lib.tokens.vasco import vasco_otp_check
from privacyidea.lib import _
from privacyidea.lib.policy import SCOPE, GROUP
from privacyidea.lib.policies.actions import PolicyAction


log = logging.getLogger(__name__)


[docs] class VascoTokenClass(TokenClass): """ Token class for VASCO Digipass tokens. Relies on vendor-specific shared library, whose location needs to be set in the PI_VASCO_LIBRARY config option. VASCO Tokens can be read from a CSV file which is structured as follows:: <serial1>,<hexlify(blob1)>,vasco <serial2>,<hexlify(blob2)>,vasco ... whereas blobX is the 248-byte blob holding the token information. Consequently, hexlify(blobX) is a 496-character hex string. The CSV file can be imported by using the "Import Tokens" feature of the Web UI, where "OATH CSV" needs to be chosen as the file type. """ def __init__(self, db_token): """ constructor - create a token class object with its db token binding :param aToken: the db bound token """ TokenClass.__init__(self, db_token) self.set_type("vasco") self.hKeyRequired = True
[docs] @staticmethod def get_class_type(): """ return the class type identifier """ return "vasco"
[docs] @staticmethod def get_class_prefix(): """ return the token type prefix """ # TODO: Revisit token type? return "VASC"
[docs] @staticmethod @log_with(log) def get_class_info(key=None, ret='all'): """ :param key: subsection identifier :type key: string :param ret: default return value, if nothing is found :type ret: user defined :return: subsection if key exists or user defined :rtype: dict or string """ res = {'type': 'vasco', 'title': 'VASCO Token', 'description': _('VASCO Token: Authentication using VASCO tokens'), # If this was set, the user could enroll a Vasco token via the API # 'user': ["enroll"], # only administrators can enroll the token in the UI 'ui_enroll': ["admin"], 'policy': { SCOPE.ENROLL: { PolicyAction.MAXTOKENUSER: { 'type': 'int', 'desc': _("The user may only have this maximum number of Vasco tokens assigned."), 'group': GROUP.TOKEN }, PolicyAction.MAXACTIVETOKENUSER: { 'type': 'int', 'desc': _( "The user may only have this maximum number of active Vasco tokens assigned."), 'group': GROUP.TOKEN } } }, } if key: ret = res.get(key, {}) else: if ret == 'all': ret = res return ret
[docs] @log_with(log) def update(self, param, reset_failcount=True): """ update - process initialization parameters :param param: dict of initialization parameters :type param: dict :return: nothing """ if is_true(get_optional(param, 'genkey')): raise ParameterError("Generating OTP keys is not supported") upd_param = param.copy() # If the OTP key is given, it is given as a 496-character hex string which # encodes a 248-byte blob. As we want to set a 248-byte OTPKey (= Blob), # we unhexlify the OTP key if 'otpkey' in param: if len(param['otpkey']) != 496: raise ParameterError('Expected OTP key as 496-character hex string, but length is {!s}'.format( len(param['otpkey']) )) try: upd_param['otpkey'] = binascii.unhexlify(upd_param['otpkey']) except (binascii.Error, TypeError): raise ParameterError('Expected OTP key as 496-character hex string, but it is malformed') TokenClass.update(self, upd_param, reset_failcount)
[docs] @check_token_locked def check_otp(self, otpval, counter=None, window=None, options=None): secret = self.token.get_otpkey().getKey() # vasco_otp_check expects a bytestring, so we encode ``otpval`` to bytes (should be ASCII anyway) result, new_secret = vasco_otp_check(secret, otpval.encode("utf-8")) # By default, setting a new OTP key resets the failcounter. In case of the VASCO token, # this would mean that the failcounter is reset at every authentication attempt # (regardless of success or failure), which must be avoided. self.token.set_otpkey(new_secret, reset_failcount=False) self.save() if result == 0: # Successful authentication return 0 else: if result == 1: # wrong OTP value, no log message pass elif result == 201: log.warning( "VASCO token failed to authenticate, code replay attempt, previous OTP value was used again!") elif result == 202: log.warning("Token-internal fail counter reached its maximum!") elif result == -202: log.warning( "VASCO token failed to authenticate, response too small, user did not type his complete OTP!") elif result == -205: log.warning("VASCO token failed to authenticate, response not decimal!") else: log.warning(f"VASCO token failed to authenticate, result: {result!r}") return -1
[docs] def export_token(self) -> dict: """ Export for this token is not supported. """ raise NotImplementedError("Export for VASCO token is not supported.")
[docs] def import_token(self, token_information: dict): """ Import for this token is not supported. Concern that the database token must be deleted manually. """ raise NotImplementedError("Import for VASCO token is not supported.")