One key strength of the preprocessing framework within
eyeris is its modularity.
While we encourage most users to use the glassbox()
function for simplicity and reproducibility, advanced users can create
custom preprocessing steps that seamlessly integrate into the
pipeline.
This vignette walks you through the structure required to write your
own eyeris-compatible preprocessing functions.
Under the hood, each preprocessing function in eyeris is
a wrapper around a core operation that gets tracked, versioned, and
stored using the pipeline_handler().
Custom pipeline steps must conform to the eyeris
protocol for maximum compatibility with the downstream functions we
provide.
Following the eyeris protocol also ensures: - all
operations follow a predictable structure, and - that new pupil data
columns based on previous operations in the chain are able to be
dynamically constructed within the core timeseries data
frame.
For instance:
pupil_raw -> pupil_raw_deblink -> pupil_raw_deblink_detransient -> ...
If you’re unfamiliar with how these columns are structured and
tracked, first check out the companion vignette: 📦 Anatomy of an eyeris Object.
eyerisLet’s say you want to write a new eyeris extension
function called winsorize() to apply winsorization to
extreme pupil values.
This function should accept a data frame x, a string
prev_op (i.e., the name of the previous pupil column), and
any custom parameters.
eyeris::pipeline_handler()The pipeline_handler() enables your function to
automatically:
eyeris list object’s
params field,timeseries
list of data frames, andlatest pointer.#' Winsorize pupil values
#'
#' Applies winsorization to extreme pupil values within each block.
#'
#' @param eyeris An `eyeris` object created by [load_asc()].
#' @param lower Lower quantile threshold. Default is 0.01.
#' @param upper Upper quantile threshold. Default is 0.99.
#' @param call_info A list of call information and parameters. If not provided,
#'   it will be generated from the function call.
#'
#' @return Updated `eyeris` object with new winsorized pupil column.
winsorize <- function(eyeris, lower = 0.01, upper = 0.99, call_info = NULL) {
  # create call_info if not provided
  call_info <- if (is.null(call_info)) {
    list(
      call_stack = match.call(),
      parameters = list(lower = lower, upper = upper)
    )
  } else {
    call_info
  }
  # handle binocular objects
  if (is_binocular_object(eyeris)) {
    # process left and right eyes independently
    left_result <- eyeris$left |>
      pipeline_handler(
        winsorize_pupil,
        "winsorize",
        lower = lower,
        upper = upper,
        call_info = call_info
      )
    
    right_result <- eyeris$right |>
      pipeline_handler(
        winsorize_pupil,
        "winsorize",
        lower = lower,
        upper = upper,
        call_info = call_info
      )
    
    # return combined structure
    list_out <- list(
      left = left_result,
      right = right_result,
      original_file = eyeris$original_file,
      raw_binocular_object = eyeris$raw_binocular_object
    )
    class(list_out) <- "eyeris"
    return(list_out)
  } else {
    # regular eyeris object, process normally
    eyeris |>
      pipeline_handler(
        winsorize_pupil,
        "winsorize",
        lower = lower,
        upper = upper,
        call_info = call_info
      )
  }
}The call_info parameter is crucial for maintaining
reproducibility and debugging in eyeris. It captures:
call_stack: The exact function call
that was made (using match.call())parameters: A list of all parameters
passed to your functionThis information is automatically stored in
eyeris$params[[new_suffix]] and can be used for: -
Generating reproducible reports - Debugging pipeline issues - Tracking
parameter changes across different runs
Important Notes: - Always include
call_info = NULL as the last parameter in your function
signature - Use match.call() to capture the function call -
Pass all your function parameters in the parameters list -
Always pass call_info = call_info as the last argument to
pipeline_handler()
call_info = NULL as the last parametereyeris objectcall_info = call_infoYou should now be able to use your new function extension as a
component within a new custom eyeris pipeline declaration.
To illustrate:
"winsorize") to the column and log structure:
pipeline_handler(..., "winsorize")winsorize() wrapper function,
andwinsorize_pupil() logic implementation
function.pupil_raw_*_winsorize as the new
output column!prev_op, not on any hardcoded
column names!pipeline_handler() is looking out for a vector
it can transpose into the new column that will be added to the
timeseries data frame within the resulting
eyeris object.eyeris$timeseries[[1]]) to debug before integrating it with
the eyeris pipeline protocol.call_info handling for reproducibility and debugging
capabilities.For more complex functions, you might want to customize the
call_info structure:
custom_function <- function(eyeris, param1, param2, call_info = NULL) {
  # custom call_info with additional metadata
  call_info <- if (is.null(call_info)) {
    list(
      call_stack = match.call(),
      parameters = list(param1 = param1, param2 = param2),
      metadata = list(
        timestamp = Sys.time(),
        version = "1.0.0",
        description = "Custom processing step"
      )
    )
  } else {
    call_info
  }
  pipeline_handler(
    eyeris,
    custom_pupil_function,
    "custom",
    param1 = param1,
    param2 = param2,
    call_info = call_info
  )
}We hope you are now convinced at the power and extensibility the
eyeris protocol enables! As we demonstrated here, with just
a little bit of structure, you can create custom extension steps
tailored to your specific analysis needs – all while preserving the
reproducibility and organizational core principles eyeris
was designed and built around.
If you’d like to contribute new steps to eyeris, feel
free to open a pull request or discussion on GitHub!
eyerisIf you use the eyeris package in your research, please
cite it!
Run the following in R to get the citation:
citation("eyeris")
#> To cite package 'eyeris' in publications use:
#> 
#>   Schwartz ST, Yang H, Xue AM, He M (2025). "eyeris: A flexible,
#>   extensible, and reproducible pupillometry preprocessing framework in
#>   R." _bioRxiv_, 1-37. doi:10.1101/2025.06.01.657312
#>   <https://doi.org/10.1101/2025.06.01.657312>.
#> 
#> A BibTeX entry for LaTeX users is
#> 
#>   @Article{,
#>     title = {eyeris: A flexible, extensible, and reproducible pupillometry preprocessing framework in R},
#>     author = {Shawn T Schwartz and Haopei Yang and Alice M Xue and Mingjian He},
#>     journal = {bioRxiv},
#>     year = {2025},
#>     pages = {1--37},
#>     doi = {10.1101/2025.06.01.657312},
#>   }