Source code for bokeh.model

''' Provide a base class for all objects (called Bokeh Models) that go in
Bokeh Documents.

The :class:`~bokeh.document.Document` class is the basic unit of serialization
for Bokeh visualizations and applications. Documents contain collections of
related Bokeh Models (e.g. ``Plot``, ``Range1d``, etc. ) that can be all
serialized together.

The :class:`~bokeh.model.Model` class is a base class for all objects that
can be added to a Document.

'''
from __future__ import absolute_import, print_function

import logging
logger = logging.getLogger(__file__)

from contextlib import contextmanager
from json import loads
from operator import itemgetter

from six import iteritems

from .core.json_encoder import serialize_json
from .core.properties import Any, Dict, Instance, List, String
from .core.has_props import HasProps, MetaHasProps
from .core.query import find
from .themes import default as default_theme
from .util.callback_manager import CallbackManager
from .util.future import with_metaclass
from .util.serialization import make_id


[docs]class Viewable(MetaHasProps): """ Any Bokeh Model which has its own View Model in the persistence layer. """ # Stores a mapping from subclass __view_model__ names to classes model_class_reverse_map = {} # Mmmm.. metaclass inheritance. On the one hand, it seems a little # overkill. On the other hand, this is exactly the sort of thing # it's meant for. def __new__(meta_cls, class_name, bases, class_dict): if "__view_model__" not in class_dict: class_dict["__view_model__"] = class_name class_dict["get_class"] = Viewable.get_class # Create the new class newcls = super(Viewable, meta_cls).__new__(meta_cls, class_name, bases, class_dict) entry = class_dict.get("__subtype__", class_dict["__view_model__"]) # Add it to the reverse map, but check for duplicates first if entry in Viewable.model_class_reverse_map and not hasattr(newcls, "__implementation__"): raise Warning("Duplicate __view_model__ or __subtype__ declaration of '%s' for " \ "class %s. Previous definition: %s" % \ (entry, class_name, Viewable.model_class_reverse_map[entry])) Viewable.model_class_reverse_map[entry] = newcls return newcls @classmethod def _preload_models(cls): from . import models; models from .plotting import Figure; Figure try: from .charts import Chart; Chart except RuntimeError: # this would occur if pandas is not installed but then we can't # use the bokeh.charts interface anyway pass @classmethod
[docs] def get_class(cls, view_model_name): """ Given a __view_model__ name, returns the corresponding class object """ cls._preload_models() d = Viewable.model_class_reverse_map if view_model_name in d: return d[view_model_name] else: raise KeyError("View model name '%s' not found" % view_model_name)
[docs]class Model(with_metaclass(Viewable, HasProps, CallbackManager)): ''' Base class for all objects stored in Bokeh ``Document`` instances. ''' name = String(help=""" An arbitrary, user-supplied name for this model. This name can be useful when querying the document to retrieve specific Bokeh models. .. code:: python >>> plot.circle([1,2,3], [4,5,6], name="temp") >>> plot.select(name="temp") [GlyphRenderer(id='399d53f5-73e9-44d9-9527-544b761c7705', ...)] .. note:: No uniqueness guarantees or other conditions are enforced on any names that are provided. """) tags = List(Any, help=""" An optional list of arbitrary, user-supplied values to attach to this model. This data can be useful when querying the document to retrieve specific Bokeh models: .. code:: python >>> r = plot.circle([1,2,3], [4,5,6]) >>> r.tags = ["foo", 10] >>> plot.select(tags=['foo', 10]) [GlyphRenderer(id='1de4c3df-a83d-480a-899b-fb263d3d5dd9', ...)] Or simply a convenient way to attach any necessary metadata to a model that can be accessed by CustomJS callbacks, etc. """) js_callbacks = Dict(String, List(Instance("bokeh.models.callbacks.CustomJS")), help=""" A mapping of attribute names to lists of CustomJS callbacks, to be set up on BokehJS side when the document is created. Typically, rather then modifying this property directly, callbacks should be added using the ``Model.js_on_change`` method: .. code:: python callback = CustomJS(code="console.log('stuff')") plot.x_range.js_on_change('start', callback) """) def __init__(self, **kwargs): self._id = kwargs.pop("id", make_id()) self._document = None super(Model, self).__init__(**kwargs) default_theme.apply_to_model(self) def _attach_document(self, doc): '''This should only be called by the Document implementation to set the document field''' if self._document is not None and self._document is not doc: raise RuntimeError("Models must be owned by only a single document, %r is already in a doc" % (self)) doc.theme.apply_to_model(self) self._document = doc def _detach_document(self): '''This should only be called by the Document implementation to unset the document field''' self._document = None default_theme.apply_to_model(self) @property def document(self): return self._document
[docs] def on_change(self, attr, *callbacks): ''' Add a callback on this object to trigger when ``attr`` changes. Args: attr (str) : an attribute name on this object callback (callable) : a callback function to register Returns: None ''' if attr not in self.properties(): raise ValueError("attempted to add a callback on nonexistent %s.%s property" % (self.__class__.__name__, attr)) super(Model, self).on_change(attr, *callbacks)
[docs] def js_on_change(self, event, *callbacks): ''' Attach a CustomJS callback to an arbitrary BokehJS model event. On the BokehJS side, change events for model properties have the form ``"change:property_name"``. As a convenience, if the event name passed to this method is also the name of a property on the model, then it will be prefixed with ``"change:"`` automatically: .. code:: python # these two are equivalent source.js_on_change('data', callback) source.js_on_change('change:data', callback) However, there are other kinds of events that can be useful to respond to, in addition to property change events. For example to run a callback whenever data is streamed to a ``ColumnDataSource``, use the ``"stream"`` event on the source: .. code:: python source.js_on_change('stream', callback) ''' if len(callbacks) == 0: raise ValueError("js_on_change takes an event name and one or more callbacks, got only one parameter") # handle any CustomJS callbacks here from bokeh.models.callbacks import CustomJS if not all(isinstance(x, CustomJS) for x in callbacks): raise ValueError("not all callback values are CustomJS instances") if event in self.properties(): event = "change:%s" % event if event not in self.js_callbacks: self.js_callbacks[event] = [] for callback in callbacks: if callback in self.js_callbacks[event]: continue self.js_callbacks[event].append(callback)
[docs] def trigger(self, attr, old, new, hint=None, setter=None): # The explicit assumption here is that hinted events do not # need to go through all the same invalidation steps. Currently # as of Bokeh 0.11.1 the only hinted event is ColumnsStreamedEvent. # This may need to be further refined in the future, if the # assumption does not hold for future hinted events (e.g. the hint # could specify explicitly whether to do normal invalidation or not) if not hint: dirty = { 'count' : 0 } def mark_dirty(obj): dirty['count'] += 1 if self._document is not None: self._visit_value_and_its_immediate_references(new, mark_dirty) self._visit_value_and_its_immediate_references(old, mark_dirty) if dirty['count'] > 0: self._document._invalidate_all_models() # chain up to invoke callbacks super(Model, self).trigger(attr, old, new, hint, setter)
@property def ref(self): if "__subtype__" in self.__class__.__dict__: return { 'type': self.__view_model__, 'subtype': self.__subtype__, 'id': self._id, } else: return { 'type': self.__view_model__, 'id': self._id, }
[docs] def select(self, selector): ''' Query this object and all of its references for objects that match the given selector. Args: selector (JSON-like) : Returns: seq[Model] ''' return find(self.references(), selector)
[docs] def select_one(self, selector): ''' Query this object and all of its references for objects that match the given selector. Raises an error if more than one object is found. Returns single matching object, or None if nothing is found Args: selector (JSON-like) : Returns: Model ''' result = list(self.select(selector)) if len(result) > 1: raise ValueError("Found more than one object matching %s: %r" % (selector, result)) if len(result) == 0: return None return result[0]
[docs] def set_select(self, selector, updates): ''' Update objects that match a given selector with the specified attribute/value updates. Args: selector (JSON-like) : updates (dict) : Returns: None ''' for obj in self.select(selector): for key, val in updates.items(): setattr(obj, key, val)
[docs] def layout(self, side, plot): try: return self in getattr(plot, side) except: return []
@classmethod def _visit_immediate_value_references(cls, value, visitor): ''' Visit all references to another Model without recursing into any of the child Model; may visit the same Model more than once if it's referenced more than once. Does not visit the passed-in value. ''' if isinstance(value, HasProps): for attr in value.properties_with_refs(): child = getattr(value, attr) cls._visit_value_and_its_immediate_references(child, visitor) else: cls._visit_value_and_its_immediate_references(value, visitor) @classmethod def _visit_value_and_its_immediate_references(cls, obj, visitor): if isinstance(obj, Model): visitor(obj) elif isinstance(obj, HasProps): # this isn't a Model, so recurse into it cls._visit_immediate_value_references(obj, visitor) elif isinstance(obj, (list, tuple)): for item in obj: cls._visit_value_and_its_immediate_references(item, visitor) elif isinstance(obj, dict): for key, value in iteritems(obj): cls._visit_value_and_its_immediate_references(key, visitor) cls._visit_value_and_its_immediate_references(value, visitor) @classmethod
[docs] def collect_models(cls, *input_values): """ Iterate over ``input_values`` and descend through their structure collecting all nested ``Models`` on the go. The resulting list is duplicate-free based on objects' identifiers. """ ids = set([]) collected = [] queued = [] def queue_one(obj): if obj._id not in ids: queued.append(obj) for value in input_values: cls._visit_value_and_its_immediate_references(value, queue_one) while queued: obj = queued.pop(0) if obj._id not in ids: ids.add(obj._id) collected.append(obj) cls._visit_immediate_value_references(obj, queue_one) return collected
[docs] def references(self): """Returns all ``Models`` that this object has references to. """ return set(self.collect_models(self))
def _to_json_like(self, include_defaults): """ Returns a dictionary of the attributes of this object, in a layout corresponding to what BokehJS expects at unmarshalling time. This method does not convert "Bokeh types" into "plain JSON types," for example each child Model will still be a Model, rather than turning into a reference, numpy isn't handled, etc. That's what "json like" means. This method should be considered "private" or "protected", for use internal to Bokeh; use to_json() instead because it gives you only plain JSON-compatible types. Args: include_defaults (bool) : whether to include attributes that haven't been changed from the default. """ all_attrs = self.properties_with_values(include_defaults=include_defaults) # If __subtype__ is defined, then this model may introduce properties # that don't exist on __view_model__ in bokehjs. Don't serialize such # properties. subtype = getattr(self.__class__, "__subtype__", None) if subtype is not None and subtype != self.__class__.__view_model__: attrs = {} for attr, value in all_attrs.items(): if attr in self.__class__.__dict__: continue else: attrs[attr] = value else: attrs = all_attrs for (k, v) in attrs.items(): # we can't serialize Infinity, we send it as None and # the other side has to fix it up. This transformation # can't be in our json_encoder because the json # module checks for inf before it calls the custom # encoder. if isinstance(v, float) and v == float('inf'): attrs[k] = None return attrs
[docs] def to_json(self, include_defaults): """ Returns a dictionary of the attributes of this object, containing only "JSON types" (string, number, boolean, none, dict, list). References to other objects are serialized as "refs" (just the object ID and type info), so the deserializer will need to separately have the full attributes of those other objects. There's no corresponding from_json() because to deserialize an object is normally done in the context of a Document (since the Document can resolve references). For most purposes it's best to serialize and deserialize entire documents. Args: include_defaults (bool) : whether to include attributes that haven't been changed from the default """ return loads(self.to_json_string(include_defaults=include_defaults))
[docs] def to_json_string(self, include_defaults): """Returns a JSON string encoding the attributes of this object. References to other objects are serialized as references (just the object ID and type info), so the deserializer will need to separately have the full attributes of those other objects. There's no corresponding from_json_string() because to deserialize an object is normally done in the context of a Document (since the Document can resolve references). For most purposes it's best to serialize and deserialize entire documents. Args: include_defaults (bool) : whether to include attributes that haven't been changed from the default """ json_like = self._to_json_like(include_defaults=include_defaults) json_like['id'] = self._id # serialize_json "fixes" the JSON from _to_json_like by converting # all types into plain JSON types # (it converts Model into refs, # for example). return serialize_json(json_like)
def __str__(self): return "%s(id=%r, ...)" % (self.__class__.__name__, getattr(self, "_id", None)) __repr__ = __str__ def _bokeh_repr_pretty_(self, p, cycle): name = "%s.%s" % (self.__class__.__module__, self.__class__.__name__) _id = getattr(self, "_id", None) if cycle: p.text(name) p.text('(id=') p.pretty(_id) p.text(', ...)') else: with p.group(4, '%s(' % name, ')'): props = self.properties_with_values().items() sorted_props = sorted(props, key=itemgetter(0)) all_props = [('id', _id)] + sorted_props for i, (prop, value) in enumerate(all_props): if i == 0: p.breakable('') else: p.text(',') p.breakable() p.text(prop) p.text('=') p.pretty(value) def _repr_html_(self): module = self.__class__.__module__ name = self.__class__.__name__ _id = getattr(self, "_id", None) cls_name = make_id() def row(c): return '<div style="display: table-row;">' + c + '</div>' def hidden_row(c): return '<div class="%s" style="display: none;">%s</div>' % (cls_name, c) def cell(c): return '<div style="display: table-cell;">' + c + '</div>' html = '' html += '<div style="display: table;">' ellipsis_id = make_id() ellipsis = '<span id="%s" style="cursor: pointer;">&hellip;)</span>' % ellipsis_id prefix = cell('<b title="%s.%s">%s</b>(' % (module, name, name)) html += row(prefix + cell('id' + '&nbsp;=&nbsp;' + repr(_id) + ', ' + ellipsis)) props = self.properties_with_values().items() sorted_props = sorted(props, key=itemgetter(0)) all_props = sorted_props for i, (prop, value) in enumerate(all_props): end = ')' if i == len(all_props)-1 else ',' html += hidden_row(cell("") + cell(prop + '&nbsp;=&nbsp;' + repr(value) + end)) html += '</div>' html += """ <script> (function() { var expanded = false; var ellipsis = document.getElementById("%(ellipsis_id)s"); ellipsis.addEventListener("click", function() { var rows = document.getElementsByClassName("%(cls_name)s"); for (var i = 0; i < rows.length; i++) { var el = rows[i]; el.style.display = expanded ? "none" : "table-row"; } ellipsis.innerHTML = expanded ? "&hellip;)" : "&lsaquo;&lsaquo;&lsaquo;"; expanded = !expanded; }); })(); </script> """ % dict(ellipsis_id=ellipsis_id, cls_name=cls_name) return html
def _find_some_document(models): from .document import Document # First try the easy stuff... doc = None for model in models: if isinstance(model, Document): doc = model break elif isinstance(model, Model): if model.document is not None: doc = model.document break # Now look in children of models if doc is None: for model in models: if isinstance(model, Model): # see if some child of ours is in a doc, this is meant to # handle a thing like: # p = figure() # box = HBox(children=[p]) # show(box) for r in model.references(): if r.document is not None: doc = r.document break return doc class _ModelInDocument(object): # 'models' can be a single Model, a single Document, or a list of either def __init__(self, models): from .document import Document self._to_remove_after = [] if not isinstance(models, list): models = [models] self._doc = _find_some_document(models) if self._doc is None: # oh well - just make up a doc self._doc = Document() for model in models: if isinstance(model, Model): if model.document is None: self._to_remove_after.append(model) def __exit__(self, type, value, traceback): for model in self._to_remove_after: model.document.remove_root(model) def __enter__(self): for model in self._to_remove_after: self._doc.add_root(model) @contextmanager def _ModelInEmptyDocument(model): from .document import Document full_doc = _find_some_document([model]) model._document = None for ref in model.references(): ref._document = None empty_doc = Document() empty_doc.add_root(model) yield model model._document = full_doc for ref in model.references(): ref._document = full_doc