From 3af034b294fd5443567514f0105fcde88ce2df74 Mon Sep 17 00:00:00 2001 From: jgabry Date: Thu, 21 Nov 2024 19:15:39 -0700 Subject: [PATCH 1/2] use capture.output in more places --- tests/testthat/test-fit-init.R | 2 + tests/testthat/test-fit-shared.R | 34 +++++++----- tests/testthat/test-model-expose-functions.R | 18 ++++--- tests/testthat/test-model-init.R | 24 ++++++--- tests/testthat/test-model-laplace.R | 25 ++++++--- tests/testthat/test-model-methods.R | 57 +++++++++++++------- tests/testthat/test-model-optimize.R | 7 +-- tests/testthat/test-model-sample.R | 18 ++++--- tests/testthat/test-model-variables.R | 2 +- 9 files changed, 122 insertions(+), 65 deletions(-) diff --git a/tests/testthat/test-fit-init.R b/tests/testthat/test-fit-init.R index 02917833e..57d8984b0 100644 --- a/tests/testthat/test-fit-init.R +++ b/tests/testthat/test-fit-init.R @@ -5,6 +5,7 @@ set_cmdstan_path() data_list_schools <- testing_data("schools") data_list_logistic <- testing_data("logistic") test_inits <- function(mod, fit_init, data_list = NULL) { + utils::capture.output({ fit_sample <- mod$sample(data = data_list, chains = 1, init = fit_init, iter_sampling = 100, iter_warmup = 100, refresh = 0, seed = 1234) fit_sample_multi <- mod$sample(data = data_list, chains = 5, init = fit_init, @@ -20,6 +21,7 @@ test_inits <- function(mod, fit_init, data_list = NULL) { draws = posterior::as_draws_rvars(fit_init$draws()) fit_sample_draws <- mod$sample(data = data_list, chains = 1, init = draws, iter_sampling = 100, iter_warmup = 100, refresh = 0, seed = 1234) + }) return(0) } diff --git a/tests/testthat/test-fit-shared.R b/tests/testthat/test-fit-shared.R index be62d6e2f..a4593c385 100644 --- a/tests/testthat/test-fit-shared.R +++ b/tests/testthat/test-fit-shared.R @@ -454,22 +454,26 @@ test_that("draws are returned for model with spaces", { test_that("sampling with inits works with include_paths", { stan_program_w_include <- testing_stan_file("bernoulli_include") exe <- cmdstan_ext(strip_ext(stan_program_w_include)) - if(file.exists(exe)) { + if (file.exists(exe)) { file.remove(exe) } - mod_w_include <- cmdstan_model(stan_file = stan_program_w_include, quiet = FALSE, - include_paths = test_path("resources", "stan")) + mod_w_include <- cmdstan_model(stan_file = stan_program_w_include, + include_paths = test_path("resources", "stan")) data_list <- list(N = 10, y = c(0,1,0,0,0,0,0,0,0,1)) - - fit <- mod_w_include$sample( - data = data_list, - seed = 123, - chains = 4, - parallel_chains = 4, - refresh = 500, - init = list(list(theta = 0.25), list(theta = 0.25), list(theta = 0.25), list(theta = 0.25)) + expect_no_error( + fit <- mod_w_include$sample( + data = data_list, + seed = 123, + chains = 4, + parallel_chains = 4, + refresh = 500, + init = list(list(theta = 0.25), + list(theta = 0.25), + list(theta = 0.25), + list(theta = 0.25)) + ) ) }) @@ -548,8 +552,12 @@ test_that("code() warns if model not created with Stan file", { stan_program <- testing_stan_file("bernoulli") mod <- testing_model("bernoulli") mod_exe <- cmdstan_model(exe_file = mod$exe_file()) - fit_exe <- mod_exe$sample(data = list(N = 10, y = c(0, 1, 0, 1, 0, 1, 0, 1, 0, 1)), - refresh = 0) + utils::capture.output( + fit_exe <- mod_exe$sample( + data = list(N = 10, y = c(0, 1, 0, 1, 0, 1, 0, 1, 0, 1)), + refresh = 0 + ) + ) expect_warning( expect_null(fit_exe$code()), "'$code()' will return NULL because the 'CmdStanModel' was not created with a Stan file", diff --git a/tests/testthat/test-model-expose-functions.R b/tests/testthat/test-model-expose-functions.R index b3bb39c59..4adc74a90 100644 --- a/tests/testthat/test-model-expose-functions.R +++ b/tests/testthat/test-model-expose-functions.R @@ -80,11 +80,13 @@ stan_prog <- paste(function_decl, model <- write_stan_file(stan_prog) data_list <- testing_data("bernoulli") mod <- cmdstan_model(model, force_recompile = TRUE) -fit <- mod$sample(data = data_list) +utils::capture.output( + fit <- mod$sample(data = data_list) +) test_that("Functions can be exposed in model object", { - expect_no_error(mod$expose_functions(verbose = TRUE)) + expect_no_error(mod$expose_functions()) }) @@ -260,7 +262,7 @@ test_that("Functions handle complex types correctly", { }) test_that("Functions can be exposed in fit object", { - fit$expose_functions(verbose = TRUE) + fit$expose_functions() expect_equal( fit$functions$rtn_vec(c(1,2,3,4)), @@ -284,7 +286,9 @@ test_that("Compiled functions can be copied to global environment", { test_that("Functions can be compiled with model", { mod <- cmdstan_model(model, force_recompile = TRUE, compile_standalone = TRUE) - fit <- mod$sample(data = data_list) + utils::capture.output( + fit <- mod$sample(data = data_list) + ) expect_message( fit$expose_functions(), @@ -344,9 +348,11 @@ test_that("rng functions can be exposed", { model <- write_stan_file(stan_prog) data_list <- testing_data("bernoulli") mod <- cmdstan_model(model, force_recompile = TRUE) - fit <- mod$sample(data = data_list) + utils::capture.output( + fit <- mod$sample(data = data_list) + ) - fit$expose_functions(verbose = TRUE) + fit$expose_functions() set.seed(10) res1_1 <- fit$functions$wrap_normal_rng(5,10) res2_1 <- fit$functions$wrap_normal_rng(5,10) diff --git a/tests/testthat/test-model-init.R b/tests/testthat/test-model-init.R index c5be9e625..ebe546777 100644 --- a/tests/testthat/test-model-init.R +++ b/tests/testthat/test-model-init.R @@ -303,10 +303,12 @@ test_that("Initial values for single-element containers treated correctly", { " mod <- cmdstan_model(write_stan_file(modcode), force_recompile = TRUE) expect_no_error( - fit <- mod$sample( - data = list(y_mean = 0), - init = list(list(y = c(0))), - chains = 1 + utils::capture.output( + fit <- mod$sample( + data = list(y_mean = 0), + init = list(list(y = c(0))), + chains = 1 + ) ) ) }) @@ -331,7 +333,13 @@ test_that("Pathfinder inits do not drop dimensions", { " mod <- cmdstan_model(write_stan_file(modcode), force_recompile = TRUE) data <- list(N = 100, y = rnorm(100)) - pf <- mod$pathfinder(data = data, psis_resample = FALSE) - expect_no_error(fit <- mod$sample(data = data, init = pf, chains = 1, - iter_warmup = 100, iter_sampling = 100)) -}) \ No newline at end of file + utils::capture.output( + pf <- mod$pathfinder(data = data, psis_resample = FALSE) + ) + expect_no_error( + utils::capture.output( + fit <- mod$sample(data = data, init = pf, chains = 1, + iter_warmup = 100, iter_sampling = 100) + ) + ) +}) diff --git a/tests/testthat/test-model-laplace.R b/tests/testthat/test-model-laplace.R index 8f6ae9723..caaef8e48 100644 --- a/tests/testthat/test-model-laplace.R +++ b/tests/testthat/test-model-laplace.R @@ -63,10 +63,12 @@ test_that("laplace() runs when all arguments specified validly", { }) test_that("laplace() all valid 'mode' inputs give same results", { - mode <- mod$optimize(data = data_list, jacobian = TRUE, seed = 100, refresh = 0) - fit1 <- mod$laplace(data = data_list, mode = mode, seed = 100, refresh = 0) - fit2 <- mod$laplace(data = data_list, mode = mode$output_files(), seed = 100, refresh = 0) - fit3 <- mod$laplace(data = data_list, mode = NULL, seed = 100, refresh = 0) + utils::capture.output({ + mode <- mod$optimize(data = data_list, jacobian = TRUE, seed = 100, refresh = 0) + fit1 <- mod$laplace(data = data_list, mode = mode, seed = 100, refresh = 0) + fit2 <- mod$laplace(data = data_list, mode = mode$output_files(), seed = 100, refresh = 0) + fit3 <- mod$laplace(data = data_list, mode = NULL, seed = 100, refresh = 0) + }) expect_is(fit1, "CmdStanLaplace") expect_is(fit2, "CmdStanLaplace") @@ -85,17 +87,22 @@ test_that("laplace() all valid 'mode' inputs give same results", { }) test_that("laplace() allows choosing number of draws", { - fit <- mod$laplace(data = data_list, draws = 10, refresh = 0) + utils::capture.output({ + fit <- mod$laplace(data = data_list, draws = 10, refresh = 0) + fit2 <- mod$laplace(data = data_list, draws = 100, refresh = 0) + }) + expect_equal(fit$metadata()$draws, 10) expect_equal(posterior::ndraws(fit$draws()), 10) - fit2 <- mod$laplace(data = data_list, draws = 100, refresh = 0) expect_equal(fit2$metadata()$draws, 100) expect_equal(posterior::ndraws(fit2$draws()), 100) }) test_that("laplace() errors if jacobian arg doesn't match what optimize used", { - fit <- mod$optimize(data = data_list, jacobian = FALSE, refresh = 0) + utils::capture.output( + fit <- mod$optimize(data = data_list, jacobian = FALSE, refresh = 0) + ) expect_error( mod$laplace(data = data_list, mode = fit, jacobian = TRUE), "'jacobian' argument to optimize and laplace must match" @@ -107,7 +114,9 @@ test_that("laplace() errors if jacobian arg doesn't match what optimize used", { }) test_that("laplace() errors with bad combinations of arguments", { - fit <- mod$optimize(data = data_list, jacobian = TRUE, refresh = 0) + utils::capture.output( + fit <- mod$optimize(data = data_list, jacobian = TRUE, refresh = 0) + ) expect_error( mod$laplace(data = data_list, mode = mod, opt_args = list(iter = 10)), "Cannot specify both 'opt_args' and 'mode' arguments." diff --git a/tests/testthat/test-model-methods.R b/tests/testthat/test-model-methods.R index 4ad98ef02..308077a1f 100644 --- a/tests/testthat/test-model-methods.R +++ b/tests/testthat/test-model-methods.R @@ -4,7 +4,9 @@ skip_if(os_is_wsl()) set_cmdstan_path() mod <- cmdstan_model(testing_stan_file("bernoulli_log_lik"), force_recompile = TRUE) data_list <- testing_data("bernoulli") -fit <- mod$sample(data = data_list, chains = 1, refresh = 0) +utils::capture.output( + fit <- mod$sample(data = data_list, chains = 1, refresh = 0) +) test_that("Model methods automatically initialise when needed", { expect_no_error(fit$log_prob(unconstrained_variables=c(0.1))) @@ -59,7 +61,9 @@ test_that("Model methods environments are independent", { data_list_2 <- data_list data_list_2$N <- 20 data_list_2$y <- c(data_list$y, data_list$y) - fit_2 <- mod$sample(data = data_list_2, chains = 1) + utils::capture.output( + fit_2 <- mod$sample(data = data_list_2, chains = 1) + ) fit_2$init_model_methods() expect_equal(fit$log_prob(unconstrained_variables=c(0.1)), -8.6327599208828509347) @@ -90,8 +94,10 @@ test_that("methods error for incorrect inputs", { logistic_mod <- cmdstan_model(testing_stan_file("logistic"), force_recompile = TRUE) logistic_data_list <- testing_data("logistic") - logistic_fit <- logistic_mod$sample(data = logistic_data_list, chains = 1) - logistic_fit$init_model_methods(verbose = TRUE) + utils::capture.output( + logistic_fit <- logistic_mod$sample(data = logistic_data_list, chains = 1) + ) + logistic_fit$init_model_methods() expect_error( logistic_fit$unconstrain_variables(list(alpha = 0.5)), @@ -104,7 +110,9 @@ test_that("Methods error with already-compiled model", { precompile_mod <- testing_model("bernoulli") mod <- testing_model("bernoulli") data_list <- testing_data("bernoulli") - fit <- mod$sample(data = data_list, chains = 1) + utils::capture.output( + fit <- mod$sample(data = data_list, chains = 1) + ) expect_error( fit$init_model_methods(), "Model methods cannot be used with a pre-compiled Stan executable, the model must be compiled again", @@ -116,7 +124,9 @@ test_that("Methods can be compiled with model", { mod <- cmdstan_model(testing_stan_file("bernoulli"), force_recompile = TRUE, compile_model_methods = TRUE) - fit <- mod$sample(data = data_list, chains = 1) + utils::capture.output( + fit <- mod$sample(data = data_list, chains = 1) + ) lp <- fit$log_prob(unconstrained_variables=c(0.6)) expect_equal(lp, -10.649855405830624733) @@ -156,7 +166,9 @@ test_that("unconstrain_variables correctly handles zero-length containers", { mod <- cmdstan_model(write_stan_file(model_code), force_recompile = TRUE, compile_model_methods = TRUE) - fit <- mod$sample(data = list(N = 0), chains = 1) + utils::capture.output( + fit <- mod$sample(data = list(N = 0), chains = 1) + ) unconstrained <- fit$unconstrain_variables(variables = list(x = 5)) expect_equal(unconstrained, 5) }) @@ -179,21 +191,23 @@ test_that("unconstrain_draws returns correct values", { mod <- cmdstan_model(write_stan_file(model_code), compile_model_methods = TRUE, force_recompile = TRUE) - fit <- mod$sample(data = list(N = 0), chains = 2, save_warmup = TRUE) - fit_no_warmup <- mod$sample(data = list(N = 0), chains = 2) + utils::capture.output({ + fit <- mod$sample(data = list(N = 0), chains = 2, save_warmup = TRUE) + fit_no_warmup <- mod$sample(data = list(N = 0), chains = 2) + }) x_draws <- fit$draws(format = "draws_df")$x x_draws_warmup <- fit$draws(format = "draws_df", inc_warmup = TRUE)$x - + # Unconstrain all internal draws unconstrained_internal_draws <- fit$unconstrain_draws() unconstrained_internal_draws_warmup <- fit$unconstrain_draws(inc_warmup = TRUE) expect_equal(as.numeric(x_draws), as.numeric(unconstrained_internal_draws)) expect_equal(as.numeric(x_draws_warmup), as.numeric(unconstrained_internal_draws_warmup)) - + expect_error({unconstrained_internal_draws <- fit_no_warmup$unconstrain_draws(inc_warmup = TRUE)}, "Warmup draws were requested from a fit object without them! Please rerun the model with save_warmup = TRUE.") - + # Unconstrain external CmdStan CSV files unconstrained_csv <- fit$unconstrain_draws(files = fit$output_files()) unconstrained_csv_warmup <- fit$unconstrain_draws(files = fit$output_files(), @@ -204,7 +218,7 @@ test_that("unconstrain_draws returns correct values", { # Unconstrain existing draws object unconstrained_draws <- fit$unconstrain_draws(draws = fit$draws()) expect_equal(as.numeric(x_draws), as.numeric(unconstrained_draws)) - + expect_message(fit$unconstrain_draws(draws = fit$draws(), inc_warmup = TRUE), "'inc_warmup' cannot be used with a draws object. Ignoring.") @@ -224,7 +238,9 @@ test_that("unconstrain_draws returns correct values", { mod <- cmdstan_model(write_stan_file(model_code), compile_model_methods = TRUE, force_recompile = TRUE) - fit <- mod$sample(data = list(N = 0), chains = 2) + utils::capture.output( + fit <- mod$sample(data = list(N = 0), chains = 2) + ) x_draws <- fit$draws(format = "draws_df")$x @@ -241,10 +257,13 @@ test_that("unconstrain_draws returns correct values", { }) test_that("Model methods can be initialised for models with no data", { - stan_file <- write_stan_file("parameters { real x; } model { x ~ std_normal(); }") mod <- cmdstan_model(stan_file, compile_model_methods = TRUE, force_recompile = TRUE) - expect_no_error(fit <- mod$sample()) + expect_no_error( + utils::capture.output( + fit <- mod$sample() + ) + ) expect_equal(fit$log_prob(5), -12.5) }) @@ -268,8 +287,10 @@ test_that("Variable skeleton returns correct dimensions for matrices", { force_recompile = TRUE) N <- 4 K <- 3 - fit <- mod$sample(data = list(N = N, K = K), chains = 1, - iter_warmup = 1, iter_sampling = 5) + utils::capture.output( + fit <- mod$sample(data = list(N = N, K = K), chains = 1, + iter_warmup = 1, iter_sampling = 5) + ) target_skeleton <- list( x_real = array(0, dim = 1), diff --git a/tests/testthat/test-model-optimize.R b/tests/testthat/test-model-optimize.R index 16a18faaa..a3bd89c64 100644 --- a/tests/testthat/test-model-optimize.R +++ b/tests/testthat/test-model-optimize.R @@ -146,9 +146,10 @@ test_that("optimize() method runs when the stan file is removed", { }) test_that("optimize() recognizes new jacobian argument", { - fit <- mod$optimize(data = data_list, jacobian = FALSE) + utils::capture.output({ + fit <- mod$optimize(data = data_list, jacobian = FALSE) + fit2 <- mod$optimize(data = data_list, jacobian = TRUE) + }) expect_equal(fit$metadata()$jacobian, 0) - - fit2 <- mod$optimize(data = data_list, jacobian = TRUE) expect_equal(fit2$metadata()$jacobian, 1) }) diff --git a/tests/testthat/test-model-sample.R b/tests/testthat/test-model-sample.R index 8afe3053a..26cdb6cd7 100644 --- a/tests/testthat/test-model-sample.R +++ b/tests/testthat/test-model-sample.R @@ -350,15 +350,17 @@ test_that("Errors are suppressed with show_exceptions", { " errmod <- cmdstan_model(write_stan_file(errmodcode), force_recompile = TRUE) - expect_message( - suppressWarnings(errmod$sample(data = list(y_mean = 1), chains = 1)), - "Chain 1 Exception: vector[uni] assign: accessing element out of range", - fixed = TRUE - ) + expect_sample_output( + expect_message( + suppressWarnings(errmod$sample(data = list(y_mean = 1), chains = 1)), + "Chain 1 Exception: vector[uni] assign: accessing element out of range", + fixed = TRUE + )) - expect_no_message( - suppressWarnings(errmod$sample(data = list(y_mean = 1), chains = 1, show_exceptions = FALSE)) - ) + expect_sample_output( + expect_no_message( + suppressWarnings(errmod$sample(data = list(y_mean = 1), chains = 1, show_exceptions = FALSE)) + )) }) test_that("All output can be suppressed by show_messages", { diff --git a/tests/testthat/test-model-variables.R b/tests/testthat/test-model-variables.R index 680bb7cdb..a34b72dd1 100644 --- a/tests/testthat/test-model-variables.R +++ b/tests/testthat/test-model-variables.R @@ -110,7 +110,7 @@ test_that("$variables() errors on no stan_file", { }) test_that("$variables() works with #includes, both pre and post compilation.", { - + data_code <- " data { int N; From 4be1269bd2dcc4796c2630a9c4adb96af92d51fe Mon Sep 17 00:00:00 2001 From: jgabry Date: Fri, 22 Nov 2024 12:05:14 -0700 Subject: [PATCH 2/2] a few more cases --- tests/testthat/test-fit-laplace.R | 2 +- tests/testthat/test-fit-shared.R | 4 ++-- tests/testthat/test-model-compile.R | 2 +- tests/testthat/test-model-laplace.R | 23 +++++++++++++---------- tests/testthat/test-model-pathfinder.R | 7 ++++--- tests/testthat/test-threads.R | 10 +++++++--- 6 files changed, 28 insertions(+), 20 deletions(-) diff --git a/tests/testthat/test-fit-laplace.R b/tests/testthat/test-fit-laplace.R index af94bebc8..8991a9e12 100644 --- a/tests/testthat/test-fit-laplace.R +++ b/tests/testthat/test-fit-laplace.R @@ -1,4 +1,4 @@ -context("fitted-vb") +context("fitted-laplace") set_cmdstan_path() fit_laplace <- testing_fit("logistic", method = "laplace", seed = 100) diff --git a/tests/testthat/test-fit-shared.R b/tests/testthat/test-fit-shared.R index a4593c385..0f98e64ac 100644 --- a/tests/testthat/test-fit-shared.R +++ b/tests/testthat/test-fit-shared.R @@ -462,7 +462,7 @@ test_that("sampling with inits works with include_paths", { include_paths = test_path("resources", "stan")) data_list <- list(N = 10, y = c(0,1,0,0,0,0,0,0,0,1)) - expect_no_error( + expect_no_error(utils::capture.output( fit <- mod_w_include$sample( data = data_list, seed = 123, @@ -474,7 +474,7 @@ test_that("sampling with inits works with include_paths", { list(theta = 0.25), list(theta = 0.25)) ) - ) + )) }) test_that("CmdStanModel created with exe_file works", { diff --git a/tests/testthat/test-model-compile.R b/tests/testthat/test-model-compile.R index 2be8390f5..9517aeabf 100644 --- a/tests/testthat/test-model-compile.R +++ b/tests/testthat/test-model-compile.R @@ -837,7 +837,7 @@ test_that("dirname of stan_file is used as include path if no other paths suppli mod_tmp <- cmdstan_model(stan_file, compile = FALSE) expect_true(mod_tmp$check_syntax()) - expect_true(mod_tmp$format()) + utils::capture.output(expect_true(mod_tmp$format())) expect_s3_class(mod_tmp$compile(), "CmdStanModel") }) diff --git a/tests/testthat/test-model-laplace.R b/tests/testthat/test-model-laplace.R index caaef8e48..00dc78bbf 100644 --- a/tests/testthat/test-model-laplace.R +++ b/tests/testthat/test-model-laplace.R @@ -36,7 +36,9 @@ test_that("laplace() method errors for any invalid argument before calling cmdst for (nm in names(bad_arg_values)) { args <- ok_arg_values args[[nm]] <- bad_arg_values[[nm]] - expect_error(do.call(mod$laplace, args), regexp = nm, info = nm) + utils::capture.output( + expect_error(do.call(mod$laplace, args), regexp = nm, info = nm) + ) } args <- ok_arg_values args$opt_args <- list(iter = "NOT_A_NUMBER") @@ -129,14 +131,15 @@ test_that("laplace() errors with bad combinations of arguments", { test_that("laplace() errors if optimize() fails", { mod_schools <- testing_model("schools") - expect_error( - expect_warning( - expect_message( - mod_schools$laplace(data = testing_data("schools"), refresh = 0), - "Line search failed to achieve a sufficient decrease" + utils::capture.output( + expect_error( + expect_warning( + expect_message( + mod_schools$laplace(data = testing_data("schools"), refresh = 0), + "Line search failed to achieve a sufficient decrease" + ), + "Fitting finished unexpectedly" ), - "Fitting finished unexpectedly" - ), - "Optimization failed" - ) + "Optimization failed" + )) }) diff --git a/tests/testthat/test-model-pathfinder.R b/tests/testthat/test-model-pathfinder.R index 3eba55abc..d301588b1 100644 --- a/tests/testthat/test-model-pathfinder.R +++ b/tests/testthat/test-model-pathfinder.R @@ -11,7 +11,7 @@ data_list <- testing_data("bernoulli") data_file_r <- test_path("resources", "data", "bernoulli.data.R") data_file_json <- test_path("resources", "data", "bernoulli.data.json") -# these are all valid for sample() +# these are all valid for pathfinder() ok_arg_values <- list( data = data_list, output_dir = tempdir(), @@ -32,9 +32,10 @@ ok_arg_values <- list( max_lbfgs_iters = 100, save_single_paths = FALSE, calculate_lp = TRUE, - psis_resample=TRUE) + psis_resample=TRUE +) -# using any one of these should cause sample() to error +# using any one of these should cause pathfinder() to error bad_arg_values <- list( data = "NOT_A_FILE", output_dir = "NOT_A_DIRECTORY", diff --git a/tests/testthat/test-threads.R b/tests/testthat/test-threads.R index 1a333e82c..fb5eec615 100644 --- a/tests/testthat/test-threads.R +++ b/tests/testthat/test-threads.R @@ -172,9 +172,13 @@ test_that("correct output when stan_threads not TRUE", { fixed = TRUE ) mod <- cmdstan_model(stan_program, cpp_options = list(stan_threads = FALSE), force_recompile = TRUE) - expect_warning( - mod$sample(data = data_file_json, threads_per_chain = 4), - "'threads_per_chain' is set but the model was not compiled with 'cpp_options = list(stan_threads = TRUE)' so 'threads_per_chain' will have no effect!", + expect_output( + expect_warning( + mod$sample(data = data_file_json, threads_per_chain = 4), + "'threads_per_chain' is set but the model was not compiled with 'cpp_options = list(stan_threads = TRUE)' so 'threads_per_chain' will have no effect!", + fixed = TRUE + ), + "Running MCMC with 4 sequential chains", fixed = TRUE ) })