''' Provide ancillary utility functions useful for manipulating Bokeh
documents.
'''
from __future__ import absolute_import
from ..model import get_class
# TODO (bev) remove when push_notebook is re-implemented
[docs]def compute_patch_between_json(from_json, to_json):
''' Compute a JSON diff between a two Bokeh document states.
.. note::
This function is used to send changes that happened between calls to
``show()`` and ``push_notebook()`` and will be removed when a better
implementation based on the Bokeh protocol is available.
Args:
from_json (``JSON``)
JSON representing a "starting" Bokeh document state
to_json (``JSON``)
JSON representing a "final" Bokeh document state
Returns
``JSON``
'''
def refs(json):
result = {}
for obj in json['roots']['references']:
result[obj['id']] = obj
return result
from_references = refs(from_json)
from_roots = {}
from_root_ids = []
for r in from_json['roots']['root_ids']:
from_roots[r] = from_references[r]
from_root_ids.append(r)
to_references = refs(to_json)
to_roots = {}
to_root_ids = []
for r in to_json['roots']['root_ids']:
to_roots[r] = to_references[r]
to_root_ids.append(r)
from_root_ids.sort()
to_root_ids.sort()
from_set = set(from_root_ids)
to_set = set(to_root_ids)
removed = from_set - to_set
added = to_set - from_set
combined_references = dict(from_references)
for k in to_references.keys():
combined_references[k] = to_references[k]
value_refs = {}
events = []
for removed_root_id in removed:
model = dict(combined_references[removed_root_id])
del model['attributes']
events.append({ 'kind' : 'RootRemoved',
'model' : model })
for added_root_id in added:
_value_record_references(combined_references,
combined_references[added_root_id],
value_refs)
model = dict(combined_references[added_root_id])
del model['attributes']
events.append({ 'kind' : 'RootAdded',
'model' : model })
for id in to_references:
if id in from_references:
update_model_events = _events_to_sync_objects(
combined_references,
from_references[id],
to_references[id],
value_refs
)
events.extend(update_model_events)
return dict(
events=events,
references=list(value_refs.values())
)
[docs]def initialize_references_json(references_json, references, setter=None):
''' Given a JSON representation of the models in a graph, and new model
objects, set the properties on the models from the JSON
Args:
references_json (``JSON``)
JSON specifying attributes and values to initialize new model
objects with.
references (dict[str, Model])
A dictionary mapping model IDs to newly created (but not yet
initialized) Bokeh models.
**This is an "out" parameter**. The values it contains will be
modified in-place.
setter (ClientSession or ServerSession or None, optional) :
This is used to prevent "boomerang" updates to Bokeh apps.
(default: None)
In the context of a Bokeh server application, incoming updates
to properties will be annotated with the session that is
doing the updating. This value is propagated through any
subsequent change notifications that the update triggers.
The session can compare the event setter to itself, and
suppress any updates that originate from itself.
'''
for obj in references_json:
obj_id = obj['id']
obj_attrs = obj['attributes']
instance = references[obj_id]
instance.update_from_json(obj_attrs, models=references, setter=setter)
[docs]def instantiate_references_json(references_json):
''' Given a JSON representation of all the models in a graph, return a
dict of new model objects.
Args:
references_json (``JSON``)
JSON specifying new Bokeh models to create
Returns:
dict[str, Model]
'''
# Create all instances, but without setting their props
references = {}
for obj in references_json:
obj_id = obj['id']
obj_type = obj.get('subtype', obj['type'])
cls = get_class(obj_type)
instance = cls(id=obj_id, _block_events=True)
if instance is None:
raise RuntimeError('Error loading model from JSON (type: %s, id: %s)' % (obj_type, obj_id))
references[instance._id] = instance
return references
[docs]def references_json(references):
''' Given a list of all models in a graph, return JSON representing
them and their properties.
Args:
references (seq[Model]) :
A list of models to convert to JSON
Returns:
list
'''
references_json = []
for r in references:
ref = r.ref
ref['attributes'] = r._to_json_like(include_defaults=False)
references_json.append(ref)
return references_json
# TODO (bev) remove when push_notebook is re-implemented
def _event_for_attribute_change(all_references, changed_obj, key, new_value, value_refs):
event = dict(
kind='ModelChanged',
model=dict(id=changed_obj['id'], type=changed_obj['type']),
attr=key,
new=new_value,
)
_value_record_references(all_references, new_value, value_refs)
return event
# TODO (bev) remove when push_notebook is re-implemented
def _events_to_sync_objects(all_references, from_obj, to_obj, value_refs):
from_keys = set(from_obj['attributes'].keys())
to_keys = set(to_obj['attributes'].keys())
removed = from_keys - to_keys
added = to_keys - from_keys
shared = from_keys & to_keys
events = []
for key in removed:
raise RuntimeError("internal error: should not be possible to delete attribute %s" % key)
for key in added:
new_value = to_obj['attributes'][key]
events.append(_event_for_attribute_change(all_references,
from_obj,
key,
new_value,
value_refs))
for key in shared:
old_value = from_obj['attributes'].get(key)
new_value = to_obj['attributes'].get(key)
if old_value is None and new_value is None:
continue
if old_value is None or new_value is None or old_value != new_value:
event = _event_for_attribute_change(all_references,
from_obj,
key,
new_value,
value_refs)
events.append(event)
return events
# TODO (bev) remove when push_notebook is re-implemented
def _value_record_references(all_references, value, result):
if value is None: return
if isinstance(value, dict) and set(['id', 'type']).issubset(set(value.keys())):
if value['id'] not in result:
ref = all_references[value['id']]
result[value['id']] = ref
_value_record_references(all_references, ref['attributes'], result)
elif isinstance(value, (list, tuple)):
for elem in value:
_value_record_references(all_references, elem, result)
elif isinstance(value, dict):
for k, elem in value.items():
_value_record_references(all_references, elem, result)