#grayscale");filter:gray;-webkit-filter:grayscale(100%);}.bk-root .bk-logo-small{width:20px;height:20px;background-image:url();}.bk-root .bk-logo-notebook{display:inline-block;vertical-align:middle;margin-right:5px;}.rendered_html .bk-root .bk-tooltip table,.rendered_html .bk-root .bk-tooltip tr,.rendered_html .bk-root .bk-tooltip th,.rendered_html .bk-root .bk-tooltip td{border:none;padding:1px;}.bk-root .bk-toolbar-hidden{visibility:hidden;opacity:0;transition:visibility 0.3s linear, opacity 0.3s linear;}.bk-root .bk-toolbar{display:flex;flex-wrap:nowrap;align-items:center;user-select:none;-ms-user-select:none;-moz-user-select:none;-webkit-user-select:none;}.bk-root .bk-toolbar .bk-logo{flex-shrink:0;}.bk-root .bk-toolbar.bk-above,.bk-root .bk-toolbar.bk-below{flex-direction:row;justify-content:flex-end;}.bk-root .bk-toolbar.bk-above .bk-logo,.bk-root .bk-toolbar.bk-below .bk-logo{order:1;margin-left:5px;margin-right:0px;}.bk-root .bk-toolbar.bk-left,.bk-root .bk-toolbar.bk-right{flex-direction:column;justify-content:flex-start;}.bk-root .bk-toolbar.bk-left .bk-logo,.bk-root .bk-toolbar.bk-right .bk-logo{order:0;margin-bottom:5px;margin-top:0px;}.bk-root .bk-toolbar-button{width:30px;height:30px;cursor:pointer;background-size:60% 60%;background-origin:border-box;background-color:transparent;background-repeat:no-repeat;background-position:center center;}.bk-root .bk-toolbar-button:hover,.bk-root .bk-tool-overflow:hover{background-color:rgba(192, 192, 192, 0.15);}.bk-root .bk-toolbar-button:focus,.bk-root .bk-tool-overflow:focus,.bk-root .bk-toolbar-button:focus-visible,.bk-root .bk-tool-overflow:focus-visible{outline:1px dotted #26aae1;outline-offset:-1px;}.bk-root .bk-toolbar-button::-moz-focus-inner,.bk-root .bk-tool-overflow::-moz-focus-inner{border:0;}.bk-root .bk-above .bk-toolbar-button{border-bottom:2px solid transparent;}.bk-root .bk-above .bk-toolbar-button.bk-active{border-bottom-color:#26aae1;}.bk-root .bk-below .bk-toolbar-button{border-top:2px solid transparent;}.bk-root .bk-below .bk-toolbar-button.bk-active{border-top-color:#26aae1;}.bk-root .bk-right .bk-toolbar-button{border-left:2px solid transparent;}.bk-root .bk-right .bk-toolbar-button.bk-active{border-left-color:#26aae1;}.bk-root .bk-left .bk-toolbar-button{border-right:2px solid transparent;}.bk-root .bk-left .bk-toolbar-button.bk-active{border-right-color:#26aae1;}.bk-root .bk-divider{content:" ";display:inline-block;background-color:lightgray;}.bk-root .bk-above .bk-divider,.bk-root .bk-below .bk-divider{height:10px;width:1px;}.bk-root .bk-left .bk-divider,.bk-root .bk-right .bk-divider{height:1px;width:10px;}.bk-root .bk-tool-overflow{color:gray;display:flex;align-items:center;}.bk-root .bk-above .bk-tool-overflow,.bk-root .bk-below .bk-tool-overflow,.bk-root .bk-horizontal .bk-tool-overflow{width:15px;height:30px;flex-direction:row;}.bk-root .bk-left .bk-tool-overflow,.bk-root .bk-right .bk-tool-overflow,.bk-root .bk-vertical .bk-tool-overflow{width:30px;height:15px;flex-direction:column;}.bk-root .bk-tool-icon-copy-to-clipboard{background-image:url("");}.bk-root .bk-tool-icon-replace-mode{background-image:url("");}.bk-root .bk-tool-icon-append-mode{background-image:url("");}.bk-root .bk-tool-icon-intersect-mode{background-image:url("");}.bk-root .bk-tool-icon-subtract-mode{background-image:url("");}.bk-root .bk-tool-icon-clear-selection{background-image:url("");}.bk-root .bk-tool-icon-box-select{background-image:url("");}.bk-root .bk-tool-icon-box-zoom{background-image:url("");}.bk-root .bk-tool-icon-zoom-in{background-image:url("");}.bk-root .bk-tool-icon-zoom-out{background-image:url("");}.bk-root .bk-tool-icon-help{background-image:url("");}.bk-root .bk-tool-icon-hover{background-image:url("");}.bk-root .bk-tool-icon-crosshair{background-image:url("");}.bk-root .bk-tool-icon-lasso-select{background-image:url("");}.bk-root .bk-tool-icon-pan{background-image:url("");}.bk-root .bk-tool-icon-xpan{background-image:url("");}.bk-root .bk-tool-icon-ypan{background-image:url("");}.bk-root .bk-tool-icon-range{background-image:url("");}.bk-root .bk-tool-icon-polygon-select{background-image:url("");}.bk-root .bk-tool-icon-redo{background-image:url("");}.bk-root .bk-tool-icon-reset{background-image:url("");}.bk-root .bk-tool-icon-save{background-image:url("");}.bk-root .bk-tool-icon-tap-select{background-image:url("");}.bk-root .bk-tool-icon-undo{background-image:url("");}.bk-root .bk-tool-icon-wheel-pan{background-image:url("");}.bk-root .bk-tool-icon-wheel-zoom{background-image:url("");}.bk-root .bk-tool-icon-box-edit{background-image:url("");}.bk-root .bk-tool-icon-freehand-draw{background-image:url("");}.bk-root .bk-tool-icon-poly-draw{background-image:url("");}.bk-root .bk-tool-icon-point-draw{background-image:url("");}.bk-root .bk-tool-icon-poly-edit{background-image:url("");}.bk-root .bk-tool-icon-line-edit{background-image:url("");}.bk-root .bk-menu-icon{width:28px;height:28px;background-size:60%;background-color:transparent;background-repeat:no-repeat;background-position:center center;}.bk-root .bk-context-menu{position:absolute;display:inline-flex;flex-wrap:nowrap;user-select:none;-ms-user-select:none;-moz-user-select:none;-webkit-user-select:none;width:auto;height:auto;z-index:100;cursor:pointer;font-size:12px;background-color:#fff;border:1px solid #ccc;border-radius:4px;box-shadow:0 6px 12px rgba(0, 0, 0, 0.175);}.bk-root .bk-context-menu.bk-horizontal{flex-direction:row;}.bk-root .bk-context-menu.bk-vertical{flex-direction:column;}.bk-root .bk-context-menu > .bk-divider{cursor:default;overflow:hidden;background-color:#e5e5e5;}.bk-root .bk-context-menu.bk-horizontal > .bk-divider{width:1px;margin:5px 0;}.bk-root .bk-context-menu.bk-vertical > .bk-divider{height:1px;margin:0 5px;}.bk-root .bk-context-menu > :not(.bk-divider){border:1px solid transparent;}.bk-root .bk-context-menu > :not(.bk-divider).bk-active{border-color:#26aae1;}.bk-root .bk-context-menu > :not(.bk-divider):hover{background-color:#f9f9f9;}.bk-root .bk-context-menu > :not(.bk-divider):focus,.bk-root .bk-context-menu > :not(.bk-divider):focus-visible{outline:1px dotted #26aae1;outline-offset:-1px;}.bk-root .bk-context-menu > :not(.bk-divider)::-moz-focus-inner{border:0;}.bk-root .bk-context-menu.bk-horizontal > :not(.bk-divider):first-child{border-top-left-radius:4px;border-bottom-left-radius:4px;}.bk-root .bk-context-menu.bk-horizontal > :not(.bk-divider):last-child{border-top-right-radius:4px;border-bottom-right-radius:4px;}.bk-root .bk-context-menu.bk-vertical > :not(.bk-divider):first-child{border-top-left-radius:4px;border-top-right-radius:4px;}.bk-root .bk-context-menu.bk-vertical > :not(.bk-divider):last-child{border-bottom-left-radius:4px;border-bottom-right-radius:4px;}.bk-root .bk-menu{position:absolute;left:0;width:100%;z-index:100;cursor:pointer;font-size:12px;background-color:#fff;border:1px solid #ccc;border-radius:4px;box-shadow:0 6px 12px rgba(0, 0, 0, 0.175);}.bk-root .bk-menu.bk-above{bottom:100%;}.bk-root .bk-menu.bk-below{top:100%;}.bk-root .bk-menu > .bk-divider{height:1px;margin:7.5px 0;overflow:hidden;background-color:#e5e5e5;}.bk-root .bk-menu > :not(.bk-divider){padding:6px 12px;}.bk-root .bk-menu > :not(.bk-divider):hover,.bk-root .bk-menu > :not(.bk-divider).bk-active{background-color:#e6e6e6;}.bk-root .bk-caret{display:inline-block;vertical-align:middle;width:0;height:0;margin:0 5px;}.bk-root .bk-caret.bk-down{border-top:4px solid;}.bk-root .bk-caret.bk-up{border-bottom:4px solid;}.bk-root .bk-caret.bk-down,.bk-root .bk-caret.bk-up{border-right:4px solid transparent;border-left:4px solid transparent;}.bk-root .bk-caret.bk-left{border-right:4px solid;}.bk-root .bk-caret.bk-right{border-left:4px solid;}.bk-root .bk-caret.bk-left,.bk-root .bk-caret.bk-right{border-top:4px solid transparent;border-bottom:4px solid transparent;}.bk-root{}.bk-root .bk-tooltip{font-weight:300;font-size:12px;position:absolute;padding:5px;border:1px solid #e5e5e5;color:#2f2f2f;background-color:white;pointer-events:none;opacity:0.95;z-index:100;}.bk-root .bk-tooltip > div:not(:first-child){margin-top:5px;border-top:#e5e5e5 1px dashed;}.bk-root .bk-tooltip.bk-left.bk-tooltip-arrow::before{position:absolute;margin:-7px 0 0 0;top:50%;width:0;height:0;border-style:solid;border-width:7px 0 7px 0;border-color:transparent;content:" ";display:block;left:-10px;border-right-width:10px;border-right-color:#909599;}.bk-root .bk-tooltip.bk-left::before{left:-10px;border-right-width:10px;border-right-color:#909599;}.bk-root .bk-tooltip.bk-right.bk-tooltip-arrow::after{position:absolute;margin:-7px 0 0 0;top:50%;width:0;height:0;border-style:solid;border-width:7px 0 7px 0;border-color:transparent;content:" ";display:block;right:-10px;border-left-width:10px;border-left-color:#909599;}.bk-root .bk-tooltip.bk-right::after{right:-10px;border-left-width:10px;border-left-color:#909599;}.bk-root .bk-tooltip.bk-above::before{position:absolute;margin:0 0 0 -7px;left:50%;width:0;height:0;border-style:solid;border-width:0 7px 0 7px;border-color:transparent;content:" ";display:block;top:-10px;border-bottom-width:10px;border-bottom-color:#909599;}.bk-root .bk-tooltip.bk-below::after{position:absolute;margin:0 0 0 -7px;left:50%;width:0;height:0;border-style:solid;border-width:0 7px 0 7px;border-color:transparent;content:" ";display:block;bottom:-10px;border-top-width:10px;border-top-color:#909599;}.bk-root .bk-tooltip-row-label{text-align:right;color:#26aae1;}.bk-root .bk-tooltip-row-value{color:default;}.bk-root .bk-tooltip-color-block{width:12px;height:12px;margin-left:5px;margin-right:5px;outline:#dddddd solid 1px;display:inline-block;}
Providing data — Bokeh 2.4.3 Documentation
Providing data
The basis for any data visualization is the underlying data. This section
describes the various ways to provide data to Bokeh, from passing data values
directly to creating a ColumnDataSource (CDS) and filtering the data with a
CDSView
.
Providing data with Python lists
Use standard Python lists of data to pass values directly into a plotting
function.
In this example, the lists x_values
and y_values
pass data
to the circle()
function (see
plotting function for more examples):
from bokeh.plotting import figure
x_values = [ 1 , 2 , 3 , 4 , 5 ]
y_values = [ 6 , 7 , 2 , 3 , 6 ]
p = figure ()
p . circle ( x = x_values , y = y_values )
Copy to clipboard
Providing NumPy data
Similarly to using Python lists and arrays, you can also work with NumPy data
structures in Bokeh:
import numpy as np
from bokeh.plotting import figure
x = [ 1 , 2 , 3 , 4 , 5 ]
random = np . random . standard_normal ( 5 )
cosine = np . cos ( x )
p = figure ()
p . circle ( x = x , y = random )
p . line ( x = x , y = cosine )
Copy to clipboard
Providing data as a ColumnDataSource
The ColumnDataSource (CDS) is the core of most Bokeh plots. It provides the
data to the glyphs of your plot.
When you pass sequences like Python lists or NumPy arrays to a Bokeh renderer,
Bokeh automatically creates a ColumnDataSource with this data for you. However,
creating a ColumnDataSource yourself gives you access to more advanced options.
For example: Creating your own ColumnDataSource allows you to share data
between multiple plots and widgets. If you use a single ColumnDataSource
together with multiple renderers, those renderers also share information about
data you select with a select tool from Bokeh’s toolbar (see
Linked selection ).
Think of a ColumnDataSource as a collection of sequences of data that each have
their own, unique column name.
Creating a ColumnDataSource
To create a basic ColumnDataSource object, you need a Python dictionary to
pass to the object’s data
parameter:
The data you pass as part of your dict can be any non-string ordered sequences
of values, such as lists or arrays (including NumPy arrays and pandas Series):
data = { 'x_values' : [ 1 , 2 , 3 , 4 , 5 ],
'y_values' : [ 6 , 7 , 2 , 3 , 6 ]}
source = ColumnDataSource ( data = data )
Copy to clipboard
Note
All columns in a ColumnDataSource have the same length. Therefore, all sequences
of values that you pass to a single ColumnDataSource must have the
same length as well. If you try to pass sequences of different lengths, Bokeh
will not be able to create your ColumnDataSource.
Plotting with a ColumnDataSource
To use a ColumnDataSource with a renderer function, you need to pass at least
these three arguments:
x
: the name of the ColumnDataSource’s column that contains the data for
the x values of your plot
y
: the name of the ColumnDataSource’s column that contains the data for
the y values of your plot
source
: the name of the ColumnDataSource that contains the columns you
just referenced for the x
and y
arguments.
For example:
from bokeh.plotting import figure
from bokeh.models import ColumnDataSource
# create a Python dict as the basis of your ColumnDataSource
data = { 'x_values' : [ 1 , 2 , 3 , 4 , 5 ],
'y_values' : [ 6 , 7 , 2 , 3 , 6 ]}
# create a ColumnDataSource by passing the dict
source = ColumnDataSource ( data = data )
# create a plot using the ColumnDataSource's two columns
p = figure ()
p . circle ( x = 'x_values' , y = 'y_values' , source = source )
Copy to clipboard
Modifying a ColumnDataSource
To modify the data of an existing ColumnDataSource, update the .data
property of your ColumnDataSource object:
To add a new column to an existing ColumnDataSource:
new_sequence = [ 8 , 1 , 4 , 7 , 3 ]
source . data [ "new_column" ] = new_sequence
Copy to clipboard
Note
The length of the column you are adding must match the length of the
existing columns.
To replace all data in an existing ColumnDataSource, assign the .data
property an entirely new dict:
source . data = new_dict
Copy to clipboard
Note
Replacing the entire contents of a ColumnDataSource is also the only way to
update the lengths of its columns. When you update data in a way that
changes the length of any column, you must update all columns at the same
time by passing an new dict. It is not possible to update column lengths one
column at a time.
Using a pandas DataFrame
The data
parameter can also be a pandas DataFrame
or GroupBy
object:
source = ColumnDataSource ( df )
Copy to clipboard
If you use a pandas DataFrame
, the resulting ColumnDataSource in Bokeh will
have columns that correspond to the columns of the DataFrame
. The naming of
the columns follows these rules:
If the DataFrame
has a named index column, the ColumnDataSource will also
have a column with this name.
If the index name is None
, the ColumnDataSource will have a generic name:
either index
(if that name is available) or level_0
.
Using a pandas MultiIndex
If you use a pandas MultiIndex
as the basis for a Bokeh
ColumnDataSource
, Bokeh flattens the columns and indices before creating
the ColumnDataSource. For the index, Bokeh creates an index of tuples and joins
the names of the MultiIndex
with an underscore. The column names will also be
joined with an underscore. For example:
df = pd . DataFrame ({( 'a' , 'b' ): {( 'A' , 'B' ): 1 , ( 'A' , 'C' ): 2 },
( 'b' , 'a' ): {( 'A' , 'C' ): 7 , ( 'A' , 'B' ): 8 },
( 'b' , 'b' ): {( 'A' , 'D' ): 9 , ( 'A' , 'B' ): 10 }})
cds = ColumnDataSource ( df )
Copy to clipboard
This will result in a column named index
with [(A, B), (A, C), (A, D)]
,
as well as columns named a_b
, b_a
, and b_b
.
This process only works with column names that are strings. If you are using
non-string column names, you need to manually flatten the DataFrame
before
using it as the basis of a Bokeh ColumnDataSource
.
Using pandas GroupBy
group = df . groupby (( 'colA' , 'ColB' ))
source = ColumnDataSource ( group )
Copy to clipboard
If you use a pandas GroupBy
object, the columns of the ColumnDataSource
correspond to the result of calling group.describe()
. The describe
method generates columns for statistical measures such as mean
and count
for all the non-grouped original columns.
The resulting DataFrame
has MultiIndex
columns with the original column
name and the computed measure. Bokeh flattens the data using the rules described
above.
For example: If a DataFrame
has the columns 'year'
and 'mpg'
,
passing df.groupby('year')
to a ColumnDataSource will result in columns such
as 'mpg_mean'
.
Note
Adapting GroupBy
objects requires pandas version 0.20.0 or above.
Appending data to a ColumnDataSource
ColumnDataSource streaming is an efficient way to append new data to a
ColumnDataSource. When you use the
stream()
method, Bokeh only sends
new data to the browser instead of sending the entire dataset.
The stream()
method takes a
new_data
parameter. This parameter expects a dict that maps column names
to the sequences of data that you want appended to the respective columns.
The method takes an additional, optional argument rollover
. This is the
maximum length of data to keep. When there is more data than defined by your
maximum value, Bokeh will discard data from the beginning of the column. The
default value for rollover
is None
. This default value allows data to
grow unbounded.
source = ColumnDataSource ( data = dict ( foo = [], bar = []))
# has new, identical-length updates for all columns in source
new_data = {
'foo' : [ 10 , 20 ],
'bar' : [ 100 , 200 ],
}
source . stream ( new_data )
Copy to clipboard
For an example that uses streaming, see examples/app/ohlc .
Replacing data in a ColumnDataSource
ColumnDataSource patching is an efficient way to update slices of a data
source. By using the patch()
method, Bokeh only sends new data to the browser instead of the entire
dataset.
The patch()
requires a dict which
maps column names to list of tuples that represent a patch change to apply.
Examples of tuples that you can use with
patch()
:
( index , new_value ) # replace a single column value
# or
( slice , new_values ) # replace several column values
Copy to clipboard
For a full example, see examples/howto/patch_app.py .
Filtering data
Bokeh uses a concept called “view” to select subsets of data. Views are
represented by Bokeh’s CDSView
class. When you use a view, you can use one or
more filters to select specific data points without changing the underlying
data. You can also share those views between different plots.
To plot with a filtered subset of data, pass a CDSView
to the view
argument of any renderer method on a Bokeh plot.
A CDSView
has two properties, source
and filters
:
source
is the ColumnDataSource that the you want to apply the filters
to.
filters
is a list of Filter
objects, listed and described below.
In this example, you create a CDSView
called view
. view
uses the
ColumnDataSource source
and a list of two filters, filter1
and
filter2
. view
is then passed to a circle()
renderer function:
from bokeh.plotting import figure
from bokeh.models import ColumnDataSource , CDSView
source = ColumnDataSource ( some_data )
view = CDSView ( source = source , filters = [ filter1 , filter2 ])
p = figure ()
p . circle ( x = "x" , y = "y" , source = source , view = view )
Copy to clipboard
IndexFilter
The IndexFilter
is the simplest filter type. It has an indices
property,
which is a list of integers that are the indices of the data you want to include
in your plot.
from bokeh.layouts import gridplot
from bokeh.models import CDSView , ColumnDataSource , IndexFilter
from bokeh.plotting import figure , show
source = ColumnDataSource ( data = dict ( x = [ 1 , 2 , 3 , 4 , 5 ], y = [ 1 , 2 , 3 , 4 , 5 ]))
view = CDSView ( source = source , filters = [ IndexFilter ([ 0 , 2 , 4 ])])
tools = [ "box_select" , "hover" , "reset" ]
p = figure ( height = 300 , width = 300 , tools = tools )
p . circle ( x = "x" , y = "y" , size = 10 , hover_color = "red" , source = source )
p_filtered = figure ( height = 300 , width = 300 , tools = tools )
p_filtered . circle ( x = "x" , y = "y" , size = 10 , hover_color = "red" , source = source , view = view )
show ( gridplot ([[ p , p_filtered ]]))
Copy to clipboard
BooleanFilter
A BooleanFilter
selects rows from a data source using a list of True
or
False
values in its booleans
property.
from bokeh.layouts import gridplot
from bokeh.models import BooleanFilter , CDSView , ColumnDataSource
from bokeh.plotting import figure , show
source = ColumnDataSource ( data = dict ( x = [ 1 , 2 , 3 , 4 , 5 ], y = [ 1 , 2 , 3 , 4 , 5 ]))
booleans = [ True if y_val > 2 else False for y_val in source . data [ 'y' ]]
view = CDSView ( source = source , filters = [ BooleanFilter ( booleans )])
tools = [ "box_select" , "hover" , "reset" ]
p = figure ( height = 300 , width = 300 , tools = tools )
p . circle ( x = "x" , y = "y" , size = 10 , hover_color = "red" , source = source )
p_filtered = figure ( height = 300 , width = 300 , tools = tools ,
x_range = p . x_range , y_range = p . y_range )
p_filtered . circle ( x = "x" , y = "y" , size = 10 , hover_color = "red" , source = source , view = view )
show ( gridplot ([[ p , p_filtered ]]))
Copy to clipboard
GroupFilter
The GroupFilter
is a filter for categorical data. With this filter, you can
select rows from a dataset that are members of a specific category.
The GroupFilter
has two properties:
In the example below, the data set flowers
contains a categorical variable
called species
. All data belongs to one of the three species categories
setosa
, versicolor
, or virginica
. The second plot in this example
uses a GroupFilter
to only display data points that are a member of the
category setosa
:
from bokeh.layouts import gridplot
from bokeh.models import CDSView , ColumnDataSource , GroupFilter
from bokeh.plotting import figure , show
from bokeh.sampledata.iris import flowers
source = ColumnDataSource ( flowers )
view1 = CDSView ( source = source , filters = [ GroupFilter ( column_name = 'species' , group = 'versicolor' )])
plot_size_and_tools = { 'height' : 300 , 'width' : 300 ,
'tools' :[ 'box_select' , 'reset' , 'help' ]}
p1 = figure ( title = "Full data set" , ** plot_size_and_tools )
p1 . circle ( x = 'petal_length' , y = 'petal_width' , source = source , color = 'black' )
p2 = figure ( title = "Setosa only" , x_range = p1 . x_range , y_range = p1 . y_range , ** plot_size_and_tools )
p2 . circle ( x = 'petal_length' , y = 'petal_width' , source = source , view = view1 , color = 'red' )
show ( gridplot ([[ p1 , p2 ]]))
Copy to clipboard
CustomJSFilter
You can also use your own JavaScript or TypeScript code to create customized
filters. To include your custom filter code, use Bokeh’s CustomJSFilter
class.
Pass your code as a string to the parameter code
of the CustomJSFilter.
Your JavaScript or TypeScript code needs to return either a list of indices or a
list of booleans representing the filtered subset. You can access the
ColumnDataSource you are using with CDSView
from within your JavaScript or
TypeScript code. Bokeh makes the ColumnDataSource available through the variable
source
:
custom_filter = CustomJSFilter ( code = '''
const indices = [];
// iterate through rows of data source and see if each satisfies some constraint
for (let i = 0; i < source.get_length(); i++){
if (source.data['some_column'][i] == 'some_value'){
indices.push(true);
} else {
indices.push(false);
}
}
return indices;
''' )
Copy to clipboard
AjaxDataSource
Updating and streaming data works very well with
Bokeh server applications . However, it is also possible
to use similar functionality in standalone documents. The
AjaxDataSource
provides this capability without
requiring a Bokeh server.
To set up an AjaxDataSource
, you need to configure it with a URL to a REST
endpoint and a polling interval.
In the browser, the data source requests data from the endpoint at the specified
interval. It then uses the data from the endpoint to update the data locally.
Updating data locally can happen in two ways: either by replacing the existing
local data entirely or by appending the new data to the existing data (up to a
configurable max_size
). Replacing local data is the default setting. Pass
either "replace"
or "append"``as the AjaxDataSource's ``mode
argument to
control this behavior.
The endpoint that you are using with your AjaxDataSource
needs to return a
JSON dict that matches the standard ColumnDataSource format:
{
'x' : [ 1 , 2 , 3 , ... ],
'y' : [ 9 , 3 , 2 , ... ]
}
Copy to clipboard
Otherwise, using an AjaxDataSource
is identical to using a standard
ColumnDataSource
:
# setup AjaxDataSource with URL and polling interval
source = AjaxDataSource ( data_url = 'http://some.api.com/data' ,
polling_interval = 100 )
# use the AjaxDataSource just like a ColumnDataSource
p . circle ( 'x' , 'y' , source = source )
Copy to clipboard
This a preview of what a stream of live data in Bokeh can look like using
AjaxDataSource
:
For the full example, see examples/howto/ajax_source.py in Bokeh’s
GitHub repository.
Linked selection
You can share selections between two plots if both of the plots use the same
ColumnDataSource :
from bokeh.io import output_file , show
from bokeh.layouts import gridplot
from bokeh.models import ColumnDataSource
from bokeh.plotting import figure
output_file ( "brushing.html" )
x = list ( range ( - 20 , 21 ))
y0 = [ abs ( xx ) for xx in x ]
y1 = [ xx ** 2 for xx in x ]
# create a column data source for the plots to share
source = ColumnDataSource ( data = dict ( x = x , y0 = y0 , y1 = y1 ))
TOOLS = "box_select,lasso_select,help"
# create a new plot and add a renderer
left = figure ( tools = TOOLS , width = 300 , height = 300 , title = None )
left . circle ( 'x' , 'y0' , source = source )
# create another new plot and add a renderer
right = figure ( tools = TOOLS , width = 300 , height = 300 , title = None )
right . circle ( 'x' , 'y1' , source = source )
p = gridplot ([[ left , right ]])
show ( p )
Copy to clipboard
Linked selection with filtered data
Using a ColumnDataSource , you can also have two plots that are based on the
same data but each use a different subset of that data. Both plots still share
selections and hovered inspections through the ColumnDataSource they are based
on.
The following example demonstrates this behavior:
The second plot is a subset of the data of the first plot. The second plot
uses a CDSView
to include only y values that are either greater than 250 or
less than 100.
If you make a selection with the BoxSelect
tool in either plot, the
selection is automatically reflected in the other plot as well.
If you hover on a point in one plot, the corresponding point in the other plot
is automatically highlighted as well, if it exists.
from bokeh.layouts import gridplot
from bokeh.models import BooleanFilter , CDSView , ColumnDataSource
from bokeh.plotting import figure , output_file , show
output_file ( "linked_selection_subsets.html" )
x = list ( range ( - 20 , 21 ))
y0 = [ abs ( xx ) for xx in x ]
y1 = [ xx ** 2 for xx in x ]
# create a column data source for the plots to share
source = ColumnDataSource ( data = dict ( x = x , y0 = y0 , y1 = y1 ))
# create a view of the source for one plot to use
view = CDSView ( source = source , filters = [ BooleanFilter ([ True if y > 250 or y < 100 else False for y in y1 ])])
TOOLS = "box_select,lasso_select,hover,help"
# create a new plot and add a renderer
left = figure ( tools = TOOLS , width = 300 , height = 300 , title = None )
left . circle ( 'x' , 'y0' , size = 10 , hover_color = "firebrick" , source = source )
# create another new plot, add a renderer that uses the view of the data source
right = figure ( tools = TOOLS , width = 300 , height = 300 , title = None )
right . circle ( 'x' , 'y1' , size = 10 , hover_color = "firebrick" , source = source , view = view )
p = gridplot ([[ left , right ]])
show ( p )
Copy to clipboard
Other data types
You can also use Bokeh to render network graph data and geographical data. For
more information about how to set up the data for these types of plots, see
Visualizing network graphs and Mapping geo data .