From 4e4ccb17cc121dbc5768f663f8f90b9b9638110c Mon Sep 17 00:00:00 2001 From: Ryan Patrick Kyle Date: Fri, 28 Aug 2020 10:28:48 -0400 Subject: [PATCH] Add support for user-defined server routes (#225) --- CHANGELOG.md | 4 + R/dash.R | 162 +++++++++++++++++++++++++++++- man/Dash.Rd | 186 +++++++++++++++++++++++++++++++++++ tests/testthat/test-routes.R | 126 ++++++++++++++++++++++++ 4 files changed, 476 insertions(+), 2 deletions(-) create mode 100644 tests/testthat/test-routes.R diff --git a/CHANGELOG.md b/CHANGELOG.md index 710a336d..76232b78 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,10 @@ All notable changes to `dash` will be documented in this file. This project adheres to [Semantic Versioning](http://semver.org/). +## [Unreleased] +### Added +- Dash for R now supports user-defined routes and redirects via the `app$server_route` and `app$redirect` methods. [#225](https://github.com/plotly/dashR/pull/225) + ## [0.7.1] - 2020-07-30 ### Fixed - Fixes a minor bug in debug mode that prevented display of user-defined error messages when induced by invoking the `stop` function. [#220](https://github.com/plotly/dashR/pull/220). diff --git a/R/dash.R b/R/dash.R index 8be1d382..c3a934cd 100644 --- a/R/dash.R +++ b/R/dash.R @@ -103,12 +103,13 @@ Dash <- R6::R6Class( self$config$external_stylesheets <- external_stylesheets self$config$show_undo_redo <- show_undo_redo self$config$update_title <- update_title - + # ------------------------------------------------------------ # Initialize a route stack and register a static resource route # ------------------------------------------------------------ router <- routr::RouteStack$new() - + server$set_data("user-routes", list()) # placeholder for custom routes + # ensure that assets_folder is neither NULL nor character(0) if (!(is.null(private$assets_folder)) & length(private$assets_folder) != 0) { if (!(dir.exists(private$assets_folder)) && gsub("/+", "", assets_folder) != "assets") { @@ -550,6 +551,130 @@ Dash <- R6::R6Class( self$server <- server }, + # ------------------------------------------------------------------------ + # methods to add custom server routes + # ------------------------------------------------------------------------ + #' @description + #' Connect a URL to a custom server route + #' @details + #' `fiery`, the underlying web service framework upon which Dash for R is based, + #' supports custom routing through plugins. While convenient, the plugin API + #' providing this functionality is different from that provided by Flask, as + #' used by Dash for Python. This method wraps the pluggable routing of `routr` + #' routes in a manner that should feel slightly more idiomatic to Dash users. + #' ## Querying User-Defined Routes: + #' It is possible to retrieve the list of user-defined routes by invoking the + #' `get_data` method. For example, if your Dash application object is `app`, use + #' `app$server$get_data("user-routes")`. + #' + #' If you wish to erase all user-defined routes without instantiating a new Dash + #' application object, one option is to clear the routes manually: + #' `app$server$set_data("user-routes", list())`. + #' @param path Character. Represents a URL path comprised of strings, parameters + #' (strings prefixed with :), and wildcards (*), separated by /. Wildcards can + #' be used to match any path element, rather than restricting (as by default) to + #' a single path element. For example, it is possible to catch requests to multiple + #' subpaths using a wildcard. For more information, see \link{Route}. + #' @param handler Function. Adds a handler function to the specified method and path. + #' For more information, see \link{Route}. + #' @param methods Character. A string indicating the request method (in lower case, + #' e.g. 'get', 'put', etc.), as used by `reqres`. The default is `get`. + #' For more information, see \link{Route}. + #' @examples + #' library(dash) + #' app <- Dash$new() + #' + #' # A handler to redirect requests with `307` status code (temporary redirects); + #' # for permanent redirects (`301`), see the `redirect` method described below + #' # + #' # A simple single path-to-path redirect + #' app$server_route('/getting-started', function(request, response, keys, ...) { + #' response$status <- 307L + #' response$set_header('Location', '/layout') + #' TRUE + #' }) + #' + #' # Example of a redirect with a wildcard for subpaths + #' app$server_route('/getting-started/*', function(request, response, keys, ...) { + #' response$status <- 307L + #' response$set_header('Location', '/layout') + #' TRUE + #' }) + #' + #' # Example of a parameterized redirect with wildcard for subpaths + #' app$server_route('/accounts/:user_id/*', function(request, response, keys, ...) { + #' response$status <- 307L + #' response$set_header('Location', paste0('/users/', keys$user_id)) + #' TRUE + #' }) + server_route = function(path = NULL, handler = NULL, methods = "get") { + if (is.null(path) || is.null(handler)) { + stop("The server_route method requires that a path and handler function are specified. Please ensure these arguments are non-missing.", call.=FALSE) + } + + user_routes <- self$server$get_data("user-routes") + + user_routes[[path]] <- list("path" = path, + "handler" = handler, + "methods" = methods) + + self$server$set_data("user-routes", user_routes) + }, + + #' @description + #' Redirect a Dash application URL path + #' @details + #' This is a convenience method to simplify adding redirects + #' for your Dash application which automatically return a `301` + #' HTTP status code and direct the client to load an alternate URL. + #' @param old_path Character. Represents the URL path to redirect, + #' comprised of strings, parameters (strings prefixed with :), and + #' wildcards (*), separated by /. Wildcards can be used to match any + #' path element, rather than restricting (as by default) to a single + #' path element. For example, it is possible to catch requests to multiple + #' subpaths using a wildcard. For more information, see \link{Route}. + #' @param new_path Character or function. Same as `old_path`, but represents the + #' new path which the client should load instead. If a function is + #' provided instead of a string, it should have `keys` within its formals. + #' @param methods Character. A string indicating the request method + #' (in lower case, e.g. 'get', 'put', etc.), as used by `reqres`. The + #' default is `get`. For more information, see \link{Route}. + #' @examples + #' library(dash) + #' app <- Dash$new() + #' + #' # example of a simple single path-to-path redirect + #' app$redirect("/getting-started", "/layout") + #' + #' # example of a redirect using wildcards + #' app$redirect("/getting-started/*", "/layout/*") + #' + #' # example of a parameterized redirect using a function for new_path, + #' # which requires passing in keys to take advantage of subpaths within + #' # old_path that are preceded by a colon (e.g. :user_id): + #' app$redirect("/accounts/:user_id/*", function(keys) paste0("/users/", keys$user_id)) + redirect = function(old_path = NULL, new_path = NULL, methods = "get") { + if (is.null(old_path) || is.null(new_path)) { + stop("The redirect method requires that both an old path and a new path are specified. Please ensure these arguments are non-missing.", call.=FALSE) + } + + if (is.function(new_path)) { + handler <- function(request, response, keys, ...) { + response$status <- 301L + response$set_header('Location', new_path(keys)) + TRUE + } + } else { + handler <- function(request, response, keys, ...) { + response$status <- 301L + response$set_header('Location', new_path) + TRUE + } + } + + self$server_route(old_path, handler) + }, + # ------------------------------------------------------------------------ # dash layout methods # ------------------------------------------------------------------------ @@ -1029,6 +1154,39 @@ Dash <- R6::R6Class( private$prune_errors <- getServerParam(dev_tools_prune_errors, "logical", TRUE) + # attach user-defined routes, if they exist + if (length(self$server$get_data("user-routes")) > 0) { + + plugin <- list( + on_attach = function(server) { + user_routes <- server$get_data("user-routes") + + # adding an additional route will fail if the + # route already exists, so remove user-routes + # if present and reload; user_routes will still + # have all the relevant routes in place anyhow + if (server$plugins$request_routr$has_route("user-routes")) + server$plugins$request_routr$remove_route("user-routes") + + router <- server$plugins$request_routr + + route <- routr::Route$new() + + for (routing in user_routes) { + route$add_handler(method=routing$methods, + path=routing$path, + handler=routing$handler) + } + + router$add_route(route, "user-routes") + }, + name = "user_routes", + require = "request_routr" + ) + + self$server$attach(plugin, force = TRUE) + } + if(getAppPath() != FALSE) { source_dir <- dirname(getAppPath()) private$app_root_modtime <- modtimeFromPath(source_dir, recursive = TRUE, asset_path = private$assets_folder) diff --git a/man/Dash.Rd b/man/Dash.Rd index d9e57875..32c10d05 100644 --- a/man/Dash.Rd +++ b/man/Dash.Rd @@ -12,6 +12,55 @@ A framework for building analytical web applications, Dash offers a pleasant and } \examples{ +## ------------------------------------------------ +## Method `Dash$server_route` +## ------------------------------------------------ + +library(dash) +app <- Dash$new() + +# A handler to redirect requests with `307` status code (temporary redirects); +# for permanent redirects (`301`), see the `redirect` method described below +# +# A simple single path-to-path redirect +app$server_route('/getting-started', function(request, response, keys, ...) { + response$status <- 307L + response$set_header('Location', '/layout') + TRUE +}) + +# Example of a redirect with a wildcard for subpaths +app$server_route('/getting-started/*', function(request, response, keys, ...) { + response$status <- 307L + response$set_header('Location', '/layout') + TRUE +}) + +# Example of a parameterized redirect with wildcard for subpaths +app$server_route('/accounts/:user_id/*', function(request, response, keys, ...) { + response$status <- 307L + response$set_header('Location', paste0('/users/', keys$user_id)) + TRUE +}) + +## ------------------------------------------------ +## Method `Dash$redirect` +## ------------------------------------------------ + +library(dash) +app <- Dash$new() + +# example of a simple single path-to-path redirect +app$redirect("/getting-started", "/layout") + +# example of a redirect using wildcards +app$redirect("/getting-started/*", "/layout/*") + +# example of a parameterized redirect using a function for new_path, +# which requires passing in keys to take advantage of subpaths within +# old_path that are preceded by a colon (e.g. :user_id): +app$redirect("/accounts/:user_id/*", function(keys) paste0("/users/", keys$user_id)) + ## ------------------------------------------------ ## Method `Dash$interpolate_index` ## ------------------------------------------------ @@ -97,6 +146,8 @@ where the application is making API calls.} \subsection{Public methods}{ \itemize{ \item \href{#method-new}{\code{Dash$new()}} +\item \href{#method-server_route}{\code{Dash$server_route()}} +\item \href{#method-redirect}{\code{Dash$redirect()}} \item \href{#method-layout_get}{\code{Dash$layout_get()}} \item \href{#method-layout}{\code{Dash$layout()}} \item \href{#method-react_version_set}{\code{Dash$react_version_set()}} @@ -197,6 +248,141 @@ clientside callback.} } \if{html}{\out{}} } +} +\if{html}{\out{
}} +\if{html}{\out{}} +\if{latex}{\out{\hypertarget{method-server_route}{}}} +\subsection{Method \code{server_route()}}{ +Connect a URL to a custom server route +\subsection{Usage}{ +\if{html}{\out{
}}\preformatted{Dash$server_route(path = NULL, handler = NULL, methods = "get")}\if{html}{\out{
}} +} + +\subsection{Arguments}{ +\if{html}{\out{
}} +\describe{ +\item{\code{path}}{Character. Represents a URL path comprised of strings, parameters +(strings prefixed with :), and wildcards (*), separated by /. Wildcards can +be used to match any path element, rather than restricting (as by default) to +a single path element. For example, it is possible to catch requests to multiple +subpaths using a wildcard. For more information, see \link{Route}.} + +\item{\code{handler}}{Function. Adds a handler function to the specified method and path. +For more information, see \link{Route}.} + +\item{\code{methods}}{Character. A string indicating the request method (in lower case, +e.g. 'get', 'put', etc.), as used by \code{reqres}. The default is \code{get}. +For more information, see \link{Route}.} +} +\if{html}{\out{
}} +} +\subsection{Details}{ +\code{fiery}, the underlying web service framework upon which Dash for R is based, +supports custom routing through plugins. While convenient, the plugin API +providing this functionality is different from that provided by Flask, as +used by Dash for Python. This method wraps the pluggable routing of \code{routr} +routes in a manner that should feel slightly more idiomatic to Dash users. +\subsection{Querying User-Defined Routes:}{ + +It is possible to retrieve the list of user-defined routes by invoking the +\code{get_data} method. For example, if your Dash application object is \code{app}, use +\code{app$server$get_data("user-routes")}. + +If you wish to erase all user-defined routes without instantiating a new Dash +application object, one option is to clear the routes manually: +\code{app$server$set_data("user-routes", list())}. +} +} + +\subsection{Examples}{ +\if{html}{\out{
}} +\preformatted{library(dash) +app <- Dash$new() + +# A handler to redirect requests with `307` status code (temporary redirects); +# for permanent redirects (`301`), see the `redirect` method described below +# +# A simple single path-to-path redirect +app$server_route('/getting-started', function(request, response, keys, ...) { + response$status <- 307L + response$set_header('Location', '/layout') + TRUE +}) + +# Example of a redirect with a wildcard for subpaths +app$server_route('/getting-started/*', function(request, response, keys, ...) { + response$status <- 307L + response$set_header('Location', '/layout') + TRUE +}) + +# Example of a parameterized redirect with wildcard for subpaths +app$server_route('/accounts/:user_id/*', function(request, response, keys, ...) { + response$status <- 307L + response$set_header('Location', paste0('/users/', keys$user_id)) + TRUE +}) +} +\if{html}{\out{
}} + +} + +} +\if{html}{\out{
}} +\if{html}{\out{}} +\if{latex}{\out{\hypertarget{method-redirect}{}}} +\subsection{Method \code{redirect()}}{ +Redirect a Dash application URL path +\subsection{Usage}{ +\if{html}{\out{
}}\preformatted{Dash$redirect(old_path = NULL, new_path = NULL, methods = "get")}\if{html}{\out{
}} +} + +\subsection{Arguments}{ +\if{html}{\out{
}} +\describe{ +\item{\code{old_path}}{Character. Represents the URL path to redirect, +comprised of strings, parameters (strings prefixed with :), and +wildcards (*), separated by /. Wildcards can be used to match any +path element, rather than restricting (as by default) to a single +path element. For example, it is possible to catch requests to multiple +subpaths using a wildcard. For more information, see \link{Route}.} + +\item{\code{new_path}}{Character or function. Same as \code{old_path}, but represents the +new path which the client should load instead. If a function is +provided instead of a string, it should have \code{keys} within its formals.} + +\item{\code{methods}}{Character. A string indicating the request method +(in lower case, e.g. 'get', 'put', etc.), as used by \code{reqres}. The +default is \code{get}. For more information, see \link{Route}.} +} +\if{html}{\out{
}} +} +\subsection{Details}{ +This is a convenience method to simplify adding redirects +for your Dash application which automatically return a \code{301} +HTTP status code and direct the client to load an alternate URL. +} + +\subsection{Examples}{ +\if{html}{\out{
}} +\preformatted{library(dash) +app <- Dash$new() + +# example of a simple single path-to-path redirect +app$redirect("/getting-started", "/layout") + +# example of a redirect using wildcards +app$redirect("/getting-started/*", "/layout/*") + +# example of a parameterized redirect using a function for new_path, +# which requires passing in keys to take advantage of subpaths within +# old_path that are preceded by a colon (e.g. :user_id): +app$redirect("/accounts/:user_id/*", function(keys) paste0("/users/", keys$user_id)) +} +\if{html}{\out{
}} + +} + } \if{html}{\out{
}} \if{html}{\out{}} diff --git a/tests/testthat/test-routes.R b/tests/testthat/test-routes.R new file mode 100644 index 00000000..af1ee6c9 --- /dev/null +++ b/tests/testthat/test-routes.R @@ -0,0 +1,126 @@ +context("routes") + +test_that("URLs are properly redirected with app$redirect", { + library(dashHtmlComponents) + + app <- Dash$new() + + app$redirect("/foo", "/") + app$redirect("/bar/*", "/foo") + app$redirect("/users/:user_id", function(keys) paste0("/accounts/", keys$user_id)) + + app$layout(htmlDiv( + "Hello world!" + ) + ) + + request_foo <- fiery::fake_request( + "http://127.0.0.1:8050/foo" + ) + + request_bar <- fiery::fake_request( + "http://127.0.0.1:8050/bar/foo" + ) + + request_fn <- fiery::fake_request( + "http://127.0.0.1:8050/users/johndoe" + ) + + # start up Dash briefly to load the routes + app$run_server(block=FALSE) + app$server$stop() + + response_foo <- app$server$test_request(request_foo) + response_bar <- app$server$test_request(request_bar) + response_fn <- app$server$test_request(request_fn) + + expect_equal( + response_foo$status, + 301L + ) + + expect_equal( + response_foo$headers$Location, + "/" + ) + + expect_equal( + response_bar$status, + 301L + ) + + expect_equal( + response_bar$headers$Location, + "/foo" + ) + + expect_equal( + response_fn$status, + 301L + ) + + expect_equal( + response_fn$headers$Location, + "/accounts/johndoe" + ) + +}) + +test_that("temporary redirection of URLs is possible with app$server_route", { + library(dashHtmlComponents) + + app <- Dash$new() + + app$server_route("/baz", function(request, response, keys, ...) { + response$status <- 307L + response$set_header("Location", "/") + TRUE + }) + + app$server_route("/qux/*", function(request, response, keys, ...) { + response$status <- 307L + response$set_header("Location", "/foo") + TRUE + }) + + app$layout(htmlDiv( + "Hello world!" + ) + ) + + request_baz <- fiery::fake_request( + "http://127.0.0.1:8050/baz" + ) + + request_qux <- fiery::fake_request( + "http://127.0.0.1:8050/qux/foo" + ) + + # start up Dash briefly to load the routes + app$run_server(block=FALSE) + app$server$stop() + + response_baz <- app$server$test_request(request_baz) + response_qux <- app$server$test_request(request_qux) + + expect_equal( + response_baz$status, + 307L + ) + + expect_equal( + response_baz$headers$Location, + "/" + ) + + expect_equal( + response_qux$status, + 307L + ) + + expect_equal( + response_qux$headers$Location, + "/foo" + ) + +})