traitlets.py
1523 lines
| 49.3 KiB
| text/x-python
|
PythonLexer
Brian Granger
|
r2157 | # encoding: utf-8 | ||
""" | ||||
A lightweight Traits like module. | ||||
Brian Granger
|
r2175 | This is designed to provide a lightweight, simple, pure Python version of | ||
many of the capabilities of enthought.traits. This includes: | ||||
* Validation | ||||
* Type specification with defaults | ||||
* Static and dynamic notification | ||||
* Basic predefined types | ||||
* An API that is similar to enthought.traits | ||||
We don't support: | ||||
* Delegation | ||||
* Automatic GUI generation | ||||
Brian Granger
|
r2179 | * A full set of trait types. Most importantly, we don't provide container | ||
Dav Clark
|
r2385 | traits (list, dict, tuple) that can trigger notifications if their | ||
Brian Granger
|
r2179 | contents change. | ||
Brian Granger
|
r2175 | * API compatibility with enthought.traits | ||
Brian Granger
|
r2179 | There are also some important difference in our design: | ||
* enthought.traits does not validate default values. We do. | ||||
Brian Granger
|
r2175 | We choose to create this module because we need these capabilities, but | ||
we need them to be pure Python so they work in all Python implementations, | ||||
including Jython and IronPython. | ||||
Thomas Kluyver
|
r8795 | Inheritance diagram: | ||
.. inheritance-diagram:: IPython.utils.traitlets | ||||
:parts: 3 | ||||
Brian Granger
|
r2157 | """ | ||
Min RK
|
r16216 | # Copyright (c) IPython Development Team. | ||
# Distributed under the terms of the Modified BSD License. | ||||
Brian Granger
|
r2157 | # | ||
Min RK
|
r16216 | # Adapted from enthought.traits, Copyright (c) Enthought, Inc., | ||
# also under the terms of the Modified BSD License. | ||||
Brian Granger
|
r2157 | |||
Jason Grout
|
r15008 | import contextlib | ||
Brian Granger
|
r2157 | import inspect | ||
Thomas Kluyver
|
r4047 | import re | ||
Brian Granger
|
r2177 | import sys | ||
Brian Granger
|
r2157 | import types | ||
Thomas Kluyver
|
r4732 | from types import FunctionType | ||
try: | ||||
from types import ClassType, InstanceType | ||||
ClassTypes = (ClassType, type) | ||||
except: | ||||
ClassTypes = (type,) | ||||
Bernardo B. Marques
|
r4872 | |||
Thomas Kluyver
|
r3108 | from .importstring import import_item | ||
Thomas Kluyver
|
r4732 | from IPython.utils import py3compat | ||
Thomas Kluyver
|
r13361 | from IPython.utils.py3compat import iteritems | ||
Jonathan Frederic
|
r15011 | from IPython.testing.skipdoctest import skip_doctest | ||
Brian Granger
|
r2229 | |||
Thomas Kluyver
|
r4732 | SequenceTypes = (list, tuple, set, frozenset) | ||
Brian Granger
|
r2204 | |||
Brian Granger
|
r2157 | #----------------------------------------------------------------------------- | ||
# Basic classes | ||||
#----------------------------------------------------------------------------- | ||||
class NoDefaultSpecified ( object ): pass | ||||
NoDefaultSpecified = NoDefaultSpecified() | ||||
class Undefined ( object ): pass | ||||
Undefined = Undefined() | ||||
Dav Clark
|
r2384 | class TraitError(Exception): | ||
pass | ||||
Brian Granger
|
r2157 | |||
#----------------------------------------------------------------------------- | ||||
# Utilities | ||||
#----------------------------------------------------------------------------- | ||||
def class_of ( object ): | ||||
""" Returns a string containing the class name of an object with the | ||||
correct indefinite article ('a' or 'an') preceding it (e.g., 'an Image', | ||||
'a PlotValue'). | ||||
""" | ||||
Thomas Kluyver
|
r13353 | if isinstance( object, py3compat.string_types ): | ||
Brian Granger
|
r2157 | return add_article( object ) | ||
return add_article( object.__class__.__name__ ) | ||||
def add_article ( name ): | ||||
""" Returns a string containing the correct indefinite article ('a' or 'an') | ||||
prefixed to the specified string. | ||||
""" | ||||
if name[:1].lower() in 'aeiou': | ||||
return 'an ' + name | ||||
return 'a ' + name | ||||
def repr_type(obj): | ||||
""" Return a string representation of a value and its type for readable | ||||
error messages. | ||||
""" | ||||
the_type = type(obj) | ||||
Thomas Kluyver
|
r4732 | if (not py3compat.PY3) and the_type is InstanceType: | ||
Brian Granger
|
r2157 | # Old-style class. | ||
the_type = obj.__class__ | ||||
msg = '%r %r' % (obj, the_type) | ||||
return msg | ||||
epatters
|
r8407 | def is_trait(t): | ||
""" Returns whether the given value is an instance or subclass of TraitType. | ||||
""" | ||||
return (isinstance(t, TraitType) or | ||||
(isinstance(t, type) and issubclass(t, TraitType))) | ||||
Brian Granger
|
r2157 | def parse_notifier_name(name): | ||
Brian Granger
|
r2175 | """Convert the name argument to a list of names. | ||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2175 | Examples | ||
-------- | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2175 | >>> parse_notifier_name('a') | ||
['a'] | ||||
>>> parse_notifier_name(['a','b']) | ||||
['a', 'b'] | ||||
>>> parse_notifier_name(None) | ||||
Dav Clark
|
r2384 | ['anytrait'] | ||
Brian Granger
|
r2175 | """ | ||
Brian Granger
|
r2157 | if isinstance(name, str): | ||
return [name] | ||||
elif name is None: | ||||
Dav Clark
|
r2384 | return ['anytrait'] | ||
Brian Granger
|
r2157 | elif isinstance(name, (list, tuple)): | ||
for n in name: | ||||
assert isinstance(n, str), "names must be strings" | ||||
return name | ||||
Brian Granger
|
r2184 | |||
class _SimpleTest: | ||||
def __init__ ( self, value ): self.value = value | ||||
def __call__ ( self, test ): | ||||
return test == self.value | ||||
def __repr__(self): | ||||
return "<SimpleTest(%r)" % self.value | ||||
def __str__(self): | ||||
return self.__repr__() | ||||
Brian Granger
|
r2287 | def getmembers(object, predicate=None): | ||
"""A safe version of inspect.getmembers that handles missing attributes. | ||||
Bernardo B. Marques
|
r4872 | This is useful when there are descriptor based attributes that for | ||
Brian Granger
|
r2287 | some reason raise AttributeError even though they exist. This happens | ||
in zope.inteface with the __provides__ attribute. | ||||
""" | ||||
results = [] | ||||
for key in dir(object): | ||||
try: | ||||
value = getattr(object, key) | ||||
except AttributeError: | ||||
pass | ||||
else: | ||||
if not predicate or predicate(value): | ||||
results.append((key, value)) | ||||
results.sort() | ||||
return results | ||||
Jonathan Frederic
|
r15011 | @skip_doctest | ||
Jason Grout
|
r15163 | class link(object): | ||
"""Link traits from different objects together so they remain in sync. | ||||
Jason Grout
|
r15008 | |||
Parameters | ||||
---------- | ||||
obj : pairs of objects/attributes | ||||
Examples | ||||
-------- | ||||
Jason Grout
|
r15163 | >>> c = link((obj1, 'value'), (obj2, 'value'), (obj3, 'value')) | ||
Jason Grout
|
r15008 | >>> obj1.value = 5 # updates other objects as well | ||
""" | ||||
updating = False | ||||
def __init__(self, *args): | ||||
Jonathan Frederic
|
r15013 | if len(args) < 2: | ||
raise TypeError('At least two traitlets must be provided.') | ||||
Jonathan Frederic
|
r15018 | self.objects = {} | ||
initial = getattr(args[0][0], args[0][1]) | ||||
Jason Grout
|
r15008 | for obj,attr in args: | ||
Jonathan Frederic
|
r15018 | if getattr(obj, attr) != initial: | ||
setattr(obj, attr, initial) | ||||
Jason Grout
|
r15008 | |||
Jonathan Frederic
|
r15018 | callback = self._make_closure(obj,attr) | ||
obj.on_trait_change(callback, attr) | ||||
self.objects[(obj,attr)] = callback | ||||
Jonathan Frederic
|
r15013 | |||
Jason Grout
|
r15008 | @contextlib.contextmanager | ||
def _busy_updating(self): | ||||
self.updating = True | ||||
Jonathan Frederic
|
r15009 | try: | ||
yield | ||||
finally: | ||||
self.updating = False | ||||
Jason Grout
|
r15008 | |||
Jonathan Frederic
|
r15018 | def _make_closure(self, sending_obj, sending_attr): | ||
def update(name, old, new): | ||||
self._update(sending_obj, sending_attr, new) | ||||
return update | ||||
def _update(self, sending_obj, sending_attr, new): | ||||
Jason Grout
|
r15008 | if self.updating: | ||
return | ||||
with self._busy_updating(): | ||||
Jonathan Frederic
|
r15018 | for obj,attr in self.objects.keys(): | ||
if obj is not sending_obj or attr != sending_attr: | ||||
setattr(obj, attr, new) | ||||
zah
|
r16453 | |||
Jason Grout
|
r15163 | def unlink(self): | ||
Jonathan Frederic
|
r15018 | for key, callback in self.objects.items(): | ||
(obj,attr) = key | ||||
obj.on_trait_change(callback, attr, remove=True) | ||||
Brian Granger
|
r2287 | |||
Brian Granger
|
r2157 | #----------------------------------------------------------------------------- | ||
Dav Clark
|
r2385 | # Base TraitType for all traits | ||
Brian Granger
|
r2157 | #----------------------------------------------------------------------------- | ||
Dav Clark
|
r2385 | class TraitType(object): | ||
"""A base class for all trait descriptors. | ||||
Brian Granger
|
r2183 | |||
Notes | ||||
----- | ||||
Dav Clark
|
r2385 | Our implementation of traits is based on Python's descriptor | ||
Brian Granger
|
r2183 | prototol. This class is the base class for all such descriptors. The | ||
Dav Clark
|
r2384 | only magic we use is a custom metaclass for the main :class:`HasTraits` | ||
Brian Granger
|
r2183 | class that does the following: | ||
Dav Clark
|
r2385 | 1. Sets the :attr:`name` attribute of every :class:`TraitType` | ||
Brian Granger
|
r2183 | instance in the class dict to the name of the attribute. | ||
Dav Clark
|
r2385 | 2. Sets the :attr:`this_class` attribute of every :class:`TraitType` | ||
instance in the class dict to the *class* that declared the trait. | ||||
This is used by the :class:`This` trait to allow subclasses to | ||||
Brian Granger
|
r2183 | accept superclasses for :class:`This` values. | ||
""" | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2157 | |||
metadata = {} | ||||
Brian Granger
|
r2177 | default_value = Undefined | ||
zah
|
r16453 | allow_none = False | ||
Brian Granger
|
r2157 | info_text = 'any value' | ||
zah
|
r16453 | def __init__(self, default_value=NoDefaultSpecified, allow_none=None, **metadata): | ||
Dav Clark
|
r2385 | """Create a TraitType. | ||
Brian Granger
|
r2177 | """ | ||
Brian Granger
|
r2157 | if default_value is not NoDefaultSpecified: | ||
self.default_value = default_value | ||||
zah
|
r16453 | if allow_none is not None: | ||
self.allow_none = allow_none | ||||
Brian Granger
|
r2184 | |||
if len(metadata) > 0: | ||||
if len(self.metadata) > 0: | ||||
self._metadata = self.metadata.copy() | ||||
self._metadata.update(metadata) | ||||
else: | ||||
self._metadata = metadata | ||||
else: | ||||
self._metadata = self.metadata | ||||
Brian Granger
|
r2177 | self.init() | ||
Brian Granger
|
r2157 | |||
Brian Granger
|
r2177 | def init(self): | ||
pass | ||||
def get_default_value(self): | ||||
"""Create a new instance of the default value.""" | ||||
Brian Granger
|
r2742 | return self.default_value | ||
Brian Granger
|
r2177 | |||
Brian Granger
|
r2229 | def instance_init(self, obj): | ||
Dav Clark
|
r2384 | """This is called by :meth:`HasTraits.__new__` to finish init'ing. | ||
Brian Granger
|
r2229 | |||
Some stages of initialization must be delayed until the parent | ||||
Dav Clark
|
r2384 | :class:`HasTraits` instance has been created. This method is | ||
called in :meth:`HasTraits.__new__` after the instance has been | ||||
Brian Granger
|
r2229 | created. | ||
This method trigger the creation and validation of default values | ||||
Bernardo B. Marques
|
r4872 | and also things like the resolution of str given class names in | ||
Brian Granger
|
r2229 | :class:`Type` and :class`Instance`. | ||
Parameters | ||||
---------- | ||||
Dav Clark
|
r2384 | obj : :class:`HasTraits` instance | ||
The parent :class:`HasTraits` instance that has just been | ||||
Brian Granger
|
r2229 | created. | ||
""" | ||||
self.set_default_value(obj) | ||||
Brian Granger
|
r2182 | def set_default_value(self, obj): | ||
Brian Granger
|
r2229 | """Set the default value on a per instance basis. | ||
This method is called by :meth:`instance_init` to create and | ||||
Bernardo B. Marques
|
r4872 | validate the default value. The creation and validation of | ||
Dav Clark
|
r2384 | default values must be delayed until the parent :class:`HasTraits` | ||
Brian Granger
|
r2229 | class has been instantiated. | ||
""" | ||||
Robert Kern
|
r3207 | # Check for a deferred initializer defined in the same class as the | ||
# trait declaration or above. | ||||
mro = type(obj).mro() | ||||
meth_name = '_%s_default' % self.name | ||||
for cls in mro[:mro.index(self.this_class)+1]: | ||||
if meth_name in cls.__dict__: | ||||
break | ||||
else: | ||||
# We didn't find one. Do static initialization. | ||||
dv = self.get_default_value() | ||||
newdv = self._validate(obj, dv) | ||||
obj._trait_values[self.name] = newdv | ||||
return | ||||
# Complete the dynamic initialization. | ||||
Min RK
|
r16216 | obj._trait_dyn_inits[self.name] = meth_name | ||
Brian Granger
|
r2182 | def __get__(self, obj, cls=None): | ||
Dav Clark
|
r2385 | """Get the value of the trait by self.name for the instance. | ||
Brian Granger
|
r2180 | |||
Dav Clark
|
r2384 | Default values are instantiated when :meth:`HasTraits.__new__` | ||
Bernardo B. Marques
|
r4872 | is called. Thus by the time this method gets called either the | ||
Brian Granger
|
r2182 | default value or a user defined value (they called :meth:`__set__`) | ||
Dav Clark
|
r2384 | is in the :class:`HasTraits` instance. | ||
Brian Granger
|
r2177 | """ | ||
if obj is None: | ||||
Brian Granger
|
r2157 | return self | ||
else: | ||||
Brian Granger
|
r2182 | try: | ||
Dav Clark
|
r2385 | value = obj._trait_values[self.name] | ||
Robert Kern
|
r3207 | except KeyError: | ||
# Check for a dynamic initializer. | ||||
Robert Kern
|
r3336 | if self.name in obj._trait_dyn_inits: | ||
Min RK
|
r16216 | method = getattr(obj, obj._trait_dyn_inits[self.name]) | ||
value = method() | ||||
Robert Kern
|
r3207 | # FIXME: Do we really validate here? | ||
value = self._validate(obj, value) | ||||
obj._trait_values[self.name] = value | ||||
return value | ||||
else: | ||||
raise TraitError('Unexpected error in TraitType: ' | ||||
'both default value and dynamic initializer are ' | ||||
'absent.') | ||||
except Exception: | ||||
Dav Clark
|
r2384 | # HasTraits should call set_default_value to populate | ||
Brian Granger
|
r2182 | # this. So this should never be reached. | ||
Dav Clark
|
r2385 | raise TraitError('Unexpected error in TraitType: ' | ||
Brian Granger
|
r2182 | 'default value not set properly') | ||
Brian Granger
|
r2177 | else: | ||
Brian Granger
|
r2182 | return value | ||
Brian Granger
|
r2177 | |||
Brian Granger
|
r2182 | def __set__(self, obj, value): | ||
Brian Granger
|
r2177 | new_value = self._validate(obj, value) | ||
Brian Granger
|
r2182 | old_value = self.__get__(obj) | ||
MinRK
|
r8230 | obj._trait_values[self.name] = new_value | ||
Jason Grout
|
r15436 | try: | ||
Jason Grout
|
r15462 | silent = bool(old_value == new_value) | ||
Jason Grout
|
r15436 | except: | ||
# if there is an error in comparing, default to notify | ||||
Jason Grout
|
r15462 | silent = False | ||
if silent is not True: | ||||
# we explicitly compare silent to True just in case the equality | ||||
# comparison above returns something other than True/False | ||||
Dav Clark
|
r2385 | obj._notify_trait(self.name, old_value, new_value) | ||
Brian Granger
|
r2157 | |||
Brian Granger
|
r2177 | def _validate(self, obj, value): | ||
zah
|
r16453 | if value is None and self.allow_none: | ||
return value | ||||
Brian Granger
|
r2157 | if hasattr(self, 'validate'): | ||
Brian Granger
|
r2177 | return self.validate(obj, value) | ||
Brian Granger
|
r2157 | elif hasattr(self, 'is_valid_for'): | ||
valid = self.is_valid_for(value) | ||||
if valid: | ||||
return value | ||||
else: | ||||
Dav Clark
|
r2385 | raise TraitError('invalid value for type: %r' % value) | ||
Brian Granger
|
r2157 | elif hasattr(self, 'value_for'): | ||
return self.value_for(value) | ||||
else: | ||||
return value | ||||
def info(self): | ||||
return self.info_text | ||||
def error(self, obj, value): | ||||
if obj is not None: | ||||
Dav Clark
|
r2385 | e = "The '%s' trait of %s instance must be %s, but a value of %s was specified." \ | ||
Brian Granger
|
r2157 | % (self.name, class_of(obj), | ||
self.info(), repr_type(value)) | ||||
else: | ||||
Dav Clark
|
r2385 | e = "The '%s' trait must be %s, but a value of %r was specified." \ | ||
MinRK
|
r3870 | % (self.name, self.info(), repr_type(value)) | ||
Dav Clark
|
r2385 | raise TraitError(e) | ||
Brian Granger
|
r2157 | |||
Brian Granger
|
r2184 | def get_metadata(self, key): | ||
return getattr(self, '_metadata', {}).get(key, None) | ||||
def set_metadata(self, key, value): | ||||
getattr(self, '_metadata', {})[key] = value | ||||
Brian Granger
|
r2157 | |||
#----------------------------------------------------------------------------- | ||||
Dav Clark
|
r2384 | # The HasTraits implementation | ||
Brian Granger
|
r2157 | #----------------------------------------------------------------------------- | ||
Dav Clark
|
r2384 | class MetaHasTraits(type): | ||
"""A metaclass for HasTraits. | ||||
Bernardo B. Marques
|
r4872 | |||
Dav Clark
|
r2385 | This metaclass makes sure that any TraitType class attributes are | ||
Brian Granger
|
r2157 | instantiated and sets their name attribute. | ||
""" | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2157 | def __new__(mcls, name, bases, classdict): | ||
Dav Clark
|
r2384 | """Create the HasTraits class. | ||
Bernardo B. Marques
|
r4872 | |||
Dav Clark
|
r2385 | This instantiates all TraitTypes in the class dict and sets their | ||
Brian Granger
|
r2183 | :attr:`name` attribute. | ||
""" | ||||
Brian Granger
|
r2287 | # print "MetaHasTraitlets (mcls, name): ", mcls, name | ||
# print "MetaHasTraitlets (bases): ", bases | ||||
# print "MetaHasTraitlets (classdict): ", classdict | ||||
Thomas Kluyver
|
r13361 | for k,v in iteritems(classdict): | ||
Dav Clark
|
r2385 | if isinstance(v, TraitType): | ||
Brian Granger
|
r2157 | v.name = k | ||
elif inspect.isclass(v): | ||||
Dav Clark
|
r2385 | if issubclass(v, TraitType): | ||
Brian Granger
|
r2157 | vinst = v() | ||
vinst.name = k | ||||
classdict[k] = vinst | ||||
Dav Clark
|
r2384 | return super(MetaHasTraits, mcls).__new__(mcls, name, bases, classdict) | ||
Brian Granger
|
r2157 | |||
Brian Granger
|
r2183 | def __init__(cls, name, bases, classdict): | ||
Dav Clark
|
r2384 | """Finish initializing the HasTraits class. | ||
Bernardo B. Marques
|
r4872 | |||
Dav Clark
|
r2385 | This sets the :attr:`this_class` attribute of each TraitType in the | ||
Brian Granger
|
r2183 | class dict to the newly created class ``cls``. | ||
""" | ||||
Thomas Kluyver
|
r13361 | for k, v in iteritems(classdict): | ||
Dav Clark
|
r2385 | if isinstance(v, TraitType): | ||
Brian Granger
|
r2183 | v.this_class = cls | ||
Dav Clark
|
r2384 | super(MetaHasTraits, cls).__init__(name, bases, classdict) | ||
Brian Granger
|
r2157 | |||
Thomas Kluyver
|
r13359 | class HasTraits(py3compat.with_metaclass(MetaHasTraits, object)): | ||
Brian Granger
|
r2157 | |||
MinRK
|
r10257 | def __new__(cls, *args, **kw): | ||
Matthias BUSSONNIER
|
r16320 | # This is needed because object.__new__ only accepts | ||
Brian Granger
|
r2255 | # the cls argument. | ||
Dav Clark
|
r2384 | new_meth = super(HasTraits, cls).__new__ | ||
Brian Granger
|
r2255 | if new_meth is object.__new__: | ||
inst = new_meth(cls) | ||||
else: | ||||
Brian Granger
|
r2739 | inst = new_meth(cls, **kw) | ||
Dav Clark
|
r2385 | inst._trait_values = {} | ||
inst._trait_notifiers = {} | ||||
Robert Kern
|
r3336 | inst._trait_dyn_inits = {} | ||
Dav Clark
|
r2385 | # Here we tell all the TraitType instances to set their default | ||
Bernardo B. Marques
|
r4872 | # values on the instance. | ||
Brian Granger
|
r2182 | for key in dir(cls): | ||
Brian Granger
|
r2287 | # Some descriptors raise AttributeError like zope.interface's | ||
# __provides__ attributes even though they exist. This causes | ||||
# AttributeErrors even though they are listed in dir(cls). | ||||
try: | ||||
value = getattr(cls, key) | ||||
except AttributeError: | ||||
pass | ||||
else: | ||||
Dav Clark
|
r2469 | if isinstance(value, TraitType): | ||
Brian Granger
|
r2287 | value.instance_init(inst) | ||
Dav Clark
|
r2469 | |||
Brian Granger
|
r2182 | return inst | ||
MinRK
|
r10257 | def __init__(self, *args, **kw): | ||
Brian Granger
|
r2740 | # Allow trait values to be set using keyword arguments. | ||
Brian Granger
|
r2745 | # We need to use setattr for this to trigger validation and | ||
# notifications. | ||||
Thomas Kluyver
|
r13361 | for key, value in iteritems(kw): | ||
Brian Granger
|
r2739 | setattr(self, key, value) | ||
Brian Granger
|
r2157 | |||
Dav Clark
|
r2385 | def _notify_trait(self, name, old_value, new_value): | ||
Brian Granger
|
r2175 | |||
# First dynamic ones | ||||
Jonathan Frederic
|
r13148 | callables = [] | ||
callables.extend(self._trait_notifiers.get(name,[])) | ||||
callables.extend(self._trait_notifiers.get('anytrait',[])) | ||||
Brian Granger
|
r2175 | |||
# Now static ones | ||||
try: | ||||
cb = getattr(self, '_%s_changed' % name) | ||||
except: | ||||
pass | ||||
else: | ||||
callables.append(cb) | ||||
# Call them all now | ||||
Brian Granger
|
r2157 | for c in callables: | ||
# Traits catches and logs errors here. I allow them to raise | ||||
Brian Granger
|
r2175 | if callable(c): | ||
argspec = inspect.getargspec(c) | ||||
nargs = len(argspec[0]) | ||||
# Bound methods have an additional 'self' argument | ||||
# I don't know how to treat unbound methods, but they | ||||
# can't really be used for callbacks. | ||||
if isinstance(c, types.MethodType): | ||||
offset = -1 | ||||
else: | ||||
offset = 0 | ||||
if nargs + offset == 0: | ||||
c() | ||||
elif nargs + offset == 1: | ||||
c(name) | ||||
elif nargs + offset == 2: | ||||
c(name, new_value) | ||||
elif nargs + offset == 3: | ||||
c(name, old_value, new_value) | ||||
else: | ||||
Dav Clark
|
r2385 | raise TraitError('a trait changed callback ' | ||
Brian Granger
|
r2175 | 'must have 0-3 arguments.') | ||
else: | ||||
Dav Clark
|
r2385 | raise TraitError('a trait changed callback ' | ||
Brian Granger
|
r2175 | 'must be callable.') | ||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2157 | |||
def _add_notifiers(self, handler, name): | ||||
Bradley M. Froehle
|
r7859 | if name not in self._trait_notifiers: | ||
Brian Granger
|
r2157 | nlist = [] | ||
Dav Clark
|
r2385 | self._trait_notifiers[name] = nlist | ||
Brian Granger
|
r2157 | else: | ||
Dav Clark
|
r2385 | nlist = self._trait_notifiers[name] | ||
Brian Granger
|
r2157 | if handler not in nlist: | ||
nlist.append(handler) | ||||
def _remove_notifiers(self, handler, name): | ||||
Bradley M. Froehle
|
r7859 | if name in self._trait_notifiers: | ||
Dav Clark
|
r2385 | nlist = self._trait_notifiers[name] | ||
Brian Granger
|
r2157 | try: | ||
index = nlist.index(handler) | ||||
except ValueError: | ||||
pass | ||||
else: | ||||
del nlist[index] | ||||
Dav Clark
|
r2384 | def on_trait_change(self, handler, name=None, remove=False): | ||
Dav Clark
|
r2385 | """Setup a handler to be called when a trait changes. | ||
Brian Granger
|
r2175 | |||
Dav Clark
|
r2385 | This is used to setup dynamic notifications of trait changes. | ||
Bernardo B. Marques
|
r4872 | |||
Dav Clark
|
r2384 | Static handlers can be created by creating methods on a HasTraits | ||
Dav Clark
|
r2385 | subclass with the naming convention '_[traitname]_changed'. Thus, | ||
to create static handler for the trait 'a', create the method | ||||
Brian Granger
|
r2175 | _a_changed(self, name, old, new) (fewer arguments can be used, see | ||
below). | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2175 | Parameters | ||
---------- | ||||
Brian Granger
|
r2277 | handler : callable | ||
Bernardo B. Marques
|
r4872 | A callable that is called when a trait changes. Its | ||
Brian Granger
|
r2277 | signature can be handler(), handler(name), handler(name, new) | ||
or handler(name, old, new). | ||||
name : list, str, None | ||||
Dav Clark
|
r2385 | If None, the handler will apply to all traits. If a list | ||
Brian Granger
|
r2277 | of str, handler will apply to all names in the list. If a | ||
str, the handler will apply just to that name. | ||||
remove : bool | ||||
If False (the default), then install the handler. If True | ||||
then unintall it. | ||||
Brian Granger
|
r2175 | """ | ||
Brian Granger
|
r2157 | if remove: | ||
names = parse_notifier_name(name) | ||||
for n in names: | ||||
self._remove_notifiers(handler, n) | ||||
else: | ||||
names = parse_notifier_name(name) | ||||
for n in names: | ||||
self._add_notifiers(handler, n) | ||||
Brian Granger
|
r3789 | @classmethod | ||
def class_trait_names(cls, **metadata): | ||||
Thomas A Caswell
|
r15487 | """Get a list of all the names of this class' traits. | ||
Brian Granger
|
r3789 | |||
Thomas A Caswell
|
r15487 | This method is just like the :meth:`trait_names` method, | ||
but is unbound. | ||||
Brian Granger
|
r3789 | """ | ||
return cls.class_traits(**metadata).keys() | ||||
@classmethod | ||||
def class_traits(cls, **metadata): | ||||
Thomas A Caswell
|
r15487 | """Get a `dict` of all the traits of this class. The dictionary | ||
is keyed on the name and the values are the TraitType objects. | ||||
Brian Granger
|
r3789 | |||
This method is just like the :meth:`traits` method, but is unbound. | ||||
The TraitTypes returned don't know anything about the values | ||||
that the various HasTrait's instances are holding. | ||||
Thomas A Caswell
|
r15487 | The metadata kwargs allow functions to be passed in which | ||
filter traits based on metadata values. The functions should | ||||
take a single value as an argument and return a boolean. If | ||||
any function returns False, then the trait is not included in | ||||
the output. This does not allow for any simple way of | ||||
testing that a metadata name exists and has any | ||||
value because get_metadata returns None if a metadata key | ||||
doesn't exist. | ||||
Brian Granger
|
r3789 | """ | ||
Thomas A Caswell
|
r15487 | traits = dict([memb for memb in getmembers(cls) if | ||
Brian Granger
|
r3789 | isinstance(memb[1], TraitType)]) | ||
if len(metadata) == 0: | ||||
return traits | ||||
for meta_name, meta_eval in metadata.items(): | ||||
if type(meta_eval) is not FunctionType: | ||||
metadata[meta_name] = _SimpleTest(meta_eval) | ||||
result = {} | ||||
for name, trait in traits.items(): | ||||
for meta_name, meta_eval in metadata.items(): | ||||
if not meta_eval(trait.get_metadata(meta_name)): | ||||
break | ||||
else: | ||||
result[name] = trait | ||||
return result | ||||
Dav Clark
|
r2384 | def trait_names(self, **metadata): | ||
Thomas A Caswell
|
r15487 | """Get a list of all the names of this class' traits.""" | ||
Dav Clark
|
r2384 | return self.traits(**metadata).keys() | ||
Brian Granger
|
r2184 | |||
Dav Clark
|
r2384 | def traits(self, **metadata): | ||
Thomas A Caswell
|
r15487 | """Get a `dict` of all the traits of this class. The dictionary | ||
is keyed on the name and the values are the TraitType objects. | ||||
Brian Granger
|
r2184 | |||
Dav Clark
|
r2385 | The TraitTypes returned don't know anything about the values | ||
that the various HasTrait's instances are holding. | ||||
Brian Granger
|
r2245 | |||
Thomas A Caswell
|
r15487 | The metadata kwargs allow functions to be passed in which | ||
filter traits based on metadata values. The functions should | ||||
take a single value as an argument and return a boolean. If | ||||
any function returns False, then the trait is not included in | ||||
the output. This does not allow for any simple way of | ||||
testing that a metadata name exists and has any | ||||
value because get_metadata returns None if a metadata key | ||||
doesn't exist. | ||||
Brian Granger
|
r2184 | """ | ||
Thomas A Caswell
|
r15487 | traits = dict([memb for memb in getmembers(self.__class__) if | ||
Dav Clark
|
r2385 | isinstance(memb[1], TraitType)]) | ||
Brian Granger
|
r2184 | |||
Brian Granger
|
r2245 | if len(metadata) == 0: | ||
Dav Clark
|
r2385 | return traits | ||
Brian Granger
|
r2200 | |||
Brian Granger
|
r2184 | for meta_name, meta_eval in metadata.items(): | ||
if type(meta_eval) is not FunctionType: | ||||
metadata[meta_name] = _SimpleTest(meta_eval) | ||||
result = {} | ||||
Dav Clark
|
r2385 | for name, trait in traits.items(): | ||
Brian Granger
|
r2184 | for meta_name, meta_eval in metadata.items(): | ||
Dav Clark
|
r2385 | if not meta_eval(trait.get_metadata(meta_name)): | ||
Brian Granger
|
r2184 | break | ||
else: | ||||
Dav Clark
|
r2385 | result[name] = trait | ||
Brian Granger
|
r2184 | |||
return result | ||||
Brian Granger
|
r2179 | |||
Dav Clark
|
r2385 | def trait_metadata(self, traitname, key): | ||
"""Get metadata values for trait by key.""" | ||||
Brian Granger
|
r2184 | try: | ||
Dav Clark
|
r2385 | trait = getattr(self.__class__, traitname) | ||
Brian Granger
|
r2184 | except AttributeError: | ||
Dav Clark
|
r2385 | raise TraitError("Class %s does not have a trait named %s" % | ||
(self.__class__.__name__, traitname)) | ||||
Brian Granger
|
r2184 | else: | ||
Dav Clark
|
r2385 | return trait.get_metadata(key) | ||
Brian Granger
|
r2177 | |||
Brian Granger
|
r2157 | #----------------------------------------------------------------------------- | ||
Dav Clark
|
r2385 | # Actual TraitTypes implementations/subclasses | ||
Brian Granger
|
r2157 | #----------------------------------------------------------------------------- | ||
Brian Granger
|
r2177 | |||
#----------------------------------------------------------------------------- | ||||
Dav Clark
|
r2385 | # TraitTypes subclasses for handling classes and instances of classes | ||
Brian Granger
|
r2177 | #----------------------------------------------------------------------------- | ||
zah
|
r16453 | class ClassBasedTraitType(TraitType): | ||
Dav Clark
|
r2385 | """A trait with error reporting for Type, Instance and This.""" | ||
Brian Granger
|
r2177 | |||
Brian Granger
|
r2182 | def error(self, obj, value): | ||
Brian Granger
|
r2177 | kind = type(value) | ||
Thomas Kluyver
|
r4732 | if (not py3compat.PY3) and kind is InstanceType: | ||
Brian Granger
|
r2177 | msg = 'class %s' % value.__class__.__name__ | ||
else: | ||||
msg = '%s (i.e. %s)' % ( str( kind )[1:-1], repr( value ) ) | ||||
MinRK
|
r3870 | if obj is not None: | ||
e = "The '%s' trait of %s instance must be %s, but a value of %s was specified." \ | ||||
% (self.name, class_of(obj), | ||||
self.info(), msg) | ||||
else: | ||||
e = "The '%s' trait must be %s, but a value of %r was specified." \ | ||||
% (self.name, self.info(), msg) | ||||
raise TraitError(e) | ||||
Brian Granger
|
r2177 | |||
Dav Clark
|
r2385 | class Type(ClassBasedTraitType): | ||
"""A trait whose value must be a subclass of a specified class.""" | ||||
Brian Granger
|
r2177 | |||
def __init__ (self, default_value=None, klass=None, allow_none=True, **metadata ): | ||||
Dav Clark
|
r2385 | """Construct a Type trait | ||
Brian Granger
|
r2177 | |||
Dav Clark
|
r2385 | A Type trait specifies that its values must be subclasses of | ||
Brian Granger
|
r2177 | a particular class. | ||
Brian Granger
|
r2229 | If only ``default_value`` is given, it is used for the ``klass`` as | ||
well. | ||||
Brian Granger
|
r2177 | Parameters | ||
---------- | ||||
Brian Granger
|
r2229 | default_value : class, str or None | ||
The default value must be a subclass of klass. If an str, | ||||
the str must be a fully specified class name, like 'foo.bar.Bah'. | ||||
Bernardo B. Marques
|
r4872 | The string is resolved into real class, when the parent | ||
Dav Clark
|
r2384 | :class:`HasTraits` class is instantiated. | ||
Brian Granger
|
r2177 | klass : class, str, None | ||
Dav Clark
|
r2385 | Values of this trait must be a subclass of klass. The klass | ||
Brian Granger
|
r2177 | may be specified in a string like: 'foo.bar.MyClass'. | ||
Bernardo B. Marques
|
r4872 | The string is resolved into real class, when the parent | ||
Dav Clark
|
r2384 | :class:`HasTraits` class is instantiated. | ||
Brian Granger
|
r2177 | allow_none : boolean | ||
Indicates whether None is allowed as an assignable value. Even if | ||||
``False``, the default value may be ``None``. | ||||
""" | ||||
if default_value is None: | ||||
if klass is None: | ||||
klass = object | ||||
elif klass is None: | ||||
klass = default_value | ||||
Thomas Kluyver
|
r13353 | if not (inspect.isclass(klass) or isinstance(klass, py3compat.string_types)): | ||
Dav Clark
|
r2385 | raise TraitError("A Type trait must specify a class.") | ||
Brian Granger
|
r2177 | |||
self.klass = klass | ||||
zah
|
r16544 | super(Type, self).__init__(default_value, allow_none=allow_none, **metadata) | ||
Brian Granger
|
r2177 | |||
def validate(self, obj, value): | ||||
"""Validates that the value is a valid object instance.""" | ||||
Thomas Kluyver
|
r16961 | if isinstance(value, py3compat.string_types): | ||
try: | ||||
value = import_item(value) | ||||
except ImportError: | ||||
raise TraitError("The '%s' trait of %s instance must be a type, but " | ||||
"%r could not be imported" % (self.name, obj, value)) | ||||
Brian Granger
|
r2177 | try: | ||
if issubclass(value, self.klass): | ||||
return value | ||||
except: | ||||
zah
|
r16453 | pass | ||
Brian Granger
|
r2177 | |||
self.error(obj, value) | ||||
def info(self): | ||||
""" Returns a description of the trait.""" | ||||
Thomas Kluyver
|
r13353 | if isinstance(self.klass, py3compat.string_types): | ||
Brian Granger
|
r2229 | klass = self.klass | ||
else: | ||||
klass = self.klass.__name__ | ||||
Brian Granger
|
r2177 | result = 'a subclass of ' + klass | ||
zah
|
r16453 | if self.allow_none: | ||
Brian Granger
|
r2177 | return result + ' or None' | ||
return result | ||||
Brian Granger
|
r2229 | def instance_init(self, obj): | ||
self._resolve_classes() | ||||
super(Type, self).instance_init(obj) | ||||
def _resolve_classes(self): | ||||
Thomas Kluyver
|
r13353 | if isinstance(self.klass, py3compat.string_types): | ||
Brian Granger
|
r2229 | self.klass = import_item(self.klass) | ||
Thomas Kluyver
|
r13353 | if isinstance(self.default_value, py3compat.string_types): | ||
Brian Granger
|
r2229 | self.default_value = import_item(self.default_value) | ||
def get_default_value(self): | ||||
return self.default_value | ||||
Brian Granger
|
r2177 | |||
class DefaultValueGenerator(object): | ||||
"""A class for generating new default value instances.""" | ||||
Brian Granger
|
r2229 | def __init__(self, *args, **kw): | ||
Brian Granger
|
r2177 | self.args = args | ||
self.kw = kw | ||||
Brian Granger
|
r2229 | def generate(self, klass): | ||
return klass(*self.args, **self.kw) | ||||
Brian Granger
|
r2177 | |||
Brian Granger
|
r2182 | |||
Dav Clark
|
r2385 | class Instance(ClassBasedTraitType): | ||
Brian Granger
|
r2177 | """A trait whose value must be an instance of a specified class. | ||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2177 | The value can also be an instance of a subclass of the specified class. | ||
Jason Grout
|
r17238 | |||
Subclasses can declare default classes by overriding the klass attribute | ||||
Brian Granger
|
r2177 | """ | ||
Jason Grout
|
r17238 | klass = None | ||
Bernardo B. Marques
|
r4872 | def __init__(self, klass=None, args=None, kw=None, | ||
Brian Granger
|
r2182 | allow_none=True, **metadata ): | ||
Dav Clark
|
r2385 | """Construct an Instance trait. | ||
Brian Granger
|
r2177 | |||
Dav Clark
|
r2385 | This trait allows values that are instances of a particular | ||
Brian Granger
|
r2182 | class or its sublclasses. Our implementation is quite different | ||
from that of enthough.traits as we don't allow instances to be used | ||||
for klass and we handle the ``args`` and ``kw`` arguments differently. | ||||
Brian Granger
|
r2177 | Parameters | ||
---------- | ||||
Brian Granger
|
r2229 | klass : class, str | ||
Dav Clark
|
r2385 | The class that forms the basis for the trait. Class names | ||
Brian Granger
|
r2229 | can also be specified as strings, like 'foo.bar.Bar'. | ||
Brian Granger
|
r2177 | args : tuple | ||
Positional arguments for generating the default value. | ||||
kw : dict | ||||
Keyword arguments for generating the default value. | ||||
allow_none : bool | ||||
Indicates whether None is allowed as a value. | ||||
Thomas Kluyver
|
r13587 | Notes | ||
----- | ||||
Brian Granger
|
r2182 | If both ``args`` and ``kw`` are None, then the default value is None. | ||
If ``args`` is a tuple and ``kw`` is a dict, then the default is | ||||
Jason Grout
|
r17238 | created as ``klass(*args, **kw)``. If exactly one of ``args`` or ``kw`` is | ||
None, the None is replaced by ``()`` or ``{}``, respectively. | ||||
Brian Granger
|
r2177 | """ | ||
Jason Grout
|
r17238 | if klass is None: | ||
klass = self.klass | ||||
if (klass is not None) and (inspect.isclass(klass) or isinstance(klass, py3compat.string_types)): | ||||
self.klass = klass | ||||
else: | ||||
raise TraitError('The klass attribute must be a class' | ||||
' not: %r' % klass) | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2182 | # self.klass is a class, so handle default_value | ||
if args is None and kw is None: | ||||
Brian Granger
|
r2177 | default_value = None | ||
else: | ||||
if args is None: | ||||
Brian Granger
|
r2182 | # kw is not None | ||
Brian Granger
|
r2177 | args = () | ||
Brian Granger
|
r2182 | elif kw is None: | ||
# args is not None | ||||
kw = {} | ||||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2177 | if not isinstance(kw, dict): | ||
Dav Clark
|
r2385 | raise TraitError("The 'kw' argument must be a dict or None.") | ||
Brian Granger
|
r2177 | if not isinstance(args, tuple): | ||
Dav Clark
|
r2385 | raise TraitError("The 'args' argument must be a tuple or None.") | ||
Bernardo B. Marques
|
r4872 | |||
Brian Granger
|
r2229 | default_value = DefaultValueGenerator(*args, **kw) | ||
Brian Granger
|
r2177 | |||
zah
|
r16544 | super(Instance, self).__init__(default_value, allow_none=allow_none, **metadata) | ||
Brian Granger
|
r2177 | |||
def validate(self, obj, value): | ||||
if isinstance(value, self.klass): | ||||
return value | ||||
else: | ||||
Brian Granger
|
r2182 | self.error(obj, value) | ||
Brian Granger
|
r2177 | |||
Brian Granger
|
r2182 | def info(self): | ||
Thomas Kluyver
|
r13353 | if isinstance(self.klass, py3compat.string_types): | ||
Brian Granger
|
r2229 | klass = self.klass | ||
else: | ||||
klass = self.klass.__name__ | ||||
Brian Granger
|
r2177 | result = class_of(klass) | ||
zah
|
r16453 | if self.allow_none: | ||
Brian Granger
|
r2177 | return result + ' or None' | ||
return result | ||||
Brian Granger
|
r2229 | def instance_init(self, obj): | ||
self._resolve_classes() | ||||
super(Instance, self).instance_init(obj) | ||||
def _resolve_classes(self): | ||||
Thomas Kluyver
|
r13353 | if isinstance(self.klass, py3compat.string_types): | ||
Brian Granger
|
r2229 | self.klass = import_item(self.klass) | ||
Brian Granger
|
r2182 | def get_default_value(self): | ||
Brian Granger
|
r2177 | """Instantiate a default value instance. | ||
Bernardo B. Marques
|
r4872 | |||
Dav Clark
|
r2384 | This is called when the containing HasTraits classes' | ||
Brian Granger
|
r2182 | :meth:`__new__` method is called to ensure that a unique instance | ||
Dav Clark
|
r2384 | is created for each HasTraits instance. | ||
Brian Granger
|
r2177 | """ | ||
dv = self.default_value | ||||
if isinstance(dv, DefaultValueGenerator): | ||||
Brian Granger
|
r2229 | return dv.generate(self.klass) | ||
Brian Granger
|
r2177 | else: | ||
return dv | ||||
Dav Clark
|
r2385 | class This(ClassBasedTraitType): | ||
"""A trait for instances of the class containing this trait. | ||||
Brian Granger
|
r2180 | |||
Brian Granger
|
r2182 | Because how how and when class bodies are executed, the ``This`` | ||
Bernardo B. Marques
|
r4872 | trait can only have a default value of None. This, and because we | ||
Brian Granger
|
r2182 | always validate default values, ``allow_none`` is *always* true. | ||
""" | ||||
Brian Granger
|
r2180 | |||
Brian Granger
|
r2182 | info_text = 'an instance of the same type as the receiver or None' | ||
Brian Granger
|
r2180 | |||
Brian Granger
|
r2182 | def __init__(self, **metadata): | ||
super(This, self).__init__(None, **metadata) | ||||
Brian Granger
|
r2180 | |||
Brian Granger
|
r2182 | def validate(self, obj, value): | ||
Brian Granger
|
r2183 | # What if value is a superclass of obj.__class__? This is | ||
# complicated if it was the superclass that defined the This | ||||
Dav Clark
|
r2385 | # trait. | ||
Brian Granger
|
r2183 | if isinstance(value, self.this_class) or (value is None): | ||
Brian Granger
|
r2180 | return value | ||
else: | ||||
Brian Granger
|
r2182 | self.error(obj, value) | ||
Brian Granger
|
r2180 | |||
Brian Granger
|
r2177 | #----------------------------------------------------------------------------- | ||
Dav Clark
|
r2385 | # Basic TraitTypes implementations/subclasses | ||
Brian Granger
|
r2177 | #----------------------------------------------------------------------------- | ||
Brian Granger
|
r2157 | |||
Dav Clark
|
r2385 | class Any(TraitType): | ||
Brian Granger
|
r2157 | default_value = None | ||
info_text = 'any value' | ||||
zah
|
r16453 | class Int(TraitType): | ||
MinRK
|
r5344 | """An int trait.""" | ||
Brian Granger
|
r2157 | |||
default_value = 0 | ||||
MinRK
|
r5344 | info_text = 'an int' | ||
Brian Granger
|
r2157 | |||
def validate(self, obj, value): | ||||
zah
|
r16453 | if isinstance(value, int): | ||
Brian Granger
|
r2157 | return value | ||
self.error(obj, value) | ||||
Brian Granger
|
r2178 | class CInt(Int): | ||
Dav Clark
|
r2385 | """A casting version of the int trait.""" | ||
Brian Granger
|
r2178 | |||
def validate(self, obj, value): | ||||
try: | ||||
return int(value) | ||||
except: | ||||
self.error(obj, value) | ||||
Thomas Kluyver
|
r4764 | if py3compat.PY3: | ||
Long, CLong = Int, CInt | ||||
MinRK
|
r5344 | Integer = Int | ||
Thomas Kluyver
|
r4764 | else: | ||
zah
|
r16453 | class Long(TraitType): | ||
Thomas Kluyver
|
r4732 | """A long integer trait.""" | ||
Brian Granger
|
r2157 | |||
Thomas Kluyver
|
r13352 | default_value = 0 | ||
Thomas Kluyver
|
r4732 | info_text = 'a long' | ||
Brian Granger
|
r2157 | |||
Thomas Kluyver
|
r4732 | def validate(self, obj, value): | ||
zah
|
r16453 | if isinstance(value, long): | ||
Thomas Kluyver
|
r4732 | return value | ||
if isinstance(value, int): | ||||
return long(value) | ||||
self.error(obj, value) | ||||
Brian Granger
|
r2157 | |||
Thomas Kluyver
|
r4732 | class CLong(Long): | ||
"""A casting version of the long integer trait.""" | ||||
Brian Granger
|
r2178 | |||
Thomas Kluyver
|
r4732 | def validate(self, obj, value): | ||
try: | ||||
return long(value) | ||||
except: | ||||
self.error(obj, value) | ||||
Brian Granger
|
r2178 | |||
zah
|
r16453 | class Integer(TraitType): | ||
MinRK
|
r5344 | """An integer trait. | ||
Longs that are unnecessary (<= sys.maxint) are cast to ints.""" | ||||
default_value = 0 | ||||
info_text = 'an integer' | ||||
def validate(self, obj, value): | ||||
zah
|
r16453 | if isinstance(value, int): | ||
MinRK
|
r5344 | return value | ||
Pawel Jasinski
|
r8500 | if isinstance(value, long): | ||
MinRK
|
r5344 | # downcast longs that fit in int: | ||
# note that int(n > sys.maxint) returns a long, so | ||||
# we don't need a condition on this cast | ||||
return int(value) | ||||
Pawel Jasinski
|
r8500 | if sys.platform == "cli": | ||
from System import Int64 | ||||
if isinstance(value, Int64): | ||||
return int(value) | ||||
MinRK
|
r5344 | self.error(obj, value) | ||
Brian Granger
|
r2178 | |||
zah
|
r16453 | class Float(TraitType): | ||
Dav Clark
|
r2385 | """A float trait.""" | ||
Brian Granger
|
r2157 | |||
default_value = 0.0 | ||||
info_text = 'a float' | ||||
def validate(self, obj, value): | ||||
zah
|
r16453 | if isinstance(value, float): | ||
Brian Granger
|
r2157 | return value | ||
if isinstance(value, int): | ||||
return float(value) | ||||
self.error(obj, value) | ||||
Brian Granger
|
r2178 | class CFloat(Float): | ||
Dav Clark
|
r2385 | """A casting version of the float trait.""" | ||
Brian Granger
|
r2178 | |||
def validate(self, obj, value): | ||||
try: | ||||
return float(value) | ||||
except: | ||||
self.error(obj, value) | ||||
zah
|
r16453 | class Complex(TraitType): | ||
Dav Clark
|
r2385 | """A trait for complex numbers.""" | ||
Brian Granger
|
r2157 | |||
default_value = 0.0 + 0.0j | ||||
info_text = 'a complex number' | ||||
def validate(self, obj, value): | ||||
zah
|
r16453 | if isinstance(value, complex): | ||
Brian Granger
|
r2157 | return value | ||
if isinstance(value, (float, int)): | ||||
return complex(value) | ||||
self.error(obj, value) | ||||
Brian Granger
|
r2178 | class CComplex(Complex): | ||
Dav Clark
|
r2385 | """A casting version of the complex number trait.""" | ||
Brian Granger
|
r2178 | |||
def validate (self, obj, value): | ||||
try: | ||||
return complex(value) | ||||
except: | ||||
self.error(obj, value) | ||||
Thomas Kluyver
|
r4046 | # We should always be explicit about whether we're using bytes or unicode, both | ||
# for Python 3 conversion and for reliable unicode behaviour on Python 2. So | ||||
# we don't have a Str type. | ||||
zah
|
r16453 | class Bytes(TraitType): | ||
Thomas Kluyver
|
r4732 | """A trait for byte strings.""" | ||
Brian Granger
|
r2157 | |||
Thomas Kluyver
|
r5287 | default_value = b'' | ||
Thomas Kluyver
|
r13429 | info_text = 'a bytes object' | ||
Brian Granger
|
r2157 | |||
def validate(self, obj, value): | ||||
zah
|
r16453 | if isinstance(value, bytes): | ||
Brian Granger
|
r2157 | return value | ||
self.error(obj, value) | ||||
Thomas Kluyver
|
r4044 | class CBytes(Bytes): | ||
Thomas Kluyver
|
r4732 | """A casting version of the byte string trait.""" | ||
Brian Granger
|
r2178 | |||
def validate(self, obj, value): | ||||
try: | ||||
Thomas Kluyver
|
r4044 | return bytes(value) | ||
Brian Granger
|
r2178 | except: | ||
Thomas Kluyver
|
r4045 | self.error(obj, value) | ||
Brian Granger
|
r2178 | |||
zah
|
r16453 | class Unicode(TraitType): | ||
Dav Clark
|
r2385 | """A trait for unicode strings.""" | ||
Brian Granger
|
r2157 | |||
default_value = u'' | ||||
info_text = 'a unicode string' | ||||
def validate(self, obj, value): | ||||
zah
|
r16453 | if isinstance(value, py3compat.unicode_type): | ||
Brian Granger
|
r2157 | return value | ||
Thomas Kluyver
|
r4044 | if isinstance(value, bytes): | ||
Thomas Kluyver
|
r13734 | try: | ||
return value.decode('ascii', 'strict') | ||||
except UnicodeDecodeError: | ||||
msg = "Could not decode {!r} for unicode trait '{}' of {} instance." | ||||
raise TraitError(msg.format(value, self.name, class_of(obj))) | ||||
Brian Granger
|
r2157 | self.error(obj, value) | ||
Brian Granger
|
r2178 | class CUnicode(Unicode): | ||
Dav Clark
|
r2385 | """A casting version of the unicode trait.""" | ||
Brian Granger
|
r2178 | |||
def validate(self, obj, value): | ||||
try: | ||||
Thomas Kluyver
|
r13353 | return py3compat.unicode_type(value) | ||
Brian Granger
|
r2178 | except: | ||
self.error(obj, value) | ||||
Bernardo B. Marques
|
r4872 | |||
zah
|
r16453 | class ObjectName(TraitType): | ||
Thomas Kluyver
|
r4047 | """A string holding a valid object name in this version of Python. | ||
Bernardo B. Marques
|
r4872 | |||
Thomas Kluyver
|
r4047 | This does not check that the name exists in any scope.""" | ||
info_text = "a valid object identifier in Python" | ||||
Thomas Kluyver
|
r4740 | if py3compat.PY3: | ||
# Python 3: | ||||
coerce_str = staticmethod(lambda _,s: s) | ||||
Bernardo B. Marques
|
r4872 | |||
Thomas Kluyver
|
r4740 | else: | ||
Thomas Kluyver
|
r4047 | # Python 2: | ||
def coerce_str(self, obj, value): | ||||
"In Python 2, coerce ascii-only unicode to str" | ||||
if isinstance(value, unicode): | ||||
try: | ||||
return str(value) | ||||
except UnicodeEncodeError: | ||||
self.error(obj, value) | ||||
return value | ||||
Bernardo B. Marques
|
r4872 | |||
Thomas Kluyver
|
r4047 | def validate(self, obj, value): | ||
value = self.coerce_str(obj, value) | ||||
Bernardo B. Marques
|
r4872 | |||
Thomas Kluyver
|
r4740 | if isinstance(value, str) and py3compat.isidentifier(value): | ||
Thomas Kluyver
|
r4047 | return value | ||
self.error(obj, value) | ||||
class DottedObjectName(ObjectName): | ||||
"""A string holding a valid dotted object name in Python, such as A.b3._c""" | ||||
def validate(self, obj, value): | ||||
value = self.coerce_str(obj, value) | ||||
Bernardo B. Marques
|
r4872 | |||
Thomas Kluyver
|
r4740 | if isinstance(value, str) and py3compat.isidentifier(value, dotted=True): | ||
Thomas Kluyver
|
r4047 | return value | ||
self.error(obj, value) | ||||
Brian Granger
|
r2178 | |||
Brian Granger
|
r2157 | |||
zah
|
r16453 | class Bool(TraitType): | ||
Dav Clark
|
r2385 | """A boolean (True, False) trait.""" | ||
Brian Granger
|
r2742 | |||
Brian Granger
|
r2157 | default_value = False | ||
info_text = 'a boolean' | ||||
def validate(self, obj, value): | ||||
zah
|
r16453 | if isinstance(value, bool): | ||
Brian Granger
|
r2157 | return value | ||
self.error(obj, value) | ||||
Brian Granger
|
r2178 | |||
class CBool(Bool): | ||||
Dav Clark
|
r2385 | """A casting version of the boolean trait.""" | ||
Brian Granger
|
r2178 | |||
def validate(self, obj, value): | ||||
try: | ||||
return bool(value) | ||||
except: | ||||
Brian Granger
|
r2203 | self.error(obj, value) | ||
Brian Granger
|
r2204 | |||
Dav Clark
|
r2385 | class Enum(TraitType): | ||
Brian Granger
|
r2204 | """An enum that whose value must be in a given sequence.""" | ||
Brian Granger
|
r2203 | |||
def __init__(self, values, default_value=None, allow_none=True, **metadata): | ||||
self.values = values | ||||
zah
|
r16544 | super(Enum, self).__init__(default_value, allow_none=allow_none, **metadata) | ||
Brian Granger
|
r2203 | |||
def validate(self, obj, value): | ||||
if value in self.values: | ||||
return value | ||||
self.error(obj, value) | ||||
def info(self): | ||||
""" Returns a description of the trait.""" | ||||
result = 'any of ' + repr(self.values) | ||||
zah
|
r16453 | if self.allow_none: | ||
Brian Granger
|
r2203 | return result + ' or None' | ||
return result | ||||
class CaselessStrEnum(Enum): | ||||
Brian Granger
|
r2204 | """An enum of strings that are caseless in validate.""" | ||
Brian Granger
|
r2203 | |||
def validate(self, obj, value): | ||||
Thomas Kluyver
|
r13353 | if not isinstance(value, py3compat.string_types): | ||
Brian Granger
|
r2203 | self.error(obj, value) | ||
for v in self.values: | ||||
if v.lower() == value.lower(): | ||||
return v | ||||
Brian Granger
|
r2204 | self.error(obj, value) | ||
MinRK
|
r3870 | class Container(Instance): | ||
"""An instance of a container (list, set, etc.) | ||||
Brian Granger
|
r2204 | |||
MinRK
|
r3870 | To be subclassed by overriding klass. | ||
""" | ||||
klass = None | ||||
MinRK
|
r15472 | _cast_types = () | ||
MinRK
|
r3870 | _valid_defaults = SequenceTypes | ||
_trait = None | ||||
Brian Granger
|
r2204 | |||
MinRK
|
r3870 | def __init__(self, trait=None, default_value=None, allow_none=True, | ||
**metadata): | ||||
"""Create a container trait type from a list, set, or tuple. | ||||
Brian Granger
|
r2204 | |||
MinRK
|
r3870 | The default value is created by doing ``List(default_value)``, | ||
Brian Granger
|
r2204 | which creates a copy of the ``default_value``. | ||
MinRK
|
r3870 | |||
``trait`` can be specified, which restricts the type of elements | ||||
in the container to that TraitType. | ||||
If only one arg is given and it is not a Trait, it is taken as | ||||
``default_value``: | ||||
``c = List([1,2,3])`` | ||||
Parameters | ||||
---------- | ||||
trait : TraitType [ optional ] | ||||
the type for restricting the contents of the Container. If unspecified, | ||||
types are not checked. | ||||
default_value : SequenceType [ optional ] | ||||
The default value for the Trait. Must be list/tuple/set, and | ||||
will be cast to the container type. | ||||
allow_none : Bool [ default True ] | ||||
Whether to allow the value to be None | ||||
**metadata : any | ||||
further keys for extensions to the Trait (e.g. config) | ||||
Brian Granger
|
r2204 | """ | ||
MinRK
|
r3870 | # allow List([values]): | ||
epatters
|
r8407 | if default_value is None and not is_trait(trait): | ||
MinRK
|
r3870 | default_value = trait | ||
trait = None | ||||
Brian Granger
|
r2204 | if default_value is None: | ||
MinRK
|
r3870 | args = () | ||
elif isinstance(default_value, self._valid_defaults): | ||||
Brian Granger
|
r2204 | args = (default_value,) | ||
Dav Clark
|
r2380 | else: | ||
MinRK
|
r3870 | raise TypeError('default value of %s was %s' %(self.__class__.__name__, default_value)) | ||
epatters
|
r8407 | if is_trait(trait): | ||
self._trait = trait() if isinstance(trait, type) else trait | ||||
MinRK
|
r3870 | self._trait.name = 'element' | ||
elif trait is not None: | ||||
raise TypeError("`trait` must be a Trait or None, got %s"%repr_type(trait)) | ||||
Brian Granger
|
r2204 | |||
MinRK
|
r3870 | super(Container,self).__init__(klass=self.klass, args=args, | ||
Brian Granger
|
r2204 | allow_none=allow_none, **metadata) | ||
Brian Granger
|
r2742 | |||
MinRK
|
r3870 | def element_error(self, obj, element, validator): | ||
e = "Element of the '%s' trait of %s instance must be %s, but a value of %s was specified." \ | ||||
% (self.name, class_of(obj), validator.info(), repr_type(element)) | ||||
raise TraitError(e) | ||||
Brian Granger
|
r2742 | |||
MinRK
|
r3870 | def validate(self, obj, value): | ||
MinRK
|
r15472 | if isinstance(value, self._cast_types): | ||
value = self.klass(value) | ||||
MinRK
|
r3870 | value = super(Container, self).validate(obj, value) | ||
if value is None: | ||||
return value | ||||
MinRK
|
r3604 | |||
MinRK
|
r3870 | value = self.validate_elements(obj, value) | ||
return value | ||||
def validate_elements(self, obj, value): | ||||
validated = [] | ||||
if self._trait is None or isinstance(self._trait, Any): | ||||
return value | ||||
for v in value: | ||||
try: | ||||
zah
|
r16546 | v = self._trait._validate(obj, v) | ||
MinRK
|
r3870 | except TraitError: | ||
self.element_error(obj, v, self._trait) | ||||
else: | ||||
validated.append(v) | ||||
return self.klass(validated) | ||||
Thomas Kluyver
|
r16545 | def instance_init(self, obj): | ||
if isinstance(self._trait, Instance): | ||||
self._trait._resolve_classes() | ||||
super(Container, self).instance_init(obj) | ||||
MinRK
|
r3870 | |||
class List(Container): | ||||
"""An instance of a Python list.""" | ||||
klass = list | ||||
MinRK
|
r15472 | _cast_types = (tuple,) | ||
MinRK
|
r3604 | |||
Bradley M. Froehle
|
r8494 | def __init__(self, trait=None, default_value=None, minlen=0, maxlen=sys.maxsize, | ||
MinRK
|
r3870 | allow_none=True, **metadata): | ||
"""Create a List trait type from a list, set, or tuple. | ||||
The default value is created by doing ``List(default_value)``, | ||||
MinRK
|
r3604 | which creates a copy of the ``default_value``. | ||
MinRK
|
r3870 | |||
``trait`` can be specified, which restricts the type of elements | ||||
in the container to that TraitType. | ||||
If only one arg is given and it is not a Trait, it is taken as | ||||
``default_value``: | ||||
``c = List([1,2,3])`` | ||||
Parameters | ||||
---------- | ||||
trait : TraitType [ optional ] | ||||
the type for restricting the contents of the Container. If unspecified, | ||||
types are not checked. | ||||
default_value : SequenceType [ optional ] | ||||
The default value for the Trait. Must be list/tuple/set, and | ||||
will be cast to the container type. | ||||
minlen : Int [ default 0 ] | ||||
The minimum length of the input list | ||||
Bradley M. Froehle
|
r8494 | maxlen : Int [ default sys.maxsize ] | ||
MinRK
|
r3870 | The maximum length of the input list | ||
allow_none : Bool [ default True ] | ||||
Whether to allow the value to be None | ||||
**metadata : any | ||||
further keys for extensions to the Trait (e.g. config) | ||||
MinRK
|
r3604 | """ | ||
MinRK
|
r3870 | self._minlen = minlen | ||
self._maxlen = maxlen | ||||
super(List, self).__init__(trait=trait, default_value=default_value, | ||||
allow_none=allow_none, **metadata) | ||||
def length_error(self, obj, value): | ||||
e = "The '%s' trait of %s instance must be of length %i <= L <= %i, but a value of %s was specified." \ | ||||
% (self.name, class_of(obj), self._minlen, self._maxlen, value) | ||||
raise TraitError(e) | ||||
def validate_elements(self, obj, value): | ||||
length = len(value) | ||||
if length < self._minlen or length > self._maxlen: | ||||
self.length_error(obj, value) | ||||
return super(List, self).validate_elements(obj, value) | ||||
zah
|
r16453 | |||
MinRK
|
r15472 | def validate(self, obj, value): | ||
value = super(List, self).validate(obj, value) | ||||
value = self.validate_elements(obj, value) | ||||
return value | ||||
zah
|
r16453 | |||
MinRK
|
r3870 | |||
MinRK
|
r15472 | class Set(List): | ||
MinRK
|
r3870 | """An instance of a Python set.""" | ||
klass = set | ||||
MinRK
|
r15472 | _cast_types = (tuple, list) | ||
MinRK
|
r3870 | |||
class Tuple(Container): | ||||
"""An instance of a Python tuple.""" | ||||
klass = tuple | ||||
MinRK
|
r15472 | _cast_types = (list,) | ||
MinRK
|
r3870 | |||
def __init__(self, *traits, **metadata): | ||||
"""Tuple(*traits, default_value=None, allow_none=True, **medatata) | ||||
Create a tuple from a list, set, or tuple. | ||||
Create a fixed-type tuple with Traits: | ||||
``t = Tuple(Int, Str, CStr)`` | ||||
would be length 3, with Int,Str,CStr for each element. | ||||
If only one arg is given and it is not a Trait, it is taken as | ||||
default_value: | ||||
``t = Tuple((1,2,3))`` | ||||
Otherwise, ``default_value`` *must* be specified by keyword. | ||||
Parameters | ||||
---------- | ||||
*traits : TraitTypes [ optional ] | ||||
the tsype for restricting the contents of the Tuple. If unspecified, | ||||
types are not checked. If specified, then each positional argument | ||||
corresponds to an element of the tuple. Tuples defined with traits | ||||
are of fixed length. | ||||
default_value : SequenceType [ optional ] | ||||
The default value for the Tuple. Must be list/tuple/set, and | ||||
will be cast to a tuple. If `traits` are specified, the | ||||
`default_value` must conform to the shape and type they specify. | ||||
allow_none : Bool [ default True ] | ||||
Whether to allow the value to be None | ||||
**metadata : any | ||||
further keys for extensions to the Trait (e.g. config) | ||||
""" | ||||
default_value = metadata.pop('default_value', None) | ||||
allow_none = metadata.pop('allow_none', True) | ||||
# allow Tuple((values,)): | ||||
epatters
|
r8407 | if len(traits) == 1 and default_value is None and not is_trait(traits[0]): | ||
MinRK
|
r3870 | default_value = traits[0] | ||
traits = () | ||||
MinRK
|
r3604 | if default_value is None: | ||
MinRK
|
r3870 | args = () | ||
elif isinstance(default_value, self._valid_defaults): | ||||
MinRK
|
r3604 | args = (default_value,) | ||
else: | ||||
MinRK
|
r3870 | raise TypeError('default value of %s was %s' %(self.__class__.__name__, default_value)) | ||
self._traits = [] | ||||
for trait in traits: | ||||
epatters
|
r8407 | t = trait() if isinstance(trait, type) else trait | ||
MinRK
|
r3870 | t.name = 'element' | ||
self._traits.append(t) | ||||
if self._traits and default_value is None: | ||||
# don't allow default to be an empty container if length is specified | ||||
args = None | ||||
super(Container,self).__init__(klass=self.klass, args=args, | ||||
MinRK
|
r3604 | allow_none=allow_none, **metadata) | ||
MinRK
|
r3870 | def validate_elements(self, obj, value): | ||
if not self._traits: | ||||
# nothing to validate | ||||
return value | ||||
if len(value) != len(self._traits): | ||||
e = "The '%s' trait of %s instance requires %i elements, but a value of %s was specified." \ | ||||
% (self.name, class_of(obj), len(self._traits), repr_type(value)) | ||||
raise TraitError(e) | ||||
validated = [] | ||||
for t,v in zip(self._traits, value): | ||||
try: | ||||
zah
|
r16547 | v = t._validate(obj, v) | ||
MinRK
|
r3870 | except TraitError: | ||
self.element_error(obj, v, t) | ||||
else: | ||||
validated.append(v) | ||||
return tuple(validated) | ||||
MinRK
|
r3604 | |||
Brian Granger
|
r2738 | class Dict(Instance): | ||
"""An instance of a Python dict.""" | ||||
def __init__(self, default_value=None, allow_none=True, **metadata): | ||||
"""Create a dict trait type from a dict. | ||||
Bernardo B. Marques
|
r4872 | The default value is created by doing ``dict(default_value)``, | ||
Brian Granger
|
r2738 | which creates a copy of the ``default_value``. | ||
""" | ||||
if default_value is None: | ||||
args = ((),) | ||||
elif isinstance(default_value, dict): | ||||
args = (default_value,) | ||||
elif isinstance(default_value, SequenceTypes): | ||||
args = (default_value,) | ||||
else: | ||||
raise TypeError('default value of Dict was %s' % default_value) | ||||
Bernardo B. Marques
|
r4872 | super(Dict,self).__init__(klass=dict, args=args, | ||
Brian Granger
|
r2738 | allow_none=allow_none, **metadata) | ||
Brian Granger
|
r2752 | |||
zah
|
r16453 | class TCPAddress(TraitType): | ||
Brian Granger
|
r2742 | """A trait for an (ip, port) tuple. | ||
This allows for both IPv4 IP addresses as well as hostnames. | ||||
""" | ||||
default_value = ('127.0.0.1', 0) | ||||
info_text = 'an (ip, port) tuple' | ||||
def validate(self, obj, value): | ||||
if isinstance(value, tuple): | ||||
if len(value) == 2: | ||||
Thomas Kluyver
|
r13353 | if isinstance(value[0], py3compat.string_types) and isinstance(value[1], int): | ||
Brian Granger
|
r2742 | port = value[1] | ||
if port >= 0 and port <= 65535: | ||||
return value | ||||
self.error(obj, value) | ||||
Bradley M. Froehle
|
r6732 | |||
zah
|
r16453 | class CRegExp(TraitType): | ||
Bradley M. Froehle
|
r6743 | """A casting compiled regular expression trait. | ||
Accepts both strings and compiled regular expressions. The resulting | ||||
attribute will be a compiled regular expression.""" | ||||
Bradley M. Froehle
|
r6732 | |||
info_text = 'a regular expression' | ||||
def validate(self, obj, value): | ||||
try: | ||||
return re.compile(value) | ||||
except: | ||||
self.error(obj, value) | ||||