Python裝飾器使用執行個體:驗證參數合法性

來源:互聯網
上載者:User
python是不帶靜態檢查的動態語言,有時候需要在調用函數時保證參數合法。檢查參數合法性是一個顯著的切面情境,各個函數都可能有這個需求。但另一方面,參數合法性是不是應該由調用方來保證比較好也是一個需要結合實際才能回答的問題,總之雙方約定好,不要都不檢查或者都檢查就可以了。下面這個模組用於在函數上使用裝飾器進行參數的合法性驗證。

你可以直接執行這個模組進行測試,如果完全沒有輸出則表示通過。你也可以找到幾個以_test開頭的函數,所有的測試案例都包含在這幾個函數中。使用方法參見模組文檔和測試案例。

# -*- coding: UTF-8 -*- '''@summary: 驗證器該模組提供了一個裝飾器用於驗證參數是否合法,使用方法為: from validator import validParam, nullOk, multiType @validParam(i=int)def foo(i):  return i+1 編寫驗證器: 1. 僅驗證類型:@validParam(type, ...)例如:檢查第一個位置的參數是否為int類型:@validParam(int)檢查名為x的參數是否為int類型:@validParam(x=int) 驗證多個參數:@validParam(int, int)指定參數名驗證:@validParam(int, s=str) 針對*和**參數編寫的驗證器將驗證這些參數實際包含的每個元素:@validParam(varargs=int)def foo(*varargs): pass @validParam(kws=int)def foo7(s, **kws): pass 2. 帶有條件的驗證:@validParam((type, condition), ...)其中,condition是一個運算式字串,使用x引用待驗證的對象;根據bool(運算式的值)判斷是否通過驗證,若計算運算式時拋出異常,視為失敗。例如:驗證一個10到20之間的整數:@validParam(i=(int, '1020'), nullOk(str, '/^\d+$/'))) 5. 如果有更複雜的驗證需求,還可以編寫一個函數作為驗證函式傳入。這個函數接收待驗證的對象作為參數,根據bool(傳回值)判斷是否通過驗證,拋出異常視為失敗。例如:def validFunction(x):  return isinstance(x, int) and x>0@validParam(i=validFunction)def foo(i): pass 這個驗證函式等價於:@validParam(i=(int, 'x>0'))def foo(i): pass  @author: HUXI@since: 2011-3-22@change: ''' import inspectimport re class ValidateException(Exception): pass  def validParam(*varargs, **keywords):  '''驗證參數的裝飾器。'''     varargs = map(_toStardardCondition, varargs)  keywords = dict((k, _toStardardCondition(keywords[k]))          for k in keywords)     def generator(func):    args, varargname, kwname = inspect.getargspec(func)[:3]    dctValidator = _getcallargs(args, varargname, kwname,                  varargs, keywords)         def wrapper(*callvarargs, **callkeywords):      dctCallArgs = _getcallargs(args, varargname, kwname,                    callvarargs, callkeywords)             k, item = None, None      try:        for k in dctValidator:          if k == varargname:            for item in dctCallArgs[k]:              assert dctValidator[k](item)          elif k == kwname:            for item in dctCallArgs[k].values():              assert dctValidator[k](item)          else:            item = dctCallArgs[k]            assert dctValidator[k](item)      except:        raise ValidateException,\            ('%s() parameter validation fails, param: %s, value: %s(%s)'            % (func.func_name, k, item, item.__class__.__name__))             return func(*callvarargs, **callkeywords)         wrapper = _wrapps(wrapper, func)    return wrapper     return generator  def _toStardardCondition(condition):  '''將各種格式的檢查條件轉換為檢查函數'''     if inspect.isclass(condition):    return lambda x: isinstance(x, condition)     if isinstance(condition, (tuple, list)):    cls, condition = condition[:2]    if condition is None:      return _toStardardCondition(cls)         if cls in (str, unicode) and condition[0] == condition[-1] == '/':      return lambda x: (isinstance(x, cls)               and re.match(condition[1:-1], x) is not None)         return lambda x: isinstance(x, cls) and eval(condition)     return condition  def nullOk(cls, condition=None):  '''這個函數指定的檢查條件可以接受None值'''     return lambda x: x is None or _toStardardCondition((cls, condition))(x)  def multiType(*conditions):  '''這個函數指定的檢查條件只需要有一個通過'''     lstValidator = map(_toStardardCondition, conditions)  def validate(x):    for v in lstValidator:      if v(x):        return True  return validate  def _getcallargs(args, varargname, kwname, varargs, keywords):  '''擷取調用時的各參數名-值的字典'''     dctArgs = {}  varargs = tuple(varargs)  keywords = dict(keywords)     argcount = len(args)  varcount = len(varargs)  callvarargs = None     if argcount <= varcount:    for n, argname in enumerate(args):      dctArgs[argname] = varargs[n]         callvarargs = varargs[-(varcount-argcount):]     else:    for n, var in enumerate(varargs):      dctArgs[args[n]] = var         for argname in args[-(argcount-varcount):]:      if argname in keywords:        dctArgs[argname] = keywords.pop(argname)         callvarargs = ()     if varargname is not None:    dctArgs[varargname] = callvarargs     if kwname is not None:    dctArgs[kwname] = keywords     dctArgs.update(keywords)  return dctArgs  def _wrapps(wrapper, wrapped):  '''複製中繼資料'''     for attr in ('__module__', '__name__', '__doc__'):    setattr(wrapper, attr, getattr(wrapped, attr))  for attr in ('__dict__',):    getattr(wrapper, attr).update(getattr(wrapped, attr, {}))     return wrapper  #===============================================================================# 測試#===============================================================================  def _unittest(func, *cases):  for case in cases:    _functest(func, *case)    def _functest(func, isCkPass, *args, **kws):  if isCkPass:    func(*args, **kws)  else:    try:      func(*args, **kws)      assert False    except ValidateException:      pass def _test1_simple():  #檢查第一個位置的參數是否為int類型:  @validParam(int)  def foo1(i): pass  _unittest(foo1,        (True, 1),        (False, 's'),        (False, None))   #檢查名為x的參數是否為int類型:  @validParam(x=int)  def foo2(s, x): pass  _unittest(foo2,        (True, 1, 2),        (False, 's', 's'))     #驗證多個參數:  @validParam(int, int)  def foo3(s, x): pass  _unittest(foo3,        (True, 1, 2),        (False, 's', 2))     #指定參數名驗證:  @validParam(int, s=str)  def foo4(i, s): pass  _unittest(foo4,        (True, 1, 'a'),        (False, 's', 1))     #針對*和**參數編寫的驗證器將驗證這些參數包含的每個元素:  @validParam(varargs=int)  def foo5(*varargs): pass  _unittest(foo5,       (True, 1, 2, 3, 4, 5),       (False, 'a', 1))     @validParam(kws=int)  def foo6(**kws): pass  _functest(foo6, True, a=1, b=2)  _functest(foo6, False, a='a', b=2)     @validParam(kws=int)  def foo7(s, **kws): pass  _functest(foo7, True, s='a', a=1, b=2)  def _test2_condition():  #驗證一個10到20之間的整數:  @validParam(i=(int, '1020'), nullOk(str, '/^\d+$/')))  def foo2(s): pass  _unittest(foo2,        (False, 1),       (False, 'a'),       (True, None),       (False, 1.1),       (True, 21),       (True, '21')) def _main():  d = globals()  from types import FunctionType  print  for f in d:    if f.startswith('_test'):      f = d[f]      if isinstance(f, FunctionType):        f() if __name__ == '__main__':  _main()
  • 聯繫我們

    該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

    如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

    A Free Trial That Lets You Build Big!

    Start building with 50+ products and up to 12 months usage for Elastic Compute Service

    • Sales Support

      1 on 1 presale consultation

    • After-Sales Support

      24/7 Technical Support 6 Free Tickets per Quarter Faster Response

    • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.