From e91bf80c56696fd2b8676e4284095f2d7bf472c7 Mon Sep 17 00:00:00 2001 From: Tomasz Kalinowski Date: Mon, 11 Nov 2024 15:01:15 -0500 Subject: [PATCH 1/8] initial draft of positron variables pane methods --- R/ark-variables-methods.R | 85 +++++++++++++++++++++++++++++++++++++++ R/zzz.R | 52 ++++++++++++++++++++++++ 2 files changed, 137 insertions(+) create mode 100644 R/ark-variables-methods.R diff --git a/R/ark-variables-methods.R b/R/ark-variables-methods.R new file mode 100644 index 000000000..066160f2e --- /dev/null +++ b/R/ark-variables-methods.R @@ -0,0 +1,85 @@ + +# Methods for Populating the Positron/Ark Variables Pane +# These methods primarily delegate to the implementation in the +# Positron python_ipykernel.inspectors python module. + +#' @param x Object to get the display value for +#' @param width Maximum expected width. This is just a suggestion, the UI +#' can stil truncate the string to different widths. +ark_positron_variable_display_value.python.builtin.object <- function(x, ..., width = getOption("width")) { + .globals$get_positron_variable_inspector(x)$get_display_value(width)[[1L]] +} + + +#' @param x Object to get the display type for +#' @param include_length Boolean indicating whether to include object length. +ark_positron_variable_display_type.python.builtin.object <- function(x, ..., include_length = TRUE) { + i <- .globals$get_positron_variable_inspector(x) + out <- i$get_display_type() + + if(startsWith(class(x)[1], "python.builtin.")) + out <- paste("python", out) + + out +} + + +#' @param x Object to get the variable kind for +ark_positron_variable_kind.python.builtin.object <- function(x, ...) { + i <- .globals$get_positron_variable_inspector(x) + i$get_kind() +} + + +#' @param x Check if `x` has children +ark_positron_variable_has_children.python.builtin.object <- function(x, ...) { + i <- .globals$get_positron_variable_inspector(x) + i$has_children() +} + +ark_positron_variable_get_children.python.builtin.object <- function(x, ...) { + # Return an R list of children. The order of children should be + # stable between repeated calls on the same object. For example: + i <- .globals$get_positron_variable_inspector(x) + + get_keys_and_children <- .globals$ark_variable_get_keys_and_children + if (is.null(get_keys_and_children)) { + get_keys_and_children <- .globals$ark_variable_get_keys_and_children <- py_eval( + "lambda i: zip(*((str(key), i.get_child(key)) for key in i.get_children()))", + convert = FALSE + ) + } + + keys_and_children <- iterate(get_keys_and_children(i), simplify = FALSE) + out <- iterate(keys_and_children[[2L]], simplify = FALSE) + names(out) <- as.character(py_to_r(keys_and_children[[1L]])) + + out +} + +#' @param index An integer > 1, representing the index position of the child in the +#' list returned by `ark_variable_get_children()`. +#' @param name The name of the child, corresponding to `names(ark_variable_get_children(x))[index]`. +#' This may be a string or `NULL`. If using the name, it is the method author's responsibility to ensure +#' the name is a valid, unique accessor. Additionally, if the original name from `ark_variable_get_children()` +#' was too long, `ark` may discard the name and supply `name = NULL` instead. +ark_positron_variable_get_child_at.python.builtin.object <- function(x, ..., name, index) { + # cat("name: ", name, "index: ", index, "\n", file = "~/debug.log", append = TRUE) + # This could be implemented as: + # ark_variable_get_children(x)[[index]] + i <- .globals$get_positron_variable_inspector(x) + get_child <- .globals$ark_variable_get_child + if (is.null(get_child)) { + get_child <- .globals$ark_variable_get_child <- py_run_string( + local = TRUE, convert = FALSE, " +def new_get_child(): + from itertools import islice + def get_child(inspector, index): + key = next(islice(inspector.get_children(), index-1, None)) + return inspector.get_child(key) + return get_child +")$new_get_child() + } + + get_child(i, index) +} diff --git a/R/zzz.R b/R/zzz.R index a60a419d1..a4befd2c2 100644 --- a/R/zzz.R +++ b/R/zzz.R @@ -61,9 +61,61 @@ s3_register <- asNamespace("rlang")$s3_register s3_register("pillar::type_sum", "python.builtin.object") + if (is_positron()) { + + setHook("reticulate.onPyInit", function() { + + # options("ark.testing" = TRUE) + inspectors <- import_positron_ipykernel_inspectors() + if(is.null(inspectors)) { + # warning("positron_ipykernel.inspectors could not be found, variables pane support for Python objects will be limited") + return() + } + + .globals$get_positron_variable_inspector <- inspectors$get_inspector + + .ark.register_method <- get(".ark.register_method", envir = globalenv()) + + for (method_name in c( + "ark_positron_variable_display_value", + "ark_positron_variable_display_type", + "ark_positron_variable_has_children", + "ark_positron_variable_kind", + "ark_positron_variable_get_child_at", + "ark_positron_variable_get_children" + )) { + method_sym <- as.name(paste0(method_name, ".python.builtin.object")) + .ark.register_method( + method_name, "python.builtin.object", + call(":::", quote(reticulate), method_sym)) + } + }) + } } # .onUnload <- function(libpath) { # py_finalize() # called from reg.finalizer(.globals) instead. # } + + +import_positron_ipykernel_inspectors <- function() { + if(!is_positron()) + return (NULL) + + x <- Sys.getenv("_") + # x == + # on mac: "/Applications/Positron.app/Contents/Resources/app/extensions/positron-r/resources/ark/ark" + if (grepl("positron-r", x, ignore.case = TRUE)) { + inspectors_path <- list.files( + path = sub("positron-r.*$", "positron-python", x), + pattern = "^inspectors.py$", + recursive = TRUE, full.names = TRUE + ) + # inspectors_path == + # on mac: "/Applications/Positron.app/Contents/Resources/app/extensions/positron-python/python_files/positron/positron_ipykernel/inspectors.py" + if (length(inspectors_path) == 1) { + import_from_path("positron_ipykernel.inspectors", path = dirname(dirname(inspectors_path))) + } + } +} From d77cc689ec8e87d8146ab575c4b44514308039da Mon Sep 17 00:00:00 2001 From: Tomasz Kalinowski Date: Wed, 13 Nov 2024 09:29:37 -0500 Subject: [PATCH 2/8] implement `ChildrenOverflow` for >100 children --- R/ark-variables-methods.R | 43 ++++++++++++++++----------- R/zzz.R | 11 ++++--- inst/python/rpytools/ark_variables.py | 33 ++++++++++++++++++++ 3 files changed, 65 insertions(+), 22 deletions(-) create mode 100644 inst/python/rpytools/ark_variables.py diff --git a/R/ark-variables-methods.R b/R/ark-variables-methods.R index 066160f2e..4db8c949e 100644 --- a/R/ark-variables-methods.R +++ b/R/ark-variables-methods.R @@ -17,7 +17,7 @@ ark_positron_variable_display_type.python.builtin.object <- function(x, ..., inc i <- .globals$get_positron_variable_inspector(x) out <- i$get_display_type() - if(startsWith(class(x)[1], "python.builtin.")) + if (startsWith(class(x)[1], "python.builtin.")) # display convert value? out <- paste("python", out) out @@ -44,17 +44,15 @@ ark_positron_variable_get_children.python.builtin.object <- function(x, ...) { get_keys_and_children <- .globals$ark_variable_get_keys_and_children if (is.null(get_keys_and_children)) { - get_keys_and_children <- .globals$ark_variable_get_keys_and_children <- py_eval( - "lambda i: zip(*((str(key), i.get_child(key)) for key in i.get_children()))", - convert = FALSE - ) + get_keys_and_children <- .globals$ark_variable_get_keys_and_children <- + import("rpytools.ark_variables", convert = FALSE)$get_keys_and_children } keys_and_children <- iterate(get_keys_and_children(i), simplify = FALSE) - out <- iterate(keys_and_children[[2L]], simplify = FALSE) - names(out) <- as.character(py_to_r(keys_and_children[[1L]])) + children <- iterate(keys_and_children[[2L]], simplify = FALSE) + names(children) <- as.character(py_to_r(keys_and_children[[1L]])) - out + children } #' @param index An integer > 1, representing the index position of the child in the @@ -70,16 +68,25 @@ ark_positron_variable_get_child_at.python.builtin.object <- function(x, ..., nam i <- .globals$get_positron_variable_inspector(x) get_child <- .globals$ark_variable_get_child if (is.null(get_child)) { - get_child <- .globals$ark_variable_get_child <- py_run_string( - local = TRUE, convert = FALSE, " -def new_get_child(): - from itertools import islice - def get_child(inspector, index): - key = next(islice(inspector.get_children(), index-1, None)) - return inspector.get_child(key) - return get_child -")$new_get_child() + get_child <- .globals$ark_variable_get_child <- + import("rpytools.ark_variables", convert = FALSE)$get_child + } + + get_child(i, index) } - get_child(i, index) + +ark_positron_variable_display_type.rpytools.ark_variables.ChildrenOverflow <- function(x, ..., include_length = TRUE) { + "" +} +ark_positron_variable_kind.rpytools.ark_variables.ChildrenOverflow <- function(x, ...) { + "empty" # other? collection? map? lazy? +} + +ark_positron_variable_display_value.rpytools.ark_variables.ChildrenOverflow <- function(x, ..., width = getOption("width")) { + paste(py_to_r(x$n_remaining), "more values") +} + +ark_positron_variable_has_children.rpytools.ark_variables.ChildrenOverflow <- function(x, ...) { + FALSE } diff --git a/R/zzz.R b/R/zzz.R index a4befd2c2..bbdfe1858 100644 --- a/R/zzz.R +++ b/R/zzz.R @@ -84,10 +84,13 @@ "ark_positron_variable_get_child_at", "ark_positron_variable_get_children" )) { - method_sym <- as.name(paste0(method_name, ".python.builtin.object")) - .ark.register_method( - method_name, "python.builtin.object", - call(":::", quote(reticulate), method_sym)) + for (class_name in c("python.builtin.object", + "rpytools.ark_variables.ChildrenOverflow")) { + method <- get0(paste0(method_name, ".", class_name)) + if (!is.null(method)) { + .ark.register_method(method_name, class_name, method) + } + } } }) } diff --git a/inst/python/rpytools/ark_variables.py b/inst/python/rpytools/ark_variables.py new file mode 100644 index 000000000..180c8c1eb --- /dev/null +++ b/inst/python/rpytools/ark_variables.py @@ -0,0 +1,33 @@ +from itertools import islice + +MAX_DISPLAY = 100 + +def get_keys_and_children(inspector): + keys, values = [], [] + keys_iterator = iter(inspector.get_children()) + # positron frontend only displays the first 100 items, then the length of the rest. + try: + for _ in range(MAX_DISPLAY): + key = next(keys_iterator) + keys.append(str(key)) + values.append(inspector.get_child(key)) + except StopIteration: + return keys, values + + n_remaining = 0 + for n_remaining, _ in enumerate(keys_iterator, 1): + pass + if n_remaining > 0: + keys.append("[[...]]") + values.append(ChildrenOverflow(n_remaining)) + + return keys, values + + +def get_child(inspector, index): + key = next(islice(inspector.get_children(), index - 1, None)) + return inspector.get_child(key) + +class ChildrenOverflow: + def __init__(self, n_remaining): + self.n_remaining = n_remaining From abe0f37ebc283af9d04382ebd714435dbfd40130 Mon Sep 17 00:00:00 2001 From: Tomasz Kalinowski Date: Wed, 13 Nov 2024 10:58:26 -0500 Subject: [PATCH 3/8] try to use `ins.get_length()` to detect children overflow --- inst/python/rpytools/ark_variables.py | 33 +++++++++++++++------------ 1 file changed, 18 insertions(+), 15 deletions(-) diff --git a/inst/python/rpytools/ark_variables.py b/inst/python/rpytools/ark_variables.py index 180c8c1eb..c44d82f43 100644 --- a/inst/python/rpytools/ark_variables.py +++ b/inst/python/rpytools/ark_variables.py @@ -2,24 +2,26 @@ MAX_DISPLAY = 100 + def get_keys_and_children(inspector): + # positron frontend only displays the first 100 items, then the count of the rest. keys, values = [], [] keys_iterator = iter(inspector.get_children()) - # positron frontend only displays the first 100 items, then the length of the rest. - try: - for _ in range(MAX_DISPLAY): - key = next(keys_iterator) - keys.append(str(key)) - values.append(inspector.get_child(key)) - except StopIteration: - return keys, values - - n_remaining = 0 - for n_remaining, _ in enumerate(keys_iterator, 1): - pass - if n_remaining > 0: - keys.append("[[...]]") - values.append(ChildrenOverflow(n_remaining)) + for key in islice(keys_iterator, MAX_DISPLAY): + keys.append(str(key)) + values.append(inspector.get_child(key)) + + if len(keys) == MAX_DISPLAY: + # check if there are more children + n_children = inspector.get_length() + if n_children == 0: + # no len() method, finish iteratoring over keys_iterator to get the true size + for n_children, _ in enumerate(keys_iterator, MAX_DISPLAY + 1): + pass + n_remaining = n_children - MAX_DISPLAY + if n_remaining > 0: + keys.append("[[...]]") + values.append(ChildrenOverflow(n_remaining)) return keys, values @@ -28,6 +30,7 @@ def get_child(inspector, index): key = next(islice(inspector.get_children(), index - 1, None)) return inspector.get_child(key) + class ChildrenOverflow: def __init__(self, n_remaining): self.n_remaining = n_remaining From 3f81100444ca1d6e55af78c7fa8ffcbf5ef96b0f Mon Sep 17 00:00:00 2001 From: Tomasz Kalinowski Date: Wed, 13 Nov 2024 11:43:44 -0500 Subject: [PATCH 4/8] plan ahead for ark method `.ps.positron_ipykernel_path()` --- R/zzz.R | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/R/zzz.R b/R/zzz.R index bbdfe1858..ff70ab84c 100644 --- a/R/zzz.R +++ b/R/zzz.R @@ -106,6 +106,17 @@ import_positron_ipykernel_inspectors <- function() { if(!is_positron()) return (NULL) + .ps.positron_ipykernel_path <- get0(".ps.positron_ipykernel_path", globalenv()) + if (!is.null(.ps.positron_ipykernel_path)) { + path <- .ps.positron_ipykernel_path() + if (grepl("positron_ipykernel[/\\]?$", path)) + path <- dirname(path) + inspectors <- import_from_path("positron_ipykernel.inspectors", path = path) + return(inspectors) + } + + # hacky fallback by inspecting `_` env var, until ark is updated. + # Only works in some contexts. x <- Sys.getenv("_") # x == # on mac: "/Applications/Positron.app/Contents/Resources/app/extensions/positron-r/resources/ark/ark" From 9db7df4fdca7522fc9cb51cdf70e4a8e454336d2 Mon Sep 17 00:00:00 2001 From: Tomasz Kalinowski Date: Wed, 13 Nov 2024 11:44:27 -0500 Subject: [PATCH 5/8] allow disabling; fail quietly. --- R/zzz.R | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/R/zzz.R b/R/zzz.R index ff70ab84c..593cf90bb 100644 --- a/R/zzz.R +++ b/R/zzz.R @@ -65,8 +65,12 @@ setHook("reticulate.onPyInit", function() { + disable <- nzchar(Sys.getenv("_RETICULATE_POSITRON_DISABLE_VARIABLE_INSPECTORS_")) + if (disable) + return() + # options("ark.testing" = TRUE) - inspectors <- import_positron_ipykernel_inspectors() + inspectors <- tryCatch(import_positron_ipykernel_inspectors(), error = function(e) NULL) if(is.null(inspectors)) { # warning("positron_ipykernel.inspectors could not be found, variables pane support for Python objects will be limited") return() From 9a6dbea49729889f6255a6f3b86dfe0c9dc783d2 Mon Sep 17 00:00:00 2001 From: Tomasz Kalinowski Date: Wed, 13 Nov 2024 16:26:50 -0500 Subject: [PATCH 6/8] another fallback for finding `positron_ipykernel.inspectors` --- R/zzz.R | 30 +++++++++++++++++++++++++++--- 1 file changed, 27 insertions(+), 3 deletions(-) diff --git a/R/zzz.R b/R/zzz.R index 593cf90bb..82149a46a 100644 --- a/R/zzz.R +++ b/R/zzz.R @@ -119,8 +119,10 @@ import_positron_ipykernel_inspectors <- function() { return(inspectors) } - # hacky fallback by inspecting `_` env var, until ark is updated. - # Only works in some contexts. + # hacky "usually work" fallbacks for finding the positron-python extension, + # until ark+positron are updated and can reliably provide the canonical path + + # Try inspecting `_` env var. Only works in some contexts. x <- Sys.getenv("_") # x == # on mac: "/Applications/Positron.app/Contents/Resources/app/extensions/positron-r/resources/ark/ark" @@ -133,7 +135,29 @@ import_positron_ipykernel_inspectors <- function() { # inspectors_path == # on mac: "/Applications/Positron.app/Contents/Resources/app/extensions/positron-python/python_files/positron/positron_ipykernel/inspectors.py" if (length(inspectors_path) == 1) { - import_from_path("positron_ipykernel.inspectors", path = dirname(dirname(inspectors_path))) + return(import_from_path("positron_ipykernel.inspectors", + path = dirname(dirname(inspectors_path)))) + } } + + # vscode/positron-python will place itself on the PATH sometimes + PATH <- Sys.getenv("PATH") + PATH <- strsplit(PATH, .Platform$path.sep, fixed = TRUE)[[1L]] + PATH <- grep("positron-python", PATH, value = TRUE, ignore.case = TRUE) + # PATH == + # on mac: "/Applications/Positron.app/Contents/Resources/app/extensions/positron-python/python_files/deactivate/zsh" + for (path in PATH) { + inspectors_path <- list.files( + path = sub("^(.*positron-python)(.*)$", "\\1", path), + pattern = "^inspectors.py$", + recursive = TRUE, full.names = TRUE + ) + if (length(inspectors_path) == 1) { + return(import_from_path("positron_ipykernel.inspectors", + path = dirname(dirname(inspectors_path)))) + } + } + + NULL } From 4b752c0509bddb80761d33373f0da50aad9f6636 Mon Sep 17 00:00:00 2001 From: Tomasz Kalinowski Date: Thu, 14 Nov 2024 09:04:31 -0500 Subject: [PATCH 7/8] update ark call for getting ipykernel path --- R/zzz.R | 17 ++++++++++------- 1 file changed, 10 insertions(+), 7 deletions(-) diff --git a/R/zzz.R b/R/zzz.R index 82149a46a..c41c133e9 100644 --- a/R/zzz.R +++ b/R/zzz.R @@ -110,17 +110,20 @@ import_positron_ipykernel_inspectors <- function() { if(!is_positron()) return (NULL) - .ps.positron_ipykernel_path <- get0(".ps.positron_ipykernel_path", globalenv()) - if (!is.null(.ps.positron_ipykernel_path)) { - path <- .ps.positron_ipykernel_path() - if (grepl("positron_ipykernel[/\\]?$", path)) - path <- dirname(path) - inspectors <- import_from_path("positron_ipykernel.inspectors", path = path) + tryCatch({ + # https://github.com/posit-dev/positron/pull/5368 + .ps.ui.executeCommand <- get(".ps.ui.executeCommand", globalenv()) + ipykernel_path <- .ps.ui.executeCommand("positron.reticulate.getIPykernelPath") + inspectors <- import_from_path("positron_ipykernel.inspectors", + path = dirname(ipykernel_path)) return(inspectors) - } + }, + error = function(e) NULL) + # hacky "usually work" fallbacks for finding the positron-python extension, # until ark+positron are updated and can reliably provide the canonical path + # (i.e., until https://github.com/posit-dev/positron/pull/5368 is in the release build) # Try inspecting `_` env var. Only works in some contexts. x <- Sys.getenv("_") From 85cfd0f0c16b91a67ffb5eea4ea8f707da949f6a Mon Sep 17 00:00:00 2001 From: Tomasz Kalinowski Date: Thu, 14 Nov 2024 09:44:54 -0500 Subject: [PATCH 8/8] add NEWS --- NEWS.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/NEWS.md b/NEWS.md index 0012595ac..d7c162eab 100644 --- a/NEWS.md +++ b/NEWS.md @@ -35,6 +35,9 @@ - Improved behavior when the conda binary used to create an environment could not be resolved (contributed by @tl-hbk, #1654, #1659) +- Added Positron support for the Variables Pane and `repl_python()` + (#1692, #1641, #1648, #1658, #1681, #1687). + # reticulate 1.39.0 - Python background threads can now run in parallel with the R session (#1641).