Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Provide support for multiple outputs #119

Merged
merged 26 commits into from
Sep 3, 2019
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
a44050a
Provide support for no_update in Dash for R (#111)
rpkyle Aug 13, 2019
2990c72
:sparkles: add createCallbackId
Aug 17, 2019
d476c91
:sparkles: add support for multiple outputs
Aug 21, 2019
b274c58
:rotating_light: add checks for multi-output callbacks
Aug 23, 2019
d77a0ce
:rotating_light: add test for multiple outputs
Aug 23, 2019
d26c4a2
:hammer: replace regex with strsplit
Aug 26, 2019
981ca3b
:bug: rename idlist to output_ids
Aug 26, 2019
10a04f4
:pencil2: updated comments
Aug 26, 2019
16fee5e
:hammer: support partial updates
Aug 27, 2019
83d3501
:rotating_light: add test for partial outputs
Aug 27, 2019
f1f4724
:rotating_light: add check for duplicated outputs
Aug 27, 2019
508eeb1
:sparkles: add insertIntoCallbackMap
Aug 29, 2019
df19ffe
modify R :package: install process
rpkyle Aug 29, 2019
b08eab6
:see_no_evil: fix quotation marks
rpkyle Aug 29, 2019
30ba139
condense further, run R once to install :package:
rpkyle Aug 29, 2019
5521a93
:rotating_light: for output=list, output=output
Aug 30, 2019
000fcbb
:hammer: fix missing argument
Aug 30, 2019
64dcfe3
:rotating_light: add test for repeated outputs
Aug 30, 2019
923116b
:tshirt: remove whitespace
Aug 30, 2019
787024e
:pencil2: update CircleCI config for unit tests
Aug 30, 2019
8146e5c
:pencil2: fix indent
Aug 30, 2019
542af5d
:pencil2: address test loading issue
Aug 30, 2019
029af1c
:rotating_light: test 1, 2 DashNoUpdate() els
Sep 1, 2019
f663327
Merge branch 'dev' into 114-multiple-outputs
rpkyle Sep 1, 2019
8071866
:pencil2: add comments
Sep 3, 2019
32a2724
Merge branch '114-multiple-outputs' of github.com:plotly/dashR into 1…
Sep 3, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions NAMESPACE
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

S3method(print,dash_component)
export(Dash)
export(dashNoUpdate)
export(input)
export(output)
export(state)
Expand Down
50 changes: 41 additions & 9 deletions R/dash.R
Original file line number Diff line number Diff line change
Expand Up @@ -222,7 +222,7 @@ Dash <- R6::R6Class(
payload <- Map(function(callback_signature) {
list(
inputs=callback_signature$inputs,
output=paste0(callback_signature$output, collapse="."),
output=createCallbackId(callback_signature$output),
state=callback_signature$state
)
}, private$callback_map)
Expand Down Expand Up @@ -287,24 +287,56 @@ Dash <- R6::R6Class(
output_value <- getStackTrace(do.call(callback, callback_args),
debug = private$debug,
pruned_errors = private$pruned_errors)

# reset callback context
private$callback_context_ <- NULL

if (is.null(private$stack_message)) {
# inspect the output_value to determine whether any outputs have no_update
# objects within them; these should not be updated
if (length(output_value) == 1 && class(output_value) == "no_update") {
response$body <- character(1) # return empty string
response$status <- 204L
}
else if (is.null(private$stack_message)) {
# pass on output_value to encode_plotly in case there are dccGraph
# components which include Plotly.js figures for which we'll need to
# run plotly_build from the plotly package
output_value <- encode_plotly(output_value)

# have to format the response body like this
# https://github.com/plotly/dash/blob/064c811d/dash/dash.py#L562-L584
alexcjohnson marked this conversation as resolved.
Show resolved Hide resolved
resp <- list(
response = list(
props = setNames(list(output_value), gsub( "(^.+)(\\.)", "", request$body$output))
if (grepl("\\.\\.\\.", request$body$output)) {
# if multi-output callback, isolate the output IDs and properties
idmatch <- gregexpr("(?<=\\.\\.)(([^\\.]+))(?=\\.)", request$body$output, perl=TRUE)
propmatch <- gregexpr("(?<=\\.)(([^\\.]+))(?=\\.\\.)", request$body$output, perl=TRUE)
ids <- unlist(regmatches(request$body$output, idmatch))
props <- unlist(regmatches(request$body$output, propmatch))
alexcjohnson marked this conversation as resolved.
Show resolved Hide resolved

# prepare a response object which has list elements corresponding to ids
# which themselves contain named list elements corresponding to props
# then fill in nested list elements based on output_value

allprops <- setNames(vector("list", length(unique(ids))), unique(ids))

idmap <- setNames(ids, props)

for (id in unique(ids)) {
allprops[[id]] <- output_value[grep(id, ids)]
names(allprops[[id]]) <- names(idmap[which(idmap==id)])
}

resp <- list(
response = allprops,
multi = TRUE
)
} else {
resp <- list(
response = list(
props = setNames(list(output_value), gsub( "(^.+)(\\.)", "", request$body$output))
)
)
)
}

response$body <- to_JSON(resp)
response$status <- 200L
response$type <- 'json'
Expand Down Expand Up @@ -500,7 +532,7 @@ Dash <- R6::R6Class(
state <- params[vapply(params, function(x) 'state' %in% attr(x, "class"), FUN.VALUE=logical(1))]

# register the callback_map
private$callback_map[[paste(output$id, output$property, sep='.')]] <- list(
private$callback_map[[createCallbackId(output)]] <- list(
inputs=inputs,
output=output,
state=state,
Expand Down
12 changes: 11 additions & 1 deletion R/dependencies.R
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,13 @@
#' Use in conjunction with the `callback()` method from the [dash::Dash] class
#' to define the update logic in your application.
#'
#' The `dashNoUpdate()` function permits application developers to prevent a
#' single output from updating the layout. It has no formal arguments.
#'
#' @name dependencies
#' @param id a component id
#' @param property the component property to use


#' @rdname dependencies
#' @export
output <- function(id, property) {
Expand Down Expand Up @@ -44,3 +46,11 @@ dependency <- function(id = NULL, property = NULL) {
property = property
)
}

#' @rdname dependencies
#' @export
dashNoUpdate <- function() {
x <- list(NULL)
class(x) <- "no_update"
return(x)
}
47 changes: 44 additions & 3 deletions R/utils.R
Original file line number Diff line number Diff line change
Expand Up @@ -371,10 +371,22 @@ assert_valid_callbacks <- function(output, params, func) {
}

# Assert that the component ID as passed is a string.
if(!(is.character(output$id) & !grepl("^\\s*$", output$id) & !grepl("\\.", output$id))) {
stop(sprintf("Callback IDs must be (non-empty) character strings that do not contain one or more dots/periods. Please verify that the component ID is valid."), call. = FALSE)
# This function inspects the output object to see if its ID
# is a valid string.
validateOutput <- function(string) {
return((is.character(string[["id"]]) & !grepl("^\\s*$", string[["id"]]) & !grepl("\\.", string[["id"]])))
}

# Check if the callback uses multiple outputs
if (any(sapply(output, is.list))) {
invalid_callback_ID <- (!all(vapply(output, validateOutput, logical(1))))
} else {
invalid_callback_ID <- (!validateOutput(output))
}
if (invalid_callback_ID) {
stop(sprintf("Callback IDs must be (non-empty) character strings that do not contain one or more dots/periods. Please verify that the component ID is valid."), call. = FALSE)
}

# Assert that user_function is a valid function
if(!(is.function(func))) {
stop(sprintf("The callback method's 'func' parameter requires a function as its argument. Please verify that 'func' is a valid, executable R function."), call. = FALSE)
Expand All @@ -397,7 +409,22 @@ assert_valid_callbacks <- function(output, params, func) {

# Check that outputs are not inputs
# https://github.com/plotly/dash/issues/323
inputs_vs_outputs <- lapply(inputs, function(x) identical(x, output))

# helper function to permit same mapply syntax regardless
# of whether output is defined using output function or not
listWrap <- function(x){
if (!any(sapply(x, is.list))) {
return(list(x))
} else {
x
}
}

# determine whether any input matches the output, or outputs, if
# multiple callback scenario
inputs_vs_outputs <- mapply(function(inputObject, outputObject) {
identical(outputObject[["id"]], inputObject[["id"]]) & identical(outputObject[["property"]], inputObject[["property"]])
}, inputs, listWrap(output))

if(TRUE %in% inputs_vs_outputs) {
stop(sprintf("Circular input and output arguments were found. Please verify that callback outputs are not also input arguments."), call. = FALSE)
Expand Down Expand Up @@ -828,3 +855,17 @@ getDashMetadata <- function(pkgname) {
metadataFn <- as.vector(fnList[grepl("^\\.dash.+_js_metadata$", fnList)])
return(metadataFn)
}

createCallbackId <- function(output) {
# check if callback uses single output
if (!any(sapply(output, is.list))) {
id <- paste0(output, collapse=".")
} else {
# multi-output callback, concatenate
ids <- vapply(output, function(x) {
paste(x, collapse = ".")
}, character(1))
id <- paste0("..", paste0(ids, collapse="..."), "..")
}
return(id)
}
7 changes: 7 additions & 0 deletions man/dependencies.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

88 changes: 88 additions & 0 deletions tests/integration/callbacks/multiple_outputs.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,88 @@
from selenium.webdriver.support.select import Select
import time

app = """
library(dash)
library(dashHtmlComponents)
library(dashCoreComponents)
library(plotly)
library(dashTable)

app <- Dash$new()

app$layout(
htmlDiv(list(
htmlDiv(list(
htmlH1('Multi output example'),
dccDropdown(id='data-dropdown',
options = list(
list(label = 'Movies',
value = 'movies'),
list(label = 'Series',
value = 'series')
),
value = 'movies')
),
id = 'container',
style = list(
backgroundColor = '#ff998a'
)
),
htmlDiv(list(
htmlH2('Make a selection from the dropdown menu.',
id = 'text-box')
)
)
)
)
)

app$callback(output=list(
output(id='text-box', property='children'),
output(id='container', property='style')
),
params=list(
input(id='data-dropdown', property='value')
),
function(value) {
if (is.null(value)) {
return(dashNoUpdate())
alexcjohnson marked this conversation as resolved.
Show resolved Hide resolved
}

if (value == "series") {
style <- list(
backgroundColor = '#ff998a'
)
} else {
style <- list(
backgroundColor = '#fff289'
)
}

return(list(sprintf("You have chosen %s.", value),
style))
}
)

app$run_server(debug=TRUE)
"""


def test_rsnu001_multiple_outputs(dashr):
dashr.start_server(app)
dashr.find_element("#data-dropdown").click()
dashr.find_elements("div.VirtualizedSelectOption")[1].click()
dashr.wait_for_text_to_equal(
"#text-box",
"You have chosen series."
)
backgroundColor = dashr.find_element('#container').value_of_css_property("background-color")
assert backgroundColor == "rgba(255, 153, 138, 1)"
dashr.find_element("#data-dropdown").click()
dashr.find_elements("div.VirtualizedSelectOption")[0].click()
dashr.wait_for_text_to_equal(
"#text-box",
"You have chosen movies."
)
backgroundColor = dashr.find_element('#container').value_of_css_property("background-color")
assert backgroundColor == "rgba(255, 242, 137, 1)"
Loading