Skip to content

Provide support for script and stylesheet attributes #226

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

Merged
merged 11 commits into from
Sep 4, 2020
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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 CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ This project adheres to [Semantic Versioning](http://semver.org/).

## [Unreleased]
### Added
- Support for setting attributes on `external_scripts` and `external_stylesheets`, and validation for the parameters passed (attributes are verified, and elements that are lists themselves must be named). [#226](https://github.com/plotly/dashR/pull/226)
- 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
Expand Down
17 changes: 12 additions & 5 deletions R/dash.R
Original file line number Diff line number Diff line change
Expand Up @@ -45,9 +45,13 @@ Dash <- R6::R6Class(
#' @param requests_pathname_prefix Character. A prefix applied to request endpoints
#' made by Dash's front-end. Environment variable is `DASH_REQUESTS_PATHNAME_PREFIX`.
#' @param external_scripts List. An optional list of valid URLs from which
#' to serve JavaScript source for rendered pages.
#' to serve JavaScript source for rendered pages. Each entry can be a string (the URL)
#' or a list with `src` (the URL) and optionally other `<script>` tag attributes such
#' as `integrity` and `crossorigin`.
#' @param external_stylesheets List. An optional list of valid URLs from which
#' to serve CSS for rendered pages.
#' to serve CSS for rendered pages. Each entry can be a string (the URL) or a list
#' with `href` (the URL) and optionally other `<link>` tag attributes such as
#' `rel`, `integrity` and `crossorigin`.
#' @param compress Logical. Whether to try to compress files and data served by Fiery.
#' By default, `brotli` is attempted first, then `gzip`, then the `deflate` algorithm,
#' before falling back to `identity`.
Expand Down Expand Up @@ -103,7 +107,10 @@ Dash <- R6::R6Class(
self$config$external_stylesheets <- external_stylesheets
self$config$show_undo_redo <- show_undo_redo
self$config$update_title <- update_title


# ensure attributes are valid, if using a list within a list, elements are all named
assertValidExternals(scripts = external_scripts, stylesheets = external_stylesheets)

# ------------------------------------------------------------
# Initialize a route stack and register a static resource route
# ------------------------------------------------------------
Expand Down Expand Up @@ -1736,7 +1743,7 @@ Dash <- R6::R6Class(

# collect CSS assets from dependencies
if (!(is.null(private$asset_map$css))) {
css_assets <- generate_css_dist_html(href = paste0(private$assets_url_path, names(private$asset_map$css)),
css_assets <- generate_css_dist_html(tagdata = paste0(private$assets_url_path, names(private$asset_map$css)),
local = TRUE,
local_path = private$asset_map$css,
prefix = self$config$requests_pathname_prefix)
Expand All @@ -1754,7 +1761,7 @@ Dash <- R6::R6Class(
# collect JS assets from dependencies
#
if (!(is.null(private$asset_map$scripts))) {
scripts_assets <- generate_js_dist_html(href = paste0(private$assets_url_path, names(private$asset_map$scripts)),
scripts_assets <- generate_js_dist_html(tagdata = paste0(private$assets_url_path, names(private$asset_map$scripts)),
local = TRUE,
local_path = private$asset_map$scripts,
prefix = self$config$requests_pathname_prefix)
Expand Down
155 changes: 126 additions & 29 deletions R/utils.R
Original file line number Diff line number Diff line change
Expand Up @@ -157,7 +157,7 @@ render_dependencies <- function(dependencies, local = TRUE, prefix=NULL) {
# as in Dash for Python
if ("script" %in% names(dep) && tools::file_ext(dep[["script"]]) != "map") {
if (!(is_local) & !(is.null(dep$src$href))) {
html <- generate_js_dist_html(href = dep$src$href)
html <- generate_js_dist_html(tagdata = dep$src$href)
} else {
script_mtime <- file.mtime(getDependencyPath(dep))
modtime <- as.integer(script_mtime)
Expand All @@ -172,10 +172,10 @@ render_dependencies <- function(dependencies, local = TRUE, prefix=NULL) {
"&m=",
modified)

html <- generate_js_dist_html(href = dep[["script"]], as_is = TRUE)
html <- generate_js_dist_html(tagdata = dep[["script"]], as_is = TRUE)
}
} else if (!(is_local) & "stylesheet" %in% names(dep) & src == "href") {
html <- generate_css_dist_html(href = paste(dep[["src"]][["href"]],
html <- generate_css_dist_html(tagdata = paste(dep[["src"]][["href"]],
dep[["stylesheet"]],
sep="/"),
local = FALSE)
Expand All @@ -192,20 +192,20 @@ render_dependencies <- function(dependencies, local = TRUE, prefix=NULL) {
"?v=",
dep$version)

html <- generate_css_dist_html(href = sheetpath, as_is = TRUE)
html <- generate_css_dist_html(tagdata = sheetpath, as_is = TRUE)
} else {
sheetpath <- paste0(dep[["src"]][["file"]],
dep[["stylesheet"]],
"?v=",
dep$version)

html <- generate_css_dist_html(href = sheetpath, as_is = TRUE)
html <- generate_css_dist_html(tagdata = sheetpath, as_is = TRUE)
}

} else {
sheetpath <- paste0(dep[["src"]][["file"]],
dep[["stylesheet"]])
html <- generate_css_dist_html(href = sheetpath, as_is = TRUE)
html <- generate_css_dist_html(tagdata = sheetpath, as_is = TRUE)
}
}
})
Expand Down Expand Up @@ -536,54 +536,151 @@ get_mimetype <- function(filename) {
empty = "application/octet-stream"))
}

generate_css_dist_html <- function(href,
generate_css_dist_html <- function(tagdata,
local = FALSE,
local_path = NULL,
prefix = NULL,
as_is = FALSE) {
attribs <- names(tagdata)
if (!(local)) {
if (grepl("^(?:http(s)?:\\/\\/)?[\\w.-]+(?:\\.[\\w\\.-]+)+[\\w\\-\\._~:/?#[\\]@!\\$&'\\(\\)\\*\\+,;=.]+$",
href,
perl=TRUE) || as_is) {
sprintf("<link href=\"%s\" rel=\"stylesheet\">", href)
if (any(grepl("^(?:http(s)?:\\/\\/)?[\\w.-]+(?:\\.[\\w\\.-]+)+[\\w\\-\\._~:/?#[\\]@!\\$&'\\(\\)\\*\\+,;=.]+$",
tagdata,
perl=TRUE)) || as_is) {
if (is.list(tagdata))
glue::glue('<link ', glue::glue_collapse(glue::glue('{attribs}="{tagdata}"'), sep=" "), ' rel="stylesheet">')
else
glue::glue('<link ', glue::glue('href="{tagdata}"'), ' rel="stylesheet">')
}
else
stop(sprintf("Invalid URL supplied in external_stylesheets. Please check the syntax used for this parameter."), call. = FALSE)
} else {
# strip leading slash from href if present
href <- sub("^/", "", href)
modified <- as.integer(file.mtime(local_path))
sprintf("<link href=\"%s%s?m=%s\" rel=\"stylesheet\">",
prefix,
href,
modified)
# strip leading slash from href if present
if (is.list(tagdata)) {
tagdata$href <- paste0(prefix, sub("^/", "", tagdata$href))
glue::glue('<link ', glue::glue_collapse(glue::glue('{attribs}="{tagdata}?m={modified}"'), sep=" "), ' rel="stylesheet">')
}
else {
tagdata <- sub("^/", "", tagdata)
glue::glue('<link ', glue::glue('href="{prefix}{tagdata}?m={modified}"'), ' rel="stylesheet">')
}
}
}

generate_js_dist_html <- function(href,

generate_js_dist_html <- function(tagdata,
local = FALSE,
local_path = NULL,
prefix = NULL,
as_is = FALSE) {
attribs <- names(tagdata)
if (!(local)) {
if (grepl("^(?:http(s)?:\\/\\/)?[\\w.-]+(?:\\.[\\w\\.-]+)+[\\w\\-\\._~:/?#[\\]@!\\$&'\\(\\)\\*\\+,;=.]+$",
href,
perl=TRUE) || as_is) {
sprintf("<script src=\"%s\"></script>", href)
}
if (any(grepl("^(?:http(s)?:\\/\\/)?[\\w.-]+(?:\\.[\\w\\.-]+)+[\\w\\-\\._~:/?#[\\]@!\\$&'\\(\\)\\*\\+,;=.]+$",
tagdata,
perl=TRUE)) || as_is) {
if (is.list(tagdata))
glue::glue('<script ', glue::glue_collapse(glue::glue('{attribs}="{tagdata}"'), sep=" "), '></script>')
else
glue::glue('<script ', glue::glue('src="{tagdata}"'), '></script>')
}
else
stop(sprintf("Invalid URL supplied. Please check the syntax used for this parameter."), call. = FALSE)
} else {
# strip leading slash from href if present
href <- sub("^/", "", href)
modified <- as.integer(file.mtime(local_path))
sprintf("<script src=\"%s%s?m=%s\"></script>",
prefix,
href,
modified)
# strip leading slash from href if present
if (is.list(tagdata)) {
tagdata$src <- paste0(prefix, sub("^/", "", tagdata$src))
glue::glue('<script ', glue::glue_collapse(glue::glue('{attribs}="{tagdata}?m={modified}"'), sep=" "), '></script>')
}
else {
tagdata <- sub("^/", "", tagdata)
glue::glue('<script ', glue::glue('src="{prefix}{tagdata}?m={modified}"'), '></script>')
}
}
}

assertValidExternals <- function(scripts, stylesheets) {
allowed_js_attribs <- c("async",
"crossorigin",
"defer",
"integrity",
"nomodule",
"nonce",
"referrerpolicy",
"src",
"type",
"charset",
"language")

allowed_css_attribs <- c("as",
"crossorigin",
"disabled",
"href",
"hreflang",
"importance",
"integrity",
"media",
"referrerpolicy",
"rel",
"sizes",
"title",
"type",
"methods",
"prefetch",
"target",
"charset",
"rev")
script_attributes <- character()
stylesheet_attributes <- character()

for (item in scripts) {
if (is.list(item)) {
if (!"src" %in% names(item) || !(any(grepl("^(?:http(s)?:\\/\\/)?[\\w.-]+(?:\\.[\\w\\.-]+)+[\\w\\-\\._~:/?#[\\]@!\\$&'\\(\\)\\*\\+,;=.]+$",
item,
perl=TRUE))))
stop("A valid URL must be included with every entry in external_scripts. Please sure no 'src' entries are missing or malformed.", call. = FALSE)
if (any(names(item) == ""))
stop("Please verify that all attributes are named elements when specifying URLs for scripts and stylesheets.", call. = FALSE)
script_attributes <- c(script_attributes, names(item))
}
else {
if (!grepl("^(?:http(s)?:\\/\\/)?[\\w.-]+(?:\\.[\\w\\.-]+)+[\\w\\-\\._~:/?#[\\]@!\\$&'\\(\\)\\*\\+,;=.]+$",
item,
perl=TRUE))
stop("A valid URL must be included with every entry in external_scripts. Please sure no 'src' entries are missing or malformed.", call. = FALSE)
script_attributes <- c(script_attributes, character(0))
}
}

for (item in stylesheets) {
if (is.list(item)) {
if (!"href" %in% names(item) || !(any(grepl("^(?:http(s)?:\\/\\/)?[\\w.-]+(?:\\.[\\w\\.-]+)+[\\w\\-\\._~:/?#[\\]@!\\$&'\\(\\)\\*\\+,;=.]+$",
item,
perl=TRUE))))
stop("A valid URL must be included with every entry in external_stylesheets. Please sure no 'href' entries are missing or malformed.", call. = FALSE)
if (any(names(item) == ""))
stop("Please verify that all attributes are named elements when specifying URLs for scripts and stylesheets.", call. = FALSE)
stylesheet_attributes <- c(stylesheet_attributes, names(item))
}
else {
if (!grepl("^(?:http(s)?:\\/\\/)?[\\w.-]+(?:\\.[\\w\\.-]+)+[\\w\\-\\._~:/?#[\\]@!\\$&'\\(\\)\\*\\+,;=.]+$",
item,
perl=TRUE))
stop("A valid URL must be included with every entry in external_stylesheets. Please sure no 'href' entries are missing or malformed.", call. = FALSE)
stylesheet_attributes <- c(stylesheet_attributes, character(0))
}
}

invalid_script_attributes <- setdiff(script_attributes, allowed_js_attribs)
invalid_stylesheet_attributes <- setdiff(stylesheet_attributes, allowed_css_attribs)

if (length(invalid_script_attributes) > 0 || length(invalid_stylesheet_attributes) > 0) {
stop(sprintf("The following script or stylesheet attributes are invalid: %s.",
paste0(c(invalid_script_attributes, invalid_stylesheet_attributes), collapse=", ")), call. = FALSE)
}
invisible(TRUE)
}

generate_meta_tags <- function(metas) {
has_ie_compat <- any(vapply(metas, function(x)
x$name == "http-equiv" && x$content == "X-UA-Compatible",
Expand Down
8 changes: 6 additions & 2 deletions man/Dash.Rd

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

Loading