From b4c8e5d5767925cfb374c48c83fa8821839981c2 Mon Sep 17 00:00:00 2001 From: kamilsi Date: Fri, 8 Mar 2024 14:54:05 +0000 Subject: [PATCH] =?UTF-8?q?Deploying=20to=20gh-pages=20from=20@=20ttscienc?= =?UTF-8?q?e/unbiased@563557ed9258ba30320b852d70b8edc6d946f057=20?= =?UTF-8?q?=F0=9F=9A=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- deps/JetBrains_Mono-0.4.8/font.css | 2 +- ...BX0PnT8RD8yKxjPg&skey=48ad01c60053c2ae&v=v18 | Bin 0 -> 52536 bytes deps/Montserrat-0.4.8/font.css | 2 +- index.html | 4 +--- pkgdown.yml | 2 +- search.json | 2 +- 6 files changed, 5 insertions(+), 7 deletions(-) create mode 100644 deps/JetBrains_Mono-0.4.8/font?kit=tDbY2o-flEEny0FZhsfKu5WU4zr3E_BX0PnT8RD8yKxjPg&skey=48ad01c60053c2ae&v=v18 diff --git a/deps/JetBrains_Mono-0.4.8/font.css b/deps/JetBrains_Mono-0.4.8/font.css index 395dd07..0d9466d 100644 --- a/deps/JetBrains_Mono-0.4.8/font.css +++ b/deps/JetBrains_Mono-0.4.8/font.css @@ -3,5 +3,5 @@ font-style: normal; font-weight: 400; font-display: swap; - src: url(tDbY2o-flEEny0FZhsfKu5WU4zr3E_BX0PnT8RD8yKxjPg.woff) format('woff'); + src: url(font?kit=tDbY2o-flEEny0FZhsfKu5WU4zr3E_BX0PnT8RD8yKxjPg&skey=48ad01c60053c2ae&v=v18) format('woff'); } diff --git a/deps/JetBrains_Mono-0.4.8/font?kit=tDbY2o-flEEny0FZhsfKu5WU4zr3E_BX0PnT8RD8yKxjPg&skey=48ad01c60053c2ae&v=v18 b/deps/JetBrains_Mono-0.4.8/font?kit=tDbY2o-flEEny0FZhsfKu5WU4zr3E_BX0PnT8RD8yKxjPg&skey=48ad01c60053c2ae&v=v18 new file mode 100644 index 0000000000000000000000000000000000000000..47db68d325271e7f83a0c7e56c7d40a7835075e1 GIT binary patch literal 52536 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 rAEhOl@avr#ump_&$N13y literal 0 HcmV?d00001 diff --git a/deps/Montserrat-0.4.8/font.css b/deps/Montserrat-0.4.8/font.css index 3691193..e56614e 100644 --- a/deps/Montserrat-0.4.8/font.css +++ b/deps/Montserrat-0.4.8/font.css @@ -3,5 +3,5 @@ font-style: normal; font-weight: 400; font-display: swap; - src: url(font?kit=JTUHjIg1_i6t8kCHKm4532VJOt5-QNFgpCtr6Ew9&skey=7bc19f711c0de8f&v=v26) format('woff'); + src: url(JTUHjIg1_i6t8kCHKm4532VJOt5-QNFgpCtr6Ew9.woff) format('woff'); } diff --git a/index.html b/index.html index 04f7acd..6b213e0 100644 --- a/index.html +++ b/index.html @@ -72,9 +72,7 @@ unbiased -
-

An API-based solution for Clinical Trial Randomization

-
+

Test CoverageDocsDocker Publishcodecov > An API-based solution for Clinical Trial Randomization

In clinical trials, the fair and efficient allocation of participants is essential for achieving reliable results. While there are many excellent R randomization packages available, none, to our knowledge, provide a dedicated API for this purpose. The unbiased package fills this gap by featuring a production-ready REST API designed for seamless integration. This unique combination allows for easy connection with electronic Case Report Forms (eCRF), enhancing data management and streamlining participant allocation.

Why choose unbiased? diff --git a/pkgdown.yml b/pkgdown.yml index 1f59a0e..a1e9c83 100644 --- a/pkgdown.yml +++ b/pkgdown.yml @@ -3,7 +3,7 @@ pkgdown: 2.0.7 pkgdown_sha: ~ articles: minimization_randomization_comparison: minimization_randomization_comparison.html -last_built: 2024-03-08T14:33Z +last_built: 2024-03-08T14:50Z urls: reference: https://ttscience.github.io/unbiased/reference article: https://ttscience.github.io/unbiased/articles diff --git a/search.json b/search.json index 37a900d..96be6c1 100644 --- a/search.json +++ b/search.json @@ -1 +1 @@ -[{"path":"https://ttscience.github.io/unbiased/LICENSE.html","id":null,"dir":"","previous_headings":"","what":"MIT License","title":"MIT License","text":"Copyright (c) 2024 Transition Technologies Science sp. z o.o. Permission hereby granted, free charge, person obtaining copy software associated documentation files (“Software”), deal Software without restriction, including without limitation rights use, copy, modify, merge, publish, distribute, sublicense, /sell copies Software, permit persons Software furnished , subject following conditions: copyright notice permission notice shall included copies substantial portions Software. SOFTWARE PROVIDED “”, WITHOUT WARRANTY KIND, EXPRESS IMPLIED, INCLUDING LIMITED WARRANTIES MERCHANTABILITY, FITNESS PARTICULAR PURPOSE NONINFRINGEMENT. EVENT SHALL AUTHORS COPYRIGHT HOLDERS LIABLE CLAIM, DAMAGES LIABILITY, WHETHER ACTION CONTRACT, TORT OTHERWISE, ARISING , CONNECTION SOFTWARE USE DEALINGS SOFTWARE.","code":""},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"introduction","dir":"Articles","previous_headings":"","what":"Introduction","title":"Benchmarking randomization methods","text":"Randomization clinical trials gold standard widely considered best design evaluating effectiveness new treatments compared alternative treatments (standard care) placebo. Indeed, selection appropriate randomisation important selection appropriate statistical analysis study analysis strategy, whether based randomisation population model (Berger et al. (2021)). One primary advantages randomization, particularly simple randomization (usually using flipping coin method), ability balance confounding variables across treatment groups. especially effective large sample sizes (n > 200), random allocation participants helps ensure known unknown confounders evenly distributed study arms. balanced distribution contributes significantly internal validity study, minimizes risk selection bias confounding influencing results (Lim (2019)). ’s important note, however, simple randomization powerful large trials, may always guarantee even distribution confounding factors trials smaller sample sizes (n < 100). cases, random allocation might result imbalances baseline characteristics groups, can affect interpretation treatment’s effectiveness. potential limitation sets stage considering additional methods, stratified randomization, dynamic minimization algorithms address challenges smaller trials (Kang, Ragan, Park (2008)). document provides summary comparison three randomization methods: simple randomization, block randomization, adaptive randomization. Simple randomization adaptive randomization (minimization method) tools available unbiased package randomize_simple randomize_minimisation_pocock functions (Sijko et al. (2024)). comparison aims demonstrate superiority adaptive randomization (minimization method) methods assessing least imbalance accompanying variables therapeutic groups. Monte Carlo simulations used generate data, utilizing simstudy package (Goldfeld Wujciak-Jens (2020)). Parameters binary distribution variables based data publication Mrozikiewicz-Rakowska et al. (2023) information researchers. document structure follows: first, based defined parameters, data simulated using Monte Carlo method single simulation; , generated patient data, appropriate groups assigned using three randomization methods; data summarized form descriptive statistics along relevant statistical test; next, data prepared .Rds format generated 1000 simulations loaded., results based standardised mean difference (SMD) test discussed visual form (boxplot, violin plot) percentage success achieved method given precision (tabular summary)","code":"# load packages library(unbiased) library(dplyr) library(simstudy) library(tableone) library(ggplot2) library(gt) library(gtsummary) library(truncnorm) library(tidyr) library(randomizeR)"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"the-randomization-methods-considered-for-comparison","dir":"Articles","previous_headings":"","what":"The randomization methods considered for comparison","title":"Benchmarking randomization methods","text":"process comparing balance covariates among randomization methods, three randomization methods selected evaluation: simple randomization - simple coin toss, algorithm gives participants equal chances assigned particular arm. method’s advantage lies simplicity elimination predictability. However, due complete randomness, may lead imbalance sample sizes arms imbalances prognostic factors. large sample size (n > 200), simple randomisation gives similar number generated participants group. small sample size (n < 100), results imbalance (Kang, Ragan, Park (2008)). block randomization - randomization method takes account defined covariates patients. method involves assigning patients therapeutic arms blocks fixed size, recommendation blocks different sizes. , extent, reduces risk researchers predicting future arm assignments. contrast simple randomization, block method aims balance number patients within block, hence reducing overall imbalance arms (Rosenberger Lachin (2015)). adaptive randomization using minimization method based Pocock Simon (1975) algorithm - - randomization approach aims balance prognostic factors across treatment arms within clinical study. functions evaluating total imbalance factors time new patient considered study. minimization method computes overall imbalance potential arm assignment new patient, considering factors like variance specified criteria. patient assigned arm addition results smallest total imbalance. assignment deterministic made predetermined probability, ensuring level randomness arm allocation. method particularly useful trials multiple prognostic factors smaller studies traditional randomization might fail achieve balance.","code":""},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"assessment-of-covariate-balance","dir":"Articles","previous_headings":"","what":"Assessment of covariate balance","title":"Benchmarking randomization methods","text":"proposed approach assessment randomization methods, primary objective evaluate method terms achieving balance specified covariates. assessment balance aims determine whether distributions covariates similarly balanced therapeutic group. Based literature, standardized mean differences (SMD) employed assessing balance (Berger et al. (2021)). SMD method one commonly used statistics assessing balance covariates, regardless unit measurement. statistical measure comparing differences two groups. covariates examined case expressed binary variables. case categorical variables, SMD calculated using following formula (Zhang et al. (2019)): \\[ SMD = \\frac{{p_1 - p_2}}{{\\sqrt{\\frac{{p_1 \\cdot (1 - p_1) + p_2 \\cdot (1 - p_2)}}{2}}}} \\], : \\(p_1\\) proportion first arm, \\(p_2\\) proportion second arm.","code":""},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"definied-number-of-patients","dir":"Articles","previous_headings":"","what":"Definied number of patients","title":"Benchmarking randomization methods","text":"simulation, using real use case - planned FootCell study - non-commercial clinical research area civilisation diseases - guide data generation process. FootCell study, anticipated total 105 patients randomized trial. patients equally divided among three research groups - Group , Group B, Group C - group comprising 35 patients.","code":"# defined number of patients n <- 105"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"defining-parameters-for-monte-carlo-simulation","dir":"Articles","previous_headings":"","what":"Defining parameters for Monte-Carlo simulation","title":"Benchmarking randomization methods","text":"distribution parameters individual covariates, subsequently used validate randomization methods, defined using publication Mrozikiewicz-Rakowska et al. (2023) allogenic interventions.. publication describes effectiveness comparing therapy using ADSC (Adipose-Derived Stem Cells) gel versus standard therapy fibrin gel patients diabetic foot ulcer treatment. FootCell study also aims assess safety advanced therapy involving live ASCs (Adipose-Derived Stem Cells) treatment diabetic foot syndrome, considering two groups treated ADSCs (one two administrations) compared fibrin gel. Therefore, appropriate population data extracted publication determine distributions can maintained designing FootCell study. process defining study randomization, following covariates selected: gender [male/female], diabetes type [type /type II], HbA1c [9/9 11] [%], tpo2 [50/50] [mmHg], age [55/55] [years], wound size [2/2] [cm\\(^2\\)]. case variables gender diabetes type publication Mrozikiewicz-Rakowska et al. (2023), expressed form frequencies. remaining variables presented terms measures central tendency along indication variability, well minimum maximum values. determine parameters binary distribution, truncated normal distribution available truncnorm package utilized. truncated normal distribution often used statistics probability modeling dealing data constrained certain range. particularly useful want model random variable take values beyond certain limits (Burkardt (2014)). generate necessary information remaining covariates, function simulate_proportions_trunc written, utilizing rtruncnorm function (Mersmann et al. (2023)). parameters mean, sd, lower, upper taken publication based expertise regarding ranges parameters. results presented table, assuming outcome refers first category parameter.","code":"# simulate parameters using truncated normal distribution simulate_proportions_trunc <- function(n, lower, upper, mean, sd, threshold) { simulate_data <- rtruncnorm( n = n, a = lower, b = upper, mean = mean, sd = sd ) <= threshold sum(simulate_data == TRUE) / n } set.seed(123) data.frame( hba1c = simulate_proportions_trunc(1000, 0, 11, 7.41, 1.33, 9), tpo2 = simulate_proportions_trunc(1000, 30, 100, 53.4, 18.4, 50), age = simulate_proportions_trunc(1000, 0, 100, 59.2, 9.7, 55), wound_size = simulate_proportions_trunc(1000, 0, 20, 2.7, 2.28, 2) ) |> rename(\"wound size\" = wound_size) |> pivot_longer( cols = everything(), names_to = \"parametr\", values_to = \"proportions\" ) |> mutate(\"first catogory of strata\" = c(\"<=9\", \"<=50\", \"<=55\", \"<=2\")) |> gt()"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"generate-data-using-monte-carlo-simulations","dir":"Articles","previous_headings":"","what":"Generate data using Monte-Carlo simulations","title":"Benchmarking randomization methods","text":"Monte-Carlo simulations used accumulate data. method designed model variables based defined parameters. Variables defined using simstudy package, utilizing defData function (Goldfeld Wujciak-Jens (2020)). variables specify proportions, dist = 'binary' used define variables. Due likely association type diabetes age – meaning older patient, higher probability type II diabetes – relationship diabetes established defining age variable using logit function link = \"logit\". proportions gender diabetes defined researchers consistent literature Mrozikiewicz-Rakowska et al. (2023). Using genData function simstudy package, data frame (data) generated artificially adopted variable arm, filled subsequent randomization methods arm allocation process n patients.","code":"# defining variables # male - 0.9 def <- simstudy::defData(varname = \"sex\", formula = \"0.9\", dist = \"binary\") # type I - 0.15 def <- simstudy::defData(def, varname = \"diabetes_type\", formula = \"0.15\", dist = \"binary\") # <= 9 - 0.888 def <- simstudy::defData(def, varname = \"hba1c\", formula = \"0.888\", dist = \"binary\") # <= 50 - 0.354 def <- simstudy::defData(def, varname = \"tpo2\", formula = \"0.354\", dist = \"binary\") # correlation with diabetes type def <- simstudy::defData( def, varname = \"age\", formula = \"(diabetes_type == 0) * (-0.95)\", link = \"logit\", dist = \"binary\" ) # <= 2 - 0.302 def <- simstudy::defData(def, varname = \"wound_size\", formula = \"0.302\", dist = \"binary\") # generate data using genData() data <- genData(n, def) |> mutate( sex = as.character(sex), age = as.character(age), diabetes_type = as.character(diabetes_type), hba1c = as.character(hba1c), tpo2 = as.character(tpo2), wound_size = as.character(wound_size) ) |> as_tibble() # add arm to tibble data <- data |> tibble::add_column(arm = \"\") # first 5 rows of the data head(data, 5) |> gt()"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"minimization-randomization","dir":"Articles","previous_headings":"","what":"Minimization randomization","title":"Benchmarking randomization methods","text":"generate appropriate research arms, function called minimize_results written, utilizing randomize_minimisation_pocock function available within unbiased package (Sijko et al. (2024)). probability parameter set level defined within function (p = 0.85). case minimization randomization, verify type minimization (equal weights unequal weights) used, three calls minimize_results function prepared: minimize_equal_weights - covariate weight takes value equal 1 divided number covariates. case, weight 1/6, minimize_unequal_weights - following expert assessment physicians, parameters potentially significant impact treatment outcomes (hba1c, tpo2, wound size) assigned weight 2. remaining covariates assigned weight 1. minimize_unequal_weights_3 - following expert assessment physicians, parameters potentially significant impact treatment outcomes (hba1c, tpo2, wound size) assigned weight 3. remaining covariates assigned weight 1. tables present information allocations first 5 patients. statistic_table function developed provide information : distribution number patients across research arms, distribution covariates across research arms, along p-value information statistical analyses used compare proportions - chi^2, exact Fisher’s test, typically used small samples. function relies use tbl_summary function available gtsummary package (Sjoberg et al. (2021)). table presents statistical summary results first iteration : Minimization weights equal 1/6. Minimization weights 2:1. Minimization weights 3:1.","code":"# drawing an arm for each patient minimize_results <- function(current_data, arms, weights) { for (n in seq_len(nrow(current_data))) { current_state <- current_data[1:n, 2:ncol(current_data)] current_data$arm[n] <- randomize_minimisation_pocock( arms = arms, current_state = current_state, weights = weights ) } return(current_data) } set.seed(123) # eqal weights - 1/6 minimize_equal_weights <- minimize_results( current_data = data, arms = c(\"armA\", \"armB\", \"armC\") ) head(minimize_equal_weights, 5) |> gt() set.seed(123) # double weights where the covariant is of high clinical significance minimize_unequal_weights <- minimize_results( current_data = data, arms = c(\"armA\", \"armB\", \"armC\"), weights = c( \"sex\" = 1, \"diabetes_type\" = 1, \"hba1c\" = 2, \"tpo2\" = 2, \"age\" = 1, \"wound_size\" = 2 ) ) head(minimize_unequal_weights, 5) |> gt() set.seed(123) # triple weights where the covariant is of high clinical significance minimize_unequal_weights_3 <- minimize_results( current_data = data, arms = c(\"armA\", \"armB\", \"armC\"), weights = c( \"sex\" = 1, \"diabetes_type\" = 1, \"hba1c\" = 3, \"tpo2\" = 3, \"age\" = 1, \"wound_size\" = 3 ) ) head(minimize_unequal_weights_3, 5) |> gt() # generation of frequency and chi^2 statistic values or fisher exact test statistics_table <- function(data) { data |> mutate( sex = ifelse(sex == \"1\", \"men\", \"women\"), diabetes_type = ifelse(diabetes_type == \"1\", \"type1\", \"type2\"), hba1c = ifelse(hba1c == \"1\", \"<=9\", \"(9,11>\"), tpo2 = ifelse(tpo2 == \"1\", \"<=50\", \">50\"), age = ifelse(age == \"1\", \"<=55\", \">55\"), wound_size = ifelse(wound_size == \"1\", \"<=2\", \">2\") ) |> tbl_summary( include = c(sex, diabetes_type, hba1c, tpo2, age, wound_size), by = arm ) |> modify_header(label = \"\") |> modify_header(all_stat_cols() ~ \"**{level}**, N = {n}\") |> bold_labels() |> add_p() } statistics_table(minimize_equal_weights) statistics_table(minimize_unequal_weights) statistics_table(minimize_unequal_weights_3)"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"simple-randomization","dir":"Articles","previous_headings":"","what":"Simple randomization","title":"Benchmarking randomization methods","text":"next step, appropriate arms generated patients using simple randomization, available unbiased package - randomize_simple function (Sijko et al. (2024)). simple_results function called within simple_data, considering initial assumption assigning patients three arms 1:1:1 ratio. Since simple randomization, take account initial covariates, treatment assignment occurs randomly (flip coin method). tables illustrate example data output summary statistics including summary statistical tests.","code":"# simple randomization simple_results <- function(current_data, arms, ratio) { for (n in seq_len(nrow(current_data))) { current_data$arm[n] <- randomize_simple(arms, ratio) } return(current_data) } set.seed(123) simple_data <- simple_results( current_data = data, arms = c(\"armA\", \"armB\", \"armC\"), ratio = c(\"armB\" = 1L, \"armA\" = 1L, \"armC\" = 1L) ) head(simple_data, 5) |> gt() statistics_table(simple_data)"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"block-randomization","dir":"Articles","previous_headings":"","what":"Block randomization","title":"Benchmarking randomization methods","text":"Block randomization, opposed minimization simple randomization methods, developed based rbprPar function available randomizeR package (Uschner et al. (2018)). Using , block_rand function created, , based defined number patients, arms, list stratifying factors, generates randomization list length equal number patients multiplied product categories covariate. case specified data document, one iteration, amounts 105 * 2^6 = 6720 rows. ensures appropriate number randomisation codes opportunity. case equal characteristics, certain right number codes defined n patients. Based block_rand function, possible generate randomisation list, based patients allocated, characteristics output data frame. Due 3 arms need blind allocation consecutive patients, block sizes 3,6 9 used calculations. next step, patients assigned research groups using block_results function (based list generated function block_rand). first available code randomization list meets specific conditions selected, removed list available codes. Based , research arms generated ensure appropriate number patients group (based assumed ratio 1:1:1). tables show assignment patients groups using block randomisation summary statistics including summary statistical tests.","code":"# Function to generate a randomisation list block_rand <- function(n, block, n_groups, strata, arms = LETTERS[1:n_groups]) { strata_grid <- expand.grid(strata) strata_n <- nrow(strata_grid) ratio <- rep(1, n_groups) gen_seq_list <- lapply(seq_len(strata_n), function(i) { rand <- rpbrPar( N = n, rb = block, K = n_groups, ratio = ratio, groups = arms, filledBlock = FALSE ) getRandList(genSeq(rand))[1, ] }) df_list <- tibble::tibble() for (i in seq_len(strata_n)) { local_df <- strata_grid |> dplyr::slice(i) |> dplyr::mutate(count_n = n) |> tidyr::uncount(count_n) |> tibble::add_column(rand_arm = gen_seq_list[[i]]) df_list <- rbind(local_df, df_list) } return(df_list) } # Generate a research arm for patients in each iteration block_results <- function(current_data) { simulation_result <- block_rand( n = n, block = c(3, 6, 9), n_groups = 3, strata = list( sex = c(\"0\", \"1\"), diabetes_type = c(\"0\", \"1\"), hba1c = c(\"0\", \"1\"), tpo2 = c(\"0\", \"1\"), age = c(\"0\", \"1\"), wound_size = c(\"0\", \"1\") ), arms = c(\"armA\", \"armB\", \"armC\") ) for (n in seq_len(nrow(current_data))) { # \"-1\" is for \"arm\" column current_state <- current_data[n, 2:(ncol(current_data) - 1)] matching_rows <- which(apply( simulation_result[, -ncol(simulation_result)], 1, function(row) all(row == current_state) )) if (length(matching_rows) > 0) { current_data$arm[n] <- simulation_result[matching_rows[1], \"rand_arm\"] # Delete row from randomization list simulation_result <- simulation_result[-matching_rows[1], , drop = FALSE] } } return(current_data) } set.seed(123) block_data <- block_results(data) head(block_data, 5) |> gt() statistics_table(block_data)"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"generate-1000-simulations","dir":"Articles","previous_headings":"","what":"Generate 1000 simulations","title":"Benchmarking randomization methods","text":"performed 1000 iterations data generation parameters defined . number iterations indicates number iterations included Monte-Carlo simulations accumulate data given parameters. allowed generation data 1000 times 105 patients efficiently assess effect randomization methods context covariate balance. data assigned variable sim_data based data stored .Rds file 1000_sim_data.Rds, available within vignette information GitHub repository unbiased package.","code":"# define number of iterations # no_of_iterations <- 1000 # nolint # define number of cores # no_of_cores <- 20 # nolint # perform simulations (run carefully!) # source(\"~/unbiased/vignettes/helpers/run_parallel.R\") # nolint # read data from file sim_data <- readRDS(\"1000_sim_data.Rds\")"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"check-balance-using-smd-test","dir":"Articles","previous_headings":"","what":"Check balance using smd test","title":"Benchmarking randomization methods","text":"order select test define precision specified level, assume imbalance, literature analysis conducted based publications Lee et al. (2021), Austin (2009), Doah et al. (2021), Brown et al. (2020), Nguyen et al. (2017), Sánchez-Meca, Marı́n-Martı́nez, Chacón-Moscoso (2003), Lee, Acharya, et al. (2022), Berger et al. (2021). assess balance covariates research groups , B, C, Standardized Mean Difference (SMD) test employed, compares two groups. Since three groups example, SMD test computed pair comparisons: vs B, vs C, B vs C. average SMD test given covariate calculated based comparisons. literature analysis, precision level ranged 0.1-0.2. small samples, expected SMD test exceed 0.2 (Austin (2009)). Additionally, according publication Sánchez-Meca, Marı́n-Martı́nez, Chacón-Moscoso (2003), golden standard dictates specific threshold SMD test considered balanced. Generally, smaller SMD test, smaller difference covariate imbalance. analyzed example, due sample size 105 patients, threshold 0.2 SMD test adopted. function called smd_covariants_data written generate frames produce SMD test covariate iteration, utilizing CreateTableOne function available tableone package (Yoshida Bartel (2022)). cases test result <0.001, value 0 assigned. results randomization method stored cov_balance_data. results SMD test presented form boxplot violin plot, depicting outcomes randomization method. red dashed line indicates adopted precision threshold. Boxplot combined results Summary average smd randomization methods Violin plot Summary smd randomization methods covariants Summary table success Based specified precision threshold 0.2, function defining randomization success, named success_power, developed. SMD test value covariate given iteration 0.2, function defines analysis data ‘failure’ - 0; otherwise, defined ‘success’ - 1. final success power calculated sum successes iteration divided total number specified iterations. results summarized table percentage success randomization method.","code":"# definied covariants vars <- c(\"sex\", \"age\", \"diabetes_type\", \"wound_size\", \"tpo2\", \"hba1c\") smd_covariants_data <- function(data, vars, strata) { result_table <- lapply(unique(data$simnr), function(i) { current_data <- data[data$simnr == i, ] arms_to_check <- setdiff(names(current_data), c(vars, \"id\", \"simnr\")) # check SMD for any covariants lapply(arms_to_check, function(arm) { tab <- CreateTableOne( vars = vars, data = current_data, strata = arm ) results_smd <- ExtractSmd(tab) |> as.data.frame() |> tibble::rownames_to_column(\"covariants\") |> select(covariants, results = average) |> mutate(results = round(as.numeric(results), 3)) results <- bind_cols( simnr = i, strata = arm, results_smd ) return(results) }) |> bind_rows() }) |> bind_rows() return(result_table) } cov_balance_data <- smd_covariants_data( data = sim_data, vars = vars ) |> mutate(method = case_when( strata == \"minimize_equal_weights_arms\" ~ \"minimize equal\", strata == \"minimize_unequal_weights_arms\" ~ \"minimize unequal 2:1\", strata == \"minimize_unequal_weights_triple_arms\" ~ \"minimize unequal 3:1\", strata == \"simple_data_arms\" ~ \"simple randomization\", strata == \"block_data_arms\" ~ \"block randomization\" )) |> select(-strata) # boxplot cov_balance_data |> select(simnr, results, method) |> group_by(simnr, method) |> mutate(results = mean(results)) |> distinct() |> ggplot(aes(x = method, y = results, fill = method)) + geom_boxplot() + geom_hline(yintercept = 0.2, linetype = \"dashed\", color = \"red\") + theme_bw() # violin plot cov_balance_data |> ggplot(aes(x = method, y = results, fill = method)) + geom_violin() + geom_hline( yintercept = 0.2, linetype = \"dashed\", color = \"red\" ) + facet_wrap(~covariants, ncol = 3) + theme_bw() + theme(axis.text = element_text(angle = 45, vjust = 0.5, hjust = 1)) # function defining success of randomisation success_power <- function(cov_data) { result_table <- lapply(unique(cov_data$simnr), function(i) { current_data <- cov_data[cov_data$simnr == i, ] current_data |> group_by(method) |> summarise(success = ifelse(any(results > 0.2), 0, 1)) |> tibble::add_column(simnr = i, .before = 1) }) |> bind_rows() success <- result_table |> group_by(method) |> summarise(results_power = sum(success) / n() * 100) return(success) } success_power(cov_balance_data) |> as.data.frame() |> rename(`power results [%]` = results_power) |> gt()"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"conclusion","dir":"Articles","previous_headings":"","what":"Conclusion","title":"Benchmarking randomization methods","text":"Considering three randomization methods: minimization, block randomization, simple randomization, minimization performs best terms covariate balance. Simple randomization significant drawback, patient allocation arms occurs randomly equal probability. leads imbalance number patients covariate balance, also random. particularly case small samples. Balancing number patients possible larger samples n > 200. hand, block randomization performs well balancing number patients groups specified allocation ratio. However, compared adaptive randomisation using minimisation method, block randomisation lower probability terms balancing co-variables. Minimization method, provides highest success power ensuring balance across covariates groups. made possible appropriate algorithm implemented part minimisation randomisation. assigning next patient group, method examines total imbalance assigns patient appropriate study group specified probability balance sample terms size, covariates.","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/authors.html","id":null,"dir":"","previous_headings":"","what":"Authors","title":"Authors and Citation","text":"Kamil Sijko. Author, maintainer. Kinga Sałata. Author. Aleksandra Duda. Author. Łukasz Wałejko. Author. Jagoda Głowacka-Walas. Author. Laura Bąkała. Author. Michał Seweryn. Contributor. Transition Technologies Science Sp. z o.o.. Funder, copyright holder.","code":""},{"path":"https://ttscience.github.io/unbiased/authors.html","id":"citation","dir":"","previous_headings":"","what":"Citation","title":"Authors and Citation","text":"Sijko K, Sałata K, Duda , Wałejko Ł, Głowacka-Walas J, Bąkała L (2024). unbiased: Unbiased: Production-Grade Randomization API. R package version 1.0.0, https://ttscience.github.io/unbiased/.","code":"@Manual{, title = {unbiased: Unbiased: Production-Grade Randomization API}, author = {Kamil Sijko and Kinga Sałata and Aleksandra Duda and Łukasz Wałejko and Jagoda Głowacka-Walas and Laura Bąkała}, year = {2024}, note = {R package version 1.0.0}, url = {https://ttscience.github.io/unbiased/}, }"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"unbiased-","dir":"","previous_headings":"","what":"Unbiased: Production-Grade Randomization API","title":"Unbiased: Production-Grade Randomization API","text":"API-based solution Clinical Trial Randomization clinical trials, fair efficient allocation participants essential achieving reliable results. many excellent R randomization packages available, none, knowledge, provide dedicated API purpose. unbiased package fills gap featuring production-ready REST API designed seamless integration. unique combination allows easy connection electronic Case Report Forms (eCRF), enhancing data management streamlining participant allocation.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"why-choose-unbiased","dir":"","previous_headings":"","what":"Why choose unbiased?","title":"Unbiased: Production-Grade Randomization API","text":"goal creating unbiased provide user-friendly yet powerful tool addresses nuanced demands clinical trial randomization. offers: Production-Ready REST API: Built seamless integration eCRF/EDC systems, facilitating real-time randomization automation. Extensive Database Integration: Supports robust data management practices, ensuring participant information randomization outcomes securely managed easily accessible. Commitment Quality: Emphasizes development best practices, including comprehensive code coverage, deliver reliable trustworthy solution. Adaptability: Whether small-scale studies large, multi-center trials, unbiased scales meet needs. Comprehensive Documentation: support applying package effectively. choosing unbiased, ’re adopting sophisticated approach trial randomization, ensuring fair efficient participant allocation across studies support broader objectives clinical research technology.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"table-of-contents","dir":"","previous_headings":"","what":"Table of Contents","title":"Unbiased: Production-Grade Randomization API","text":"Quick Setup Docker Compose Quick Setup Docker API Server Configuration Alternative Installation Method Study Creation Patient Randomization Study List Study Details Randomization List Audit Log Quality Assurance Measures Executing Tests R Interactive Session Executing Tests Command Line Running Tests Docker Compose Code Coverage Configuring Sentry Purpose Scope Clinical Trial Randomization Comparison Solutions","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"quickstart-guide","dir":"","previous_headings":"","what":"Quickstart Guide","title":"Unbiased: Production-Grade Randomization API","text":"Initiating work unbiased involves simple setup steps. Whether ’re integrating R environment deploying API, aim equip reliable tool enhances integrity efficiency clinical trials.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"quick-setup-with-docker-compose","dir":"","previous_headings":"","what":"Quick Setup with Docker Compose","title":"Unbiased: Production-Grade Randomization API","text":"prepared simple Docker Compose setup allows run unbiased minimal configuration. method ideal users want quickly test unbiased locally development environment. get started, follow steps: Clone unbiased repository local machine just copy docker-compose.yml file repository. Navigate directory containing docker-compose.yml file. Run following command start unbiased: command start unbiased API server, making accessible specified port. PostgreSQL also started, ensuring API can connect database. can use docker-compose.yml file starting point deployment, modifying suit specific requirements.","code":"docker compose pull docker compose up"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"configuration","dir":"","previous_headings":"Quick Setup with Docker Compose","what":"Configuration","title":"Unbiased: Production-Grade Randomization API","text":"docker-compose.yml file contains necessary configuration running unbiased Docker Compose. can override default environment variables creating .env file directory docker-compose.yml file. following environment variables can set: UNBIASED_PORT: port API listen. Defaults 3838 provided. POSTGRES_USER: username authentication PostgreSQL database. Defaults postgres provided. POSTGRES_PASSWORD: password authentication PostgreSQL database. Warning: can set first run, work database already exists. Defaults postgres provided. UNBIASED_VERSION: version unbiased Docker image use. Defaults latest provided. POSTGRES_VERSION: version PostgreSQL Docker image use. Defaults latest provided.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"building-the-docker-image-locally","dir":"","previous_headings":"Quick Setup with Docker Compose","what":"Building the Docker Image Locally","title":"Unbiased: Production-Grade Randomization API","text":"cloned repository, can also build Docker image locally using following command:","code":"docker compose build"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"quick-setup-with-docker","dir":"","previous_headings":"","what":"Quick Setup with Docker","title":"Unbiased: Production-Grade Randomization API","text":"straightforward way deploy unbiased Docker images. ensures can get unbiased running minimal setup, regardless local environment. use unbiased, pull latest Docker image: run unbiased Docker, ensuring set necessary environment variables: command starts unbiased API, making accessible specified port. ’s crucial PostgreSQL database ready, unbiased automatically configure necessary database structures upon startup. PostgreSQL can run separate container local machine. Make sure use PostgreSQL instance temporal_tables extension available. can use ghcr.io/ttscience/postgres-temporal-tables/postgres-temporal-tables:latest image run PostgreSQL temporal_tables extension.","code":"docker pull ghcr.io/ttscience/unbiased docker run -e POSTGRES_DB=mydb -e POSTGRES_USER=myuser -e POSTGRES_PASSWORD=mypassword -e UNBIASED_PORT=3838 ghcr.io/ttscience/unbiased"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"api-server-configuration","dir":"","previous_headings":"","what":"API Server configuration","title":"Unbiased: Production-Grade Randomization API","text":"unbiased API server can configured using environment variables. following environment variables need set server start: POSTGRES_DB: name PostgreSQL database connect . POSTGRES_HOST: host PostgreSQL database. hostname, localhost database.example.com, IP address. POSTGRES_PORT: port PostgreSQL database listening. Defaults 5432 provided. POSTGRES_USER: username authentication PostgreSQL database. POSTGRES_PASSWORD: password authentication PostgreSQL database. UNBIASED_HOST: host API run. Defaults 0.0.0.0 provided. UNBIASED_PORT: port API listen. Defaults 3838 provided.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"alternative-installation-method","dir":"","previous_headings":"","what":"Alternative Installation Method","title":"Unbiased: Production-Grade Randomization API","text":"preferring work directly within R environment, unbiased package offers alternative installation method via GitHub. approach allows users easily integrate unbiased R projects. proceed method, utilize devtools package installation executing following command: Following package installation, unbiased API can launched within R. Simply invoke run_unbiased() function start API: initiates API server, default, local machine (http://localhost:3838), making accessible interaction various HTTP clients, including curl, Postman, R’s httr package.","code":"devtools::install_github(\"ttscience/unbiased\") unbiased::run_unbiased()"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"getting-started-with-unbiased","dir":"","previous_headings":"","what":"Getting started with unbiased","title":"Unbiased: Production-Grade Randomization API","text":"unbiased package offers functions randomizing participants clinical trials, ensuring fair transparent process. Complete documentation implemented methodology examples use available GitHub Pages, providing information need integrate unbiased trial management workflow. , present basic steps using unbiased API.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"api-endpoints","dir":"","previous_headings":"","what":"API Endpoints","title":"Unbiased: Production-Grade Randomization API","text":"unbiased API designed facilitate clinical trial management set endpoints: Study Management: Create configure new studies, including specifying randomization parameters treatment arms. Participant Randomization: Dynamically randomize participants treatment groups based study’s configuration existing participant data. Study List: List previously defined studies. Study Details: Show details selected study. Randomization List: Generate list randomized patients selected study. Audit Log: Show audit log selected study.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"study-creation","dir":"","previous_headings":"API Endpoints","what":"Study Creation","title":"Unbiased: Production-Grade Randomization API","text":"initialize study using Pocock’s minimization method, use POST /minimisation_pocock endpoint. required JSON payload detail study, including treatment groups, allocation ratios, covariates. endpoint sets study returns ID accessing study-related endpoints.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"patient-randomization","dir":"","previous_headings":"API Endpoints","what":"Patient Randomization","title":"Unbiased: Production-Grade Randomization API","text":"POST /{study_id}/patient endpoint assigns new patient treatment group, requiring patient details covariate information JSON payload. endpoint determines patient’s treatment group.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"study-list","dir":"","previous_headings":"API Endpoints","what":"Study List","title":"Unbiased: Production-Grade Randomization API","text":"GET /study/ endpoint allow list previously defined studies. returns information : Study ID Identifier Name study Randomization method Last edit date","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"study-details","dir":"","previous_headings":"API Endpoints","what":"Study Details","title":"Unbiased: Production-Grade Randomization API","text":"GET /study/{study_id} endpoint allows retrieve details selected study. response body return: Name study Randomization method Last edit date Input parameters Strata","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"randomization-list","dir":"","previous_headings":"API Endpoints","what":"Randomization List","title":"Unbiased: Production-Grade Randomization API","text":"GET /study/{study_id}/randomization_list endpoint allows generate list randomized patients along assigned study arms.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"audit-log","dir":"","previous_headings":"API Endpoints","what":"Audit Log","title":"Unbiased: Production-Grade Randomization API","text":"GET /study/{study_id}/audit endpoint allows print records audit log selected study. response body includes following information: Log ID Creation date Type event Request ID Study ID Endpoint URL Request method Request body study definition Response code Response body study details endpoint facilitates tracking history requests sent database, along corresponding responses. enables us trace actions involving API.","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/index.html","id":"running-tests","dir":"","previous_headings":"","what":"Running Tests","title":"Unbiased: Production-Grade Randomization API","text":"Unbiased provides extensive collection tests ensure correct functionality.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"executing-tests-from-an-r-interactive-session","dir":"","previous_headings":"Running Tests","what":"Executing Tests from an R Interactive Session","title":"Unbiased: Production-Grade Randomization API","text":"execute tests using interactive R session, run following commands: Make sure devtools package installed environment. Ensure necessary database connection environment variables set running tests. can set environment variables using methods Sys.setenv. Running tests start Unbiased API random port.","code":"devtools::load_all() testthat::test_package(**unbiased**)"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"executing-tests-from-the-command-line","dir":"","previous_headings":"Running Tests","what":"Executing Tests from the Command Line","title":"Unbiased: Production-Grade Randomization API","text":"Use helper script run_tests.sh execute tests command line. Remember set database connection environment variables running tests.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"running-tests-with-docker-compose","dir":"","previous_headings":"Running Tests","what":"Running Tests with Docker Compose","title":"Unbiased: Production-Grade Randomization API","text":"Docker Compose can used build Unbiased Docker image execute tests. can done using provided docker-compose.test.yml file. method ensures consistent testing environment simplifies setup process.","code":"docker compose -f docker-compose.test.yml build docker compose -f docker-compose.test.yml run tests"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"code-coverage","dir":"","previous_headings":"Running Tests","what":"Code Coverage","title":"Unbiased: Production-Grade Randomization API","text":"Unbiased supports code coverage analysis covr package. allows measure effectiveness tests showing parts R code R directory actually tested. calculate code coverage, need install covr package. installed, can use following methods: covr::report(): method runs tests generates detailed coverage report HTML format. covr::package_coverage(): method provides simpler, text-based code coverage report. Alternatively, can use provided run_tests_with_coverage.sh script run Unbiased tests code coverage.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"configuring-sentry","dir":"","previous_headings":"Running Tests","what":"Configuring Sentry","title":"Unbiased: Production-Grade Randomization API","text":"Unbiased server offers robust error reporting capabilities integration Sentry service. activate Sentry, simply set SENTRY_DSN environment variable. Additionally, flexibility customize setup configuring following environment variables: SENTRY_ENVIRONMENT used set environment (e.g., “production”, “staging”, “development”). set, environment defaults “development”. SENTRY_RELEASE used set release Sentry. set, release defaults “unspecified”.","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/index.html","id":"purpose-and-scope-for-clinical-trial-randomization","dir":"","previous_headings":"","what":"Purpose and Scope for Clinical Trial Randomization","title":"Unbiased: Production-Grade Randomization API","text":"Randomization gold standard conducting clinical trials fundamental aspect clinical trials, studies comparing two arms. cases randomization desirable technique ensure patients randomly allocated defined groups. essential maintaining integrity trial ensuring results reliable, blinding research personnel. However, situations desirable studies balance patients terms numbers group , addition, achieve balance respect relevant factors, sex diabetes type. Adequate selection randomization methods allows intended randomization goals realized. Unbiased compared standard commonly used randomization methods, e.g. simple method block method, apart methods, additionally offers enhanced features flexible adaptive methods, based current information allocation patients trial. Compared , example, block randomization, adaptive randomization ensures relatively equal allocation patient groups, also allows groups balanced basis certain important covariates, key advantage. randomization requires predefined criteria, probability given patient assigned group based minimizing total imbalance, weights can assigned personally individual covariate. advanced algorithmic approach sets apart others minimizing selection bias improving overall efficiency randomization process clinical trials. Unbiased allows use simple, block adaptive minimization randomization methods relevant conduct clinical trials, package caters needs clinical trial randomization. … find differences randomization methods, read vignette Comparative Analysis Randomization Methods.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"comparison-with-other-solutions","dir":"","previous_headings":"","what":"Comparison with other solutions","title":"Unbiased: Production-Grade Randomization API","text":"many packages perform specific randomization methods R. designed stratified randomization permuted blocks, blockrand randomizeR. also utilize options using minimization randomization - e.g. randpack Minirand. unique contribution landscape integration comprehensive API commitment rigorous testing. dual focus ensures unbiased supports practical needs clinical trials, also aligns technical requirements modern clinical research environments. prioritizing aspects, unbiased addresses critical gap market: need eCRF-compatible randomization solution dependable easily integrated existing workflows. , together implementation minimization techniques, sets unbiased apart novel, comprehensive tool.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":null,"dir":"Reference","previous_headings":"","what":"AuditLog Class — AuditLog","title":"AuditLog Class — AuditLog","text":"class used internally store audit logs request.","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"public-methods","dir":"Reference","previous_headings":"","what":"Public methods","title":"AuditLog Class — AuditLog","text":"AuditLog$new() AuditLog$disable() AuditLog$is_enabled() AuditLog$set_request_body() AuditLog$set_response_body() AuditLog$set_ip_address() AuditLog$set_user_agent() AuditLog$set_event_type() AuditLog$set_study_id() AuditLog$set_response_code() AuditLog$validate_log() AuditLog$persist() AuditLog$clone()","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$new(request_method, endpoint_url)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-1","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$disable()"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-2","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$is_enabled()"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-3","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_request_body(request_body)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-4","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_response_body(response_body)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-5","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_ip_address(ip_address)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-6","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_user_agent(user_agent)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-7","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_event_type(event_type)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-8","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_study_id(study_id)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-9","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_response_code(response_code)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-10","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$validate_log()"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-11","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$persist()"},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"method-clone-","dir":"Reference","previous_headings":"","what":"Method clone()","title":"AuditLog Class — AuditLog","text":"objects class cloneable method.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-12","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$clone(deep = FALSE)"},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"AuditLog Class — AuditLog","text":"deep Whether make deep clone.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_event_type.html","id":null,"dir":"Reference","previous_headings":"","what":"Set Audit Log Event Type — audit_log_set_event_type","title":"Set Audit Log Event Type — audit_log_set_event_type","text":"function sets event type audit log. retrieves audit log request's internal data, calls audit log's set_event_type method provided event type.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_event_type.html","id":"ref-usage","dir":"Reference","previous_headings":"","what":"Usage","title":"Set Audit Log Event Type — audit_log_set_event_type","text":"","code":"audit_log_set_event_type(event_type, req)"},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_event_type.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"Set Audit Log Event Type — audit_log_set_event_type","text":"event_type event type set audit log. req request object, contain audit log internal data.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_event_type.html","id":"value","dir":"Reference","previous_headings":"","what":"Value","title":"Set Audit Log Event Type — audit_log_set_event_type","text":"Returns nothing modifies audit log -place.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_study_id.html","id":null,"dir":"Reference","previous_headings":"","what":"Set Audit Log Study ID — audit_log_set_study_id","title":"Set Audit Log Study ID — audit_log_set_study_id","text":"function sets study ID audit log. retrieves audit log request's internal data, calls audit log's set_study_id method provided study ID.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_study_id.html","id":"ref-usage","dir":"Reference","previous_headings":"","what":"Usage","title":"Set Audit Log Study ID — audit_log_set_study_id","text":"","code":"audit_log_set_study_id(study_id, req)"},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_study_id.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"Set Audit Log Study ID — audit_log_set_study_id","text":"study_id study ID set audit log. req request object, contain audit log internal data.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_study_id.html","id":"value","dir":"Reference","previous_headings":"","what":"Value","title":"Set Audit Log Study ID — audit_log_set_study_id","text":"Returns nothing modifies audit log -place.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":null,"dir":"Reference","previous_headings":"","what":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"randomize_dynamic function implements dynamic randomization algorithm using minimization method proposed Pocock (Pocock Simon, 1975). requires defining basic study parameters: number arms (K), number covariates (C), patient allocation ratios (\\(a_k\\)) (k = 1,2,…., K), weights covariates (\\(w_i\\)) (= 1,2,…., C), maximum probability (p) assigning patient group smallest total unbalance multiplied respective weights (\\(G_k\\)). total unbalance first patient regardless assigned arm, patient randomly allocated given arm. Subsequent patients randomized based calculation unbalance depending selected method: \"range\", \"var\" (variance), \"sd\" (standard deviation). case two arms, \"range\" method equivalent \"sd\" method.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"ref-usage","dir":"Reference","previous_headings":"","what":"Usage","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"","code":"randomize_minimisation_pocock( arms, current_state, weights, ratio, method = \"var\", p = 0.85 )"},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"arms character() Arm names. current_state tibble() table covariates current arm assignments column arm, last row contains new patient empty string arm weights numeric() vector positive weights, equal length number covariates, numbered covariates, defaults equal weights ratio integer() Vector positive integers (0 allowed), equal length number arms, named arms, defaults equal weight method character() Function used compute within-arm variability, must one : sd, var, range, defaults var p numeric() single value, proportion randomness (0, 1) randomization vs determinism, defaults 85% deterministic","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"value","dir":"Reference","previous_headings":"","what":"Value","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"character() name arm assigned patient","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"details","dir":"Reference","previous_headings":"","what":"Details","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"Initially, algorithm creates matrix results comparing newly randomized patient current balance patients based defined covariates. next step, arm specified covariate, various scenarios patient allocation calculated. existing results (n) updated new patient, , considering ratio coefficients, results divided specific allocation ratio. Depending method, total unbalance calculated, taking account weights, number covariates using one three methods (“sd”, “range”, “var”). Based number defined arms, minimum value (\\(G_k\\)) (defined weighted sum level-based imbalance) selects arm patient assigned predefined probability (p). probability patient assigned arm equal (1-p)/(K-1) remaining arms.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"note","dir":"Reference","previous_headings":"","what":"Note","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"function's implementation refactored adaptation codebase 'Minirand' package.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"references","dir":"Reference","previous_headings":"","what":"References","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"Pocock, S. J., & Simon, R. (1975). Minimization: new method assigning patients treatment control groups clinical trials. Minirand Package: Man Jin, Adam Polis, Jonathan Hartzel. (https://CRAN.R-project.org/package=Minirand)","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"ref-examples","dir":"Reference","previous_headings":"","what":"Examples","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"","code":"n_at_the_moment <- 10 arms <- c(\"control\", \"active low\", \"active high\") sex <- sample(c(\"F\", \"M\"), n_at_the_moment + 1, replace = TRUE, prob = c(0.4, 0.6) ) diabetes <- sample(c(\"diabetes\", \"no diabetes\"), n_at_the_moment + 1, replace = TRUE, prob = c(0.2, 0.8) ) arm <- sample(arms, n_at_the_moment, replace = TRUE, prob = c(0.4, 0.4, 0.2) ) |> c(\"\") covar_df <- tibble::tibble(sex, diabetes, arm) covar_df #> # A tibble: 11 × 3 #> sex diabetes arm #> #> 1 M no diabetes \"control\" #> 2 F no diabetes \"active low\" #> 3 F no diabetes \"active high\" #> 4 M no diabetes \"control\" #> 5 M no diabetes \"active low\" #> 6 M no diabetes \"control\" #> 7 M no diabetes \"control\" #> 8 M no diabetes \"control\" #> 9 F diabetes \"active low\" #> 10 F no diabetes \"active low\" #> 11 F no diabetes \"\" randomize_minimisation_pocock(arms = arms, current_state = covar_df) #> [1] \"active high\" randomize_minimisation_pocock( arms = arms, current_state = covar_df, ratio = c( \"control\" = 1, \"active low\" = 2, \"active high\" = 2 ), weights = c( \"sex\" = 0.5, \"diabetes\" = 1 ) ) #> [1] \"active high\""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_simple.html","id":null,"dir":"Reference","previous_headings":"","what":"Simple randomization — randomize_simple","title":"Simple randomization — randomize_simple","text":"Randomly assigns patient one arms according specified ratios, regardless already performed assignments.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_simple.html","id":"ref-usage","dir":"Reference","previous_headings":"","what":"Usage","title":"Simple randomization — randomize_simple","text":"","code":"randomize_simple(arms, ratio)"},{"path":"https://ttscience.github.io/unbiased/reference/randomize_simple.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"Simple randomization — randomize_simple","text":"arms character() Arm names. ratio integer() Vector positive integers (0 allowed), equal length number arms, named arms, defaults equal weight","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_simple.html","id":"value","dir":"Reference","previous_headings":"","what":"Value","title":"Simple randomization — randomize_simple","text":"Selected arm assignment.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_simple.html","id":"ref-examples","dir":"Reference","previous_headings":"","what":"Examples","title":"Simple randomization — randomize_simple","text":"","code":"randomize_simple(c(\"active\", \"placebo\"), c(\"active\" = 2, \"placebo\" = 1)) #> [1] \"active\""},{"path":"https://ttscience.github.io/unbiased/reference/run_unbiased.html","id":null,"dir":"Reference","previous_headings":"","what":"Run API — run_unbiased","title":"Run API — run_unbiased","text":"Starts unbiased API.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/run_unbiased.html","id":"ref-usage","dir":"Reference","previous_headings":"","what":"Usage","title":"Run API — run_unbiased","text":"","code":"run_unbiased()"},{"path":"https://ttscience.github.io/unbiased/reference/run_unbiased.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"Run API — run_unbiased","text":"host character(1) Host URL. port integer(1) Port serve API .","code":""},{"path":"https://ttscience.github.io/unbiased/reference/run_unbiased.html","id":"value","dir":"Reference","previous_headings":"","what":"Value","title":"Run API — run_unbiased","text":"Function called serve API caller thread.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/unbiased-package.html","id":null,"dir":"Reference","previous_headings":"","what":"unbiased: Unbiased: Production-Grade Randomization API — unbiased-package","title":"unbiased: Unbiased: Production-Grade Randomization API — unbiased-package","text":"Unbiased package delivers minimization-based randomization algorithm patient allocation clinical trials, fully integrated production-ready API. designed work seamlessly persistent PostgreSQL database, ensuring reliable data management integrity. Packaged precompiled Docker images, Unbiased simplifies deployment just running docker-compose , making exceptionally straightforward incorporate environment.","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/unbiased-package.html","id":"author","dir":"Reference","previous_headings":"","what":"Author","title":"unbiased: Unbiased: Production-Grade Randomization API — unbiased-package","text":"Maintainer: Kamil Sijko kamil.sijko@ttsi.com.pl (ORCID) Authors: Kinga Sałata kinga.salata@ttsi.com.pl Aleksandra Duda aleksandra.duda@ttsi.com.pl Łukasz Wałejko lukasz.walejko@ttsi.com.pl Jagoda Głowacka-Walas jagoda.glowacka-walas@ttsi.com.pl (ORCID) Laura Bąkała contributors: Michał Seweryn michal.seweryn@biol.uni.lodz.pl (ORCID) [contributor] Transition Technologies Science Sp. z o.o. [funder, copyright holder]","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/news/index.html","id":"new-features-1-0-0","dir":"Changelog","previous_headings":"","what":"New Features","title":"unbiased 1.0.0","text":"Implemented support adaptive randomization using Pocock’s minimization algorithm, integrating new R function based Minirand package Man Jin, Adam Polis, Jonathan Hartzel (Minirand Package). Introduced new POST endpoints facilitating creation studies randomization subjects. Added new GET endpoints comprehensive access study overviews, -depth details, information randomized patients. Implemented audit trial mechanism systematically logs stores request database alongside unbiased’s response. Introduced new GET endpoint enabling users access complete audit trail specific study.","code":""},{"path":"https://ttscience.github.io/unbiased/news/index.html","id":"vignettes--articles-1-0-0","dir":"Changelog","previous_headings":"","what":"Vignettes / Articles","title":"unbiased 1.0.0","text":"Added new article benchmarking Pocock’s minimization algorithm permuted block randomization simple randomization, focusing balance covariates.","code":""},{"path":"https://ttscience.github.io/unbiased/news/index.html","id":"devops-integration-1-0-0","dir":"Changelog","previous_headings":"","what":"DevOps Integration","title":"unbiased 1.0.0","text":"Integrated project Sentry capture errors, empowering users provide credentials receive notifications event unexpected occurrences, including HTTP 500 instances. Integrated project CodeCov, achieving code coverage 95% higher, maintenance level improvement required merging. Integrated project pkgdown, hosting project site ttscience.github.io/unbiased/. Enforced use linter, ensuring errors present upon merging.","code":""},{"path":"https://ttscience.github.io/unbiased/news/index.html","id":"smaller-improvements-1-0-0","dir":"Changelog","previous_headings":"","what":"Smaller Improvements","title":"unbiased 1.0.0","text":"Improved handling malformed JSONs, now returning HTTP 400 instead default HTTP 500 (plumber behavior).","code":""}] +[{"path":"https://ttscience.github.io/unbiased/LICENSE.html","id":null,"dir":"","previous_headings":"","what":"MIT License","title":"MIT License","text":"Copyright (c) 2024 Transition Technologies Science sp. z o.o. Permission hereby granted, free charge, person obtaining copy software associated documentation files (“Software”), deal Software without restriction, including without limitation rights use, copy, modify, merge, publish, distribute, sublicense, /sell copies Software, permit persons Software furnished , subject following conditions: copyright notice permission notice shall included copies substantial portions Software. SOFTWARE PROVIDED “”, WITHOUT WARRANTY KIND, EXPRESS IMPLIED, INCLUDING LIMITED WARRANTIES MERCHANTABILITY, FITNESS PARTICULAR PURPOSE NONINFRINGEMENT. EVENT SHALL AUTHORS COPYRIGHT HOLDERS LIABLE CLAIM, DAMAGES LIABILITY, WHETHER ACTION CONTRACT, TORT OTHERWISE, ARISING , CONNECTION SOFTWARE USE DEALINGS SOFTWARE.","code":""},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"introduction","dir":"Articles","previous_headings":"","what":"Introduction","title":"Benchmarking randomization methods","text":"Randomization clinical trials gold standard widely considered best design evaluating effectiveness new treatments compared alternative treatments (standard care) placebo. Indeed, selection appropriate randomisation important selection appropriate statistical analysis study analysis strategy, whether based randomisation population model (Berger et al. (2021)). One primary advantages randomization, particularly simple randomization (usually using flipping coin method), ability balance confounding variables across treatment groups. especially effective large sample sizes (n > 200), random allocation participants helps ensure known unknown confounders evenly distributed study arms. balanced distribution contributes significantly internal validity study, minimizes risk selection bias confounding influencing results (Lim (2019)). ’s important note, however, simple randomization powerful large trials, may always guarantee even distribution confounding factors trials smaller sample sizes (n < 100). cases, random allocation might result imbalances baseline characteristics groups, can affect interpretation treatment’s effectiveness. potential limitation sets stage considering additional methods, stratified randomization, dynamic minimization algorithms address challenges smaller trials (Kang, Ragan, Park (2008)). document provides summary comparison three randomization methods: simple randomization, block randomization, adaptive randomization. Simple randomization adaptive randomization (minimization method) tools available unbiased package randomize_simple randomize_minimisation_pocock functions (Sijko et al. (2024)). comparison aims demonstrate superiority adaptive randomization (minimization method) methods assessing least imbalance accompanying variables therapeutic groups. Monte Carlo simulations used generate data, utilizing simstudy package (Goldfeld Wujciak-Jens (2020)). Parameters binary distribution variables based data publication Mrozikiewicz-Rakowska et al. (2023) information researchers. document structure follows: first, based defined parameters, data simulated using Monte Carlo method single simulation; , generated patient data, appropriate groups assigned using three randomization methods; data summarized form descriptive statistics along relevant statistical test; next, data prepared .Rds format generated 1000 simulations loaded., results based standardised mean difference (SMD) test discussed visual form (boxplot, violin plot) percentage success achieved method given precision (tabular summary)","code":"# load packages library(unbiased) library(dplyr) library(simstudy) library(tableone) library(ggplot2) library(gt) library(gtsummary) library(truncnorm) library(tidyr) library(randomizeR)"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"the-randomization-methods-considered-for-comparison","dir":"Articles","previous_headings":"","what":"The randomization methods considered for comparison","title":"Benchmarking randomization methods","text":"process comparing balance covariates among randomization methods, three randomization methods selected evaluation: simple randomization - simple coin toss, algorithm gives participants equal chances assigned particular arm. method’s advantage lies simplicity elimination predictability. However, due complete randomness, may lead imbalance sample sizes arms imbalances prognostic factors. large sample size (n > 200), simple randomisation gives similar number generated participants group. small sample size (n < 100), results imbalance (Kang, Ragan, Park (2008)). block randomization - randomization method takes account defined covariates patients. method involves assigning patients therapeutic arms blocks fixed size, recommendation blocks different sizes. , extent, reduces risk researchers predicting future arm assignments. contrast simple randomization, block method aims balance number patients within block, hence reducing overall imbalance arms (Rosenberger Lachin (2015)). adaptive randomization using minimization method based Pocock Simon (1975) algorithm - - randomization approach aims balance prognostic factors across treatment arms within clinical study. functions evaluating total imbalance factors time new patient considered study. minimization method computes overall imbalance potential arm assignment new patient, considering factors like variance specified criteria. patient assigned arm addition results smallest total imbalance. assignment deterministic made predetermined probability, ensuring level randomness arm allocation. method particularly useful trials multiple prognostic factors smaller studies traditional randomization might fail achieve balance.","code":""},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"assessment-of-covariate-balance","dir":"Articles","previous_headings":"","what":"Assessment of covariate balance","title":"Benchmarking randomization methods","text":"proposed approach assessment randomization methods, primary objective evaluate method terms achieving balance specified covariates. assessment balance aims determine whether distributions covariates similarly balanced therapeutic group. Based literature, standardized mean differences (SMD) employed assessing balance (Berger et al. (2021)). SMD method one commonly used statistics assessing balance covariates, regardless unit measurement. statistical measure comparing differences two groups. covariates examined case expressed binary variables. case categorical variables, SMD calculated using following formula (Zhang et al. (2019)): \\[ SMD = \\frac{{p_1 - p_2}}{{\\sqrt{\\frac{{p_1 \\cdot (1 - p_1) + p_2 \\cdot (1 - p_2)}}{2}}}} \\], : \\(p_1\\) proportion first arm, \\(p_2\\) proportion second arm.","code":""},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"definied-number-of-patients","dir":"Articles","previous_headings":"","what":"Definied number of patients","title":"Benchmarking randomization methods","text":"simulation, using real use case - planned FootCell study - non-commercial clinical research area civilisation diseases - guide data generation process. FootCell study, anticipated total 105 patients randomized trial. patients equally divided among three research groups - Group , Group B, Group C - group comprising 35 patients.","code":"# defined number of patients n <- 105"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"defining-parameters-for-monte-carlo-simulation","dir":"Articles","previous_headings":"","what":"Defining parameters for Monte-Carlo simulation","title":"Benchmarking randomization methods","text":"distribution parameters individual covariates, subsequently used validate randomization methods, defined using publication Mrozikiewicz-Rakowska et al. (2023) allogenic interventions.. publication describes effectiveness comparing therapy using ADSC (Adipose-Derived Stem Cells) gel versus standard therapy fibrin gel patients diabetic foot ulcer treatment. FootCell study also aims assess safety advanced therapy involving live ASCs (Adipose-Derived Stem Cells) treatment diabetic foot syndrome, considering two groups treated ADSCs (one two administrations) compared fibrin gel. Therefore, appropriate population data extracted publication determine distributions can maintained designing FootCell study. process defining study randomization, following covariates selected: gender [male/female], diabetes type [type /type II], HbA1c [9/9 11] [%], tpo2 [50/50] [mmHg], age [55/55] [years], wound size [2/2] [cm\\(^2\\)]. case variables gender diabetes type publication Mrozikiewicz-Rakowska et al. (2023), expressed form frequencies. remaining variables presented terms measures central tendency along indication variability, well minimum maximum values. determine parameters binary distribution, truncated normal distribution available truncnorm package utilized. truncated normal distribution often used statistics probability modeling dealing data constrained certain range. particularly useful want model random variable take values beyond certain limits (Burkardt (2014)). generate necessary information remaining covariates, function simulate_proportions_trunc written, utilizing rtruncnorm function (Mersmann et al. (2023)). parameters mean, sd, lower, upper taken publication based expertise regarding ranges parameters. results presented table, assuming outcome refers first category parameter.","code":"# simulate parameters using truncated normal distribution simulate_proportions_trunc <- function(n, lower, upper, mean, sd, threshold) { simulate_data <- rtruncnorm( n = n, a = lower, b = upper, mean = mean, sd = sd ) <= threshold sum(simulate_data == TRUE) / n } set.seed(123) data.frame( hba1c = simulate_proportions_trunc(1000, 0, 11, 7.41, 1.33, 9), tpo2 = simulate_proportions_trunc(1000, 30, 100, 53.4, 18.4, 50), age = simulate_proportions_trunc(1000, 0, 100, 59.2, 9.7, 55), wound_size = simulate_proportions_trunc(1000, 0, 20, 2.7, 2.28, 2) ) |> rename(\"wound size\" = wound_size) |> pivot_longer( cols = everything(), names_to = \"parametr\", values_to = \"proportions\" ) |> mutate(\"first catogory of strata\" = c(\"<=9\", \"<=50\", \"<=55\", \"<=2\")) |> gt()"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"generate-data-using-monte-carlo-simulations","dir":"Articles","previous_headings":"","what":"Generate data using Monte-Carlo simulations","title":"Benchmarking randomization methods","text":"Monte-Carlo simulations used accumulate data. method designed model variables based defined parameters. Variables defined using simstudy package, utilizing defData function (Goldfeld Wujciak-Jens (2020)). variables specify proportions, dist = 'binary' used define variables. Due likely association type diabetes age – meaning older patient, higher probability type II diabetes – relationship diabetes established defining age variable using logit function link = \"logit\". proportions gender diabetes defined researchers consistent literature Mrozikiewicz-Rakowska et al. (2023). Using genData function simstudy package, data frame (data) generated artificially adopted variable arm, filled subsequent randomization methods arm allocation process n patients.","code":"# defining variables # male - 0.9 def <- simstudy::defData(varname = \"sex\", formula = \"0.9\", dist = \"binary\") # type I - 0.15 def <- simstudy::defData(def, varname = \"diabetes_type\", formula = \"0.15\", dist = \"binary\") # <= 9 - 0.888 def <- simstudy::defData(def, varname = \"hba1c\", formula = \"0.888\", dist = \"binary\") # <= 50 - 0.354 def <- simstudy::defData(def, varname = \"tpo2\", formula = \"0.354\", dist = \"binary\") # correlation with diabetes type def <- simstudy::defData( def, varname = \"age\", formula = \"(diabetes_type == 0) * (-0.95)\", link = \"logit\", dist = \"binary\" ) # <= 2 - 0.302 def <- simstudy::defData(def, varname = \"wound_size\", formula = \"0.302\", dist = \"binary\") # generate data using genData() data <- genData(n, def) |> mutate( sex = as.character(sex), age = as.character(age), diabetes_type = as.character(diabetes_type), hba1c = as.character(hba1c), tpo2 = as.character(tpo2), wound_size = as.character(wound_size) ) |> as_tibble() # add arm to tibble data <- data |> tibble::add_column(arm = \"\") # first 5 rows of the data head(data, 5) |> gt()"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"minimization-randomization","dir":"Articles","previous_headings":"","what":"Minimization randomization","title":"Benchmarking randomization methods","text":"generate appropriate research arms, function called minimize_results written, utilizing randomize_minimisation_pocock function available within unbiased package (Sijko et al. (2024)). probability parameter set level defined within function (p = 0.85). case minimization randomization, verify type minimization (equal weights unequal weights) used, three calls minimize_results function prepared: minimize_equal_weights - covariate weight takes value equal 1 divided number covariates. case, weight 1/6, minimize_unequal_weights - following expert assessment physicians, parameters potentially significant impact treatment outcomes (hba1c, tpo2, wound size) assigned weight 2. remaining covariates assigned weight 1. minimize_unequal_weights_3 - following expert assessment physicians, parameters potentially significant impact treatment outcomes (hba1c, tpo2, wound size) assigned weight 3. remaining covariates assigned weight 1. tables present information allocations first 5 patients. statistic_table function developed provide information : distribution number patients across research arms, distribution covariates across research arms, along p-value information statistical analyses used compare proportions - chi^2, exact Fisher’s test, typically used small samples. function relies use tbl_summary function available gtsummary package (Sjoberg et al. (2021)). table presents statistical summary results first iteration : Minimization weights equal 1/6. Minimization weights 2:1. Minimization weights 3:1.","code":"# drawing an arm for each patient minimize_results <- function(current_data, arms, weights) { for (n in seq_len(nrow(current_data))) { current_state <- current_data[1:n, 2:ncol(current_data)] current_data$arm[n] <- randomize_minimisation_pocock( arms = arms, current_state = current_state, weights = weights ) } return(current_data) } set.seed(123) # eqal weights - 1/6 minimize_equal_weights <- minimize_results( current_data = data, arms = c(\"armA\", \"armB\", \"armC\") ) head(minimize_equal_weights, 5) |> gt() set.seed(123) # double weights where the covariant is of high clinical significance minimize_unequal_weights <- minimize_results( current_data = data, arms = c(\"armA\", \"armB\", \"armC\"), weights = c( \"sex\" = 1, \"diabetes_type\" = 1, \"hba1c\" = 2, \"tpo2\" = 2, \"age\" = 1, \"wound_size\" = 2 ) ) head(minimize_unequal_weights, 5) |> gt() set.seed(123) # triple weights where the covariant is of high clinical significance minimize_unequal_weights_3 <- minimize_results( current_data = data, arms = c(\"armA\", \"armB\", \"armC\"), weights = c( \"sex\" = 1, \"diabetes_type\" = 1, \"hba1c\" = 3, \"tpo2\" = 3, \"age\" = 1, \"wound_size\" = 3 ) ) head(minimize_unequal_weights_3, 5) |> gt() # generation of frequency and chi^2 statistic values or fisher exact test statistics_table <- function(data) { data |> mutate( sex = ifelse(sex == \"1\", \"men\", \"women\"), diabetes_type = ifelse(diabetes_type == \"1\", \"type1\", \"type2\"), hba1c = ifelse(hba1c == \"1\", \"<=9\", \"(9,11>\"), tpo2 = ifelse(tpo2 == \"1\", \"<=50\", \">50\"), age = ifelse(age == \"1\", \"<=55\", \">55\"), wound_size = ifelse(wound_size == \"1\", \"<=2\", \">2\") ) |> tbl_summary( include = c(sex, diabetes_type, hba1c, tpo2, age, wound_size), by = arm ) |> modify_header(label = \"\") |> modify_header(all_stat_cols() ~ \"**{level}**, N = {n}\") |> bold_labels() |> add_p() } statistics_table(minimize_equal_weights) statistics_table(minimize_unequal_weights) statistics_table(minimize_unequal_weights_3)"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"simple-randomization","dir":"Articles","previous_headings":"","what":"Simple randomization","title":"Benchmarking randomization methods","text":"next step, appropriate arms generated patients using simple randomization, available unbiased package - randomize_simple function (Sijko et al. (2024)). simple_results function called within simple_data, considering initial assumption assigning patients three arms 1:1:1 ratio. Since simple randomization, take account initial covariates, treatment assignment occurs randomly (flip coin method). tables illustrate example data output summary statistics including summary statistical tests.","code":"# simple randomization simple_results <- function(current_data, arms, ratio) { for (n in seq_len(nrow(current_data))) { current_data$arm[n] <- randomize_simple(arms, ratio) } return(current_data) } set.seed(123) simple_data <- simple_results( current_data = data, arms = c(\"armA\", \"armB\", \"armC\"), ratio = c(\"armB\" = 1L, \"armA\" = 1L, \"armC\" = 1L) ) head(simple_data, 5) |> gt() statistics_table(simple_data)"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"block-randomization","dir":"Articles","previous_headings":"","what":"Block randomization","title":"Benchmarking randomization methods","text":"Block randomization, opposed minimization simple randomization methods, developed based rbprPar function available randomizeR package (Uschner et al. (2018)). Using , block_rand function created, , based defined number patients, arms, list stratifying factors, generates randomization list length equal number patients multiplied product categories covariate. case specified data document, one iteration, amounts 105 * 2^6 = 6720 rows. ensures appropriate number randomisation codes opportunity. case equal characteristics, certain right number codes defined n patients. Based block_rand function, possible generate randomisation list, based patients allocated, characteristics output data frame. Due 3 arms need blind allocation consecutive patients, block sizes 3,6 9 used calculations. next step, patients assigned research groups using block_results function (based list generated function block_rand). first available code randomization list meets specific conditions selected, removed list available codes. Based , research arms generated ensure appropriate number patients group (based assumed ratio 1:1:1). tables show assignment patients groups using block randomisation summary statistics including summary statistical tests.","code":"# Function to generate a randomisation list block_rand <- function(n, block, n_groups, strata, arms = LETTERS[1:n_groups]) { strata_grid <- expand.grid(strata) strata_n <- nrow(strata_grid) ratio <- rep(1, n_groups) gen_seq_list <- lapply(seq_len(strata_n), function(i) { rand <- rpbrPar( N = n, rb = block, K = n_groups, ratio = ratio, groups = arms, filledBlock = FALSE ) getRandList(genSeq(rand))[1, ] }) df_list <- tibble::tibble() for (i in seq_len(strata_n)) { local_df <- strata_grid |> dplyr::slice(i) |> dplyr::mutate(count_n = n) |> tidyr::uncount(count_n) |> tibble::add_column(rand_arm = gen_seq_list[[i]]) df_list <- rbind(local_df, df_list) } return(df_list) } # Generate a research arm for patients in each iteration block_results <- function(current_data) { simulation_result <- block_rand( n = n, block = c(3, 6, 9), n_groups = 3, strata = list( sex = c(\"0\", \"1\"), diabetes_type = c(\"0\", \"1\"), hba1c = c(\"0\", \"1\"), tpo2 = c(\"0\", \"1\"), age = c(\"0\", \"1\"), wound_size = c(\"0\", \"1\") ), arms = c(\"armA\", \"armB\", \"armC\") ) for (n in seq_len(nrow(current_data))) { # \"-1\" is for \"arm\" column current_state <- current_data[n, 2:(ncol(current_data) - 1)] matching_rows <- which(apply( simulation_result[, -ncol(simulation_result)], 1, function(row) all(row == current_state) )) if (length(matching_rows) > 0) { current_data$arm[n] <- simulation_result[matching_rows[1], \"rand_arm\"] # Delete row from randomization list simulation_result <- simulation_result[-matching_rows[1], , drop = FALSE] } } return(current_data) } set.seed(123) block_data <- block_results(data) head(block_data, 5) |> gt() statistics_table(block_data)"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"generate-1000-simulations","dir":"Articles","previous_headings":"","what":"Generate 1000 simulations","title":"Benchmarking randomization methods","text":"performed 1000 iterations data generation parameters defined . number iterations indicates number iterations included Monte-Carlo simulations accumulate data given parameters. allowed generation data 1000 times 105 patients efficiently assess effect randomization methods context covariate balance. data assigned variable sim_data based data stored .Rds file 1000_sim_data.Rds, available within vignette information GitHub repository unbiased package.","code":"# define number of iterations # no_of_iterations <- 1000 # nolint # define number of cores # no_of_cores <- 20 # nolint # perform simulations (run carefully!) # source(\"~/unbiased/vignettes/helpers/run_parallel.R\") # nolint # read data from file sim_data <- readRDS(\"1000_sim_data.Rds\")"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"check-balance-using-smd-test","dir":"Articles","previous_headings":"","what":"Check balance using smd test","title":"Benchmarking randomization methods","text":"order select test define precision specified level, assume imbalance, literature analysis conducted based publications Lee et al. (2021), Austin (2009), Doah et al. (2021), Brown et al. (2020), Nguyen et al. (2017), Sánchez-Meca, Marı́n-Martı́nez, Chacón-Moscoso (2003), Lee, Acharya, et al. (2022), Berger et al. (2021). assess balance covariates research groups , B, C, Standardized Mean Difference (SMD) test employed, compares two groups. Since three groups example, SMD test computed pair comparisons: vs B, vs C, B vs C. average SMD test given covariate calculated based comparisons. literature analysis, precision level ranged 0.1-0.2. small samples, expected SMD test exceed 0.2 (Austin (2009)). Additionally, according publication Sánchez-Meca, Marı́n-Martı́nez, Chacón-Moscoso (2003), golden standard dictates specific threshold SMD test considered balanced. Generally, smaller SMD test, smaller difference covariate imbalance. analyzed example, due sample size 105 patients, threshold 0.2 SMD test adopted. function called smd_covariants_data written generate frames produce SMD test covariate iteration, utilizing CreateTableOne function available tableone package (Yoshida Bartel (2022)). cases test result <0.001, value 0 assigned. results randomization method stored cov_balance_data. results SMD test presented form boxplot violin plot, depicting outcomes randomization method. red dashed line indicates adopted precision threshold. Boxplot combined results Summary average smd randomization methods Violin plot Summary smd randomization methods covariants Summary table success Based specified precision threshold 0.2, function defining randomization success, named success_power, developed. SMD test value covariate given iteration 0.2, function defines analysis data ‘failure’ - 0; otherwise, defined ‘success’ - 1. final success power calculated sum successes iteration divided total number specified iterations. results summarized table percentage success randomization method.","code":"# definied covariants vars <- c(\"sex\", \"age\", \"diabetes_type\", \"wound_size\", \"tpo2\", \"hba1c\") smd_covariants_data <- function(data, vars, strata) { result_table <- lapply(unique(data$simnr), function(i) { current_data <- data[data$simnr == i, ] arms_to_check <- setdiff(names(current_data), c(vars, \"id\", \"simnr\")) # check SMD for any covariants lapply(arms_to_check, function(arm) { tab <- CreateTableOne( vars = vars, data = current_data, strata = arm ) results_smd <- ExtractSmd(tab) |> as.data.frame() |> tibble::rownames_to_column(\"covariants\") |> select(covariants, results = average) |> mutate(results = round(as.numeric(results), 3)) results <- bind_cols( simnr = i, strata = arm, results_smd ) return(results) }) |> bind_rows() }) |> bind_rows() return(result_table) } cov_balance_data <- smd_covariants_data( data = sim_data, vars = vars ) |> mutate(method = case_when( strata == \"minimize_equal_weights_arms\" ~ \"minimize equal\", strata == \"minimize_unequal_weights_arms\" ~ \"minimize unequal 2:1\", strata == \"minimize_unequal_weights_triple_arms\" ~ \"minimize unequal 3:1\", strata == \"simple_data_arms\" ~ \"simple randomization\", strata == \"block_data_arms\" ~ \"block randomization\" )) |> select(-strata) # boxplot cov_balance_data |> select(simnr, results, method) |> group_by(simnr, method) |> mutate(results = mean(results)) |> distinct() |> ggplot(aes(x = method, y = results, fill = method)) + geom_boxplot() + geom_hline(yintercept = 0.2, linetype = \"dashed\", color = \"red\") + theme_bw() # violin plot cov_balance_data |> ggplot(aes(x = method, y = results, fill = method)) + geom_violin() + geom_hline( yintercept = 0.2, linetype = \"dashed\", color = \"red\" ) + facet_wrap(~covariants, ncol = 3) + theme_bw() + theme(axis.text = element_text(angle = 45, vjust = 0.5, hjust = 1)) # function defining success of randomisation success_power <- function(cov_data) { result_table <- lapply(unique(cov_data$simnr), function(i) { current_data <- cov_data[cov_data$simnr == i, ] current_data |> group_by(method) |> summarise(success = ifelse(any(results > 0.2), 0, 1)) |> tibble::add_column(simnr = i, .before = 1) }) |> bind_rows() success <- result_table |> group_by(method) |> summarise(results_power = sum(success) / n() * 100) return(success) } success_power(cov_balance_data) |> as.data.frame() |> rename(`power results [%]` = results_power) |> gt()"},{"path":"https://ttscience.github.io/unbiased/articles/minimization_randomization_comparison.html","id":"conclusion","dir":"Articles","previous_headings":"","what":"Conclusion","title":"Benchmarking randomization methods","text":"Considering three randomization methods: minimization, block randomization, simple randomization, minimization performs best terms covariate balance. Simple randomization significant drawback, patient allocation arms occurs randomly equal probability. leads imbalance number patients covariate balance, also random. particularly case small samples. Balancing number patients possible larger samples n > 200. hand, block randomization performs well balancing number patients groups specified allocation ratio. However, compared adaptive randomisation using minimisation method, block randomisation lower probability terms balancing co-variables. Minimization method, provides highest success power ensuring balance across covariates groups. made possible appropriate algorithm implemented part minimisation randomisation. assigning next patient group, method examines total imbalance assigns patient appropriate study group specified probability balance sample terms size, covariates.","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/authors.html","id":null,"dir":"","previous_headings":"","what":"Authors","title":"Authors and Citation","text":"Kamil Sijko. Author, maintainer. Kinga Sałata. Author. Aleksandra Duda. Author. Łukasz Wałejko. Author. Jagoda Głowacka-Walas. Author. Laura Bąkała. Author. Michał Seweryn. Contributor. Transition Technologies Science Sp. z o.o.. Funder, copyright holder.","code":""},{"path":"https://ttscience.github.io/unbiased/authors.html","id":"citation","dir":"","previous_headings":"","what":"Citation","title":"Authors and Citation","text":"Sijko K, Sałata K, Duda , Wałejko Ł, Głowacka-Walas J, Bąkała L (2024). unbiased: Unbiased: Production-Grade Randomization API. R package version 1.0.0, https://ttscience.github.io/unbiased/.","code":"@Manual{, title = {unbiased: Unbiased: Production-Grade Randomization API}, author = {Kamil Sijko and Kinga Sałata and Aleksandra Duda and Łukasz Wałejko and Jagoda Głowacka-Walas and Laura Bąkała}, year = {2024}, note = {R package version 1.0.0}, url = {https://ttscience.github.io/unbiased/}, }"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"unbiased-","dir":"","previous_headings":"","what":"Unbiased: Production-Grade Randomization API","title":"Unbiased: Production-Grade Randomization API","text":"> API-based solution Clinical Trial Randomization clinical trials, fair efficient allocation participants essential achieving reliable results. many excellent R randomization packages available, none, knowledge, provide dedicated API purpose. unbiased package fills gap featuring production-ready REST API designed seamless integration. unique combination allows easy connection electronic Case Report Forms (eCRF), enhancing data management streamlining participant allocation.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"why-choose-unbiased","dir":"","previous_headings":"","what":"Why choose unbiased?","title":"Unbiased: Production-Grade Randomization API","text":"goal creating unbiased provide user-friendly yet powerful tool addresses nuanced demands clinical trial randomization. offers: Production-Ready REST API: Built seamless integration eCRF/EDC systems, facilitating real-time randomization automation. Extensive Database Integration: Supports robust data management practices, ensuring participant information randomization outcomes securely managed easily accessible. Commitment Quality: Emphasizes development best practices, including comprehensive code coverage, deliver reliable trustworthy solution. Adaptability: Whether small-scale studies large, multi-center trials, unbiased scales meet needs. Comprehensive Documentation: support applying package effectively. choosing unbiased, ’re adopting sophisticated approach trial randomization, ensuring fair efficient participant allocation across studies support broader objectives clinical research technology.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"table-of-contents","dir":"","previous_headings":"","what":"Table of Contents","title":"Unbiased: Production-Grade Randomization API","text":"Quick Setup Docker Compose Quick Setup Docker API Server Configuration Alternative Installation Method Study Creation Patient Randomization Study List Study Details Randomization List Audit Log Quality Assurance Measures Executing Tests R Interactive Session Executing Tests Command Line Running Tests Docker Compose Code Coverage Configuring Sentry Purpose Scope Clinical Trial Randomization Comparison Solutions","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"quickstart-guide","dir":"","previous_headings":"","what":"Quickstart Guide","title":"Unbiased: Production-Grade Randomization API","text":"Initiating work unbiased involves simple setup steps. Whether ’re integrating R environment deploying API, aim equip reliable tool enhances integrity efficiency clinical trials.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"quick-setup-with-docker-compose","dir":"","previous_headings":"","what":"Quick Setup with Docker Compose","title":"Unbiased: Production-Grade Randomization API","text":"prepared simple Docker Compose setup allows run unbiased minimal configuration. method ideal users want quickly test unbiased locally development environment. get started, follow steps: Clone unbiased repository local machine just copy docker-compose.yml file repository. Navigate directory containing docker-compose.yml file. Run following command start unbiased: command start unbiased API server, making accessible specified port. PostgreSQL also started, ensuring API can connect database. can use docker-compose.yml file starting point deployment, modifying suit specific requirements.","code":"docker compose pull docker compose up"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"configuration","dir":"","previous_headings":"Quick Setup with Docker Compose","what":"Configuration","title":"Unbiased: Production-Grade Randomization API","text":"docker-compose.yml file contains necessary configuration running unbiased Docker Compose. can override default environment variables creating .env file directory docker-compose.yml file. following environment variables can set: UNBIASED_PORT: port API listen. Defaults 3838 provided. POSTGRES_USER: username authentication PostgreSQL database. Defaults postgres provided. POSTGRES_PASSWORD: password authentication PostgreSQL database. Warning: can set first run, work database already exists. Defaults postgres provided. UNBIASED_VERSION: version unbiased Docker image use. Defaults latest provided. POSTGRES_VERSION: version PostgreSQL Docker image use. Defaults latest provided.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"building-the-docker-image-locally","dir":"","previous_headings":"Quick Setup with Docker Compose","what":"Building the Docker Image Locally","title":"Unbiased: Production-Grade Randomization API","text":"cloned repository, can also build Docker image locally using following command:","code":"docker compose build"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"quick-setup-with-docker","dir":"","previous_headings":"","what":"Quick Setup with Docker","title":"Unbiased: Production-Grade Randomization API","text":"straightforward way deploy unbiased Docker images. ensures can get unbiased running minimal setup, regardless local environment. use unbiased, pull latest Docker image: run unbiased Docker, ensuring set necessary environment variables: command starts unbiased API, making accessible specified port. ’s crucial PostgreSQL database ready, unbiased automatically configure necessary database structures upon startup. PostgreSQL can run separate container local machine. Make sure use PostgreSQL instance temporal_tables extension available. can use ghcr.io/ttscience/postgres-temporal-tables/postgres-temporal-tables:latest image run PostgreSQL temporal_tables extension.","code":"docker pull ghcr.io/ttscience/unbiased docker run -e POSTGRES_DB=mydb -e POSTGRES_USER=myuser -e POSTGRES_PASSWORD=mypassword -e UNBIASED_PORT=3838 ghcr.io/ttscience/unbiased"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"api-server-configuration","dir":"","previous_headings":"","what":"API Server configuration","title":"Unbiased: Production-Grade Randomization API","text":"unbiased API server can configured using environment variables. following environment variables need set server start: POSTGRES_DB: name PostgreSQL database connect . POSTGRES_HOST: host PostgreSQL database. hostname, localhost database.example.com, IP address. POSTGRES_PORT: port PostgreSQL database listening. Defaults 5432 provided. POSTGRES_USER: username authentication PostgreSQL database. POSTGRES_PASSWORD: password authentication PostgreSQL database. UNBIASED_HOST: host API run. Defaults 0.0.0.0 provided. UNBIASED_PORT: port API listen. Defaults 3838 provided.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"alternative-installation-method","dir":"","previous_headings":"","what":"Alternative Installation Method","title":"Unbiased: Production-Grade Randomization API","text":"preferring work directly within R environment, unbiased package offers alternative installation method via GitHub. approach allows users easily integrate unbiased R projects. proceed method, utilize devtools package installation executing following command: Following package installation, unbiased API can launched within R. Simply invoke run_unbiased() function start API: initiates API server, default, local machine (http://localhost:3838), making accessible interaction various HTTP clients, including curl, Postman, R’s httr package.","code":"devtools::install_github(\"ttscience/unbiased\") unbiased::run_unbiased()"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"getting-started-with-unbiased","dir":"","previous_headings":"","what":"Getting started with unbiased","title":"Unbiased: Production-Grade Randomization API","text":"unbiased package offers functions randomizing participants clinical trials, ensuring fair transparent process. Complete documentation implemented methodology examples use available GitHub Pages, providing information need integrate unbiased trial management workflow. , present basic steps using unbiased API.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"api-endpoints","dir":"","previous_headings":"","what":"API Endpoints","title":"Unbiased: Production-Grade Randomization API","text":"unbiased API designed facilitate clinical trial management set endpoints: Study Management: Create configure new studies, including specifying randomization parameters treatment arms. Participant Randomization: Dynamically randomize participants treatment groups based study’s configuration existing participant data. Study List: List previously defined studies. Study Details: Show details selected study. Randomization List: Generate list randomized patients selected study. Audit Log: Show audit log selected study.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"study-creation","dir":"","previous_headings":"API Endpoints","what":"Study Creation","title":"Unbiased: Production-Grade Randomization API","text":"initialize study using Pocock’s minimization method, use POST /minimisation_pocock endpoint. required JSON payload detail study, including treatment groups, allocation ratios, covariates. endpoint sets study returns ID accessing study-related endpoints.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"patient-randomization","dir":"","previous_headings":"API Endpoints","what":"Patient Randomization","title":"Unbiased: Production-Grade Randomization API","text":"POST /{study_id}/patient endpoint assigns new patient treatment group, requiring patient details covariate information JSON payload. endpoint determines patient’s treatment group.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"study-list","dir":"","previous_headings":"API Endpoints","what":"Study List","title":"Unbiased: Production-Grade Randomization API","text":"GET /study/ endpoint allow list previously defined studies. returns information : Study ID Identifier Name study Randomization method Last edit date","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"study-details","dir":"","previous_headings":"API Endpoints","what":"Study Details","title":"Unbiased: Production-Grade Randomization API","text":"GET /study/{study_id} endpoint allows retrieve details selected study. response body return: Name study Randomization method Last edit date Input parameters Strata","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"randomization-list","dir":"","previous_headings":"API Endpoints","what":"Randomization List","title":"Unbiased: Production-Grade Randomization API","text":"GET /study/{study_id}/randomization_list endpoint allows generate list randomized patients along assigned study arms.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"audit-log","dir":"","previous_headings":"API Endpoints","what":"Audit Log","title":"Unbiased: Production-Grade Randomization API","text":"GET /study/{study_id}/audit endpoint allows print records audit log selected study. response body includes following information: Log ID Creation date Type event Request ID Study ID Endpoint URL Request method Request body study definition Response code Response body study details endpoint facilitates tracking history requests sent database, along corresponding responses. enables us trace actions involving API.","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/index.html","id":"running-tests","dir":"","previous_headings":"","what":"Running Tests","title":"Unbiased: Production-Grade Randomization API","text":"Unbiased provides extensive collection tests ensure correct functionality.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"executing-tests-from-an-r-interactive-session","dir":"","previous_headings":"Running Tests","what":"Executing Tests from an R Interactive Session","title":"Unbiased: Production-Grade Randomization API","text":"execute tests using interactive R session, run following commands: Make sure devtools package installed environment. Ensure necessary database connection environment variables set running tests. can set environment variables using methods Sys.setenv. Running tests start Unbiased API random port.","code":"devtools::load_all() testthat::test_package(**unbiased**)"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"executing-tests-from-the-command-line","dir":"","previous_headings":"Running Tests","what":"Executing Tests from the Command Line","title":"Unbiased: Production-Grade Randomization API","text":"Use helper script run_tests.sh execute tests command line. Remember set database connection environment variables running tests.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"running-tests-with-docker-compose","dir":"","previous_headings":"Running Tests","what":"Running Tests with Docker Compose","title":"Unbiased: Production-Grade Randomization API","text":"Docker Compose can used build Unbiased Docker image execute tests. can done using provided docker-compose.test.yml file. method ensures consistent testing environment simplifies setup process.","code":"docker compose -f docker-compose.test.yml build docker compose -f docker-compose.test.yml run tests"},{"path":"https://ttscience.github.io/unbiased/index.html","id":"code-coverage","dir":"","previous_headings":"Running Tests","what":"Code Coverage","title":"Unbiased: Production-Grade Randomization API","text":"Unbiased supports code coverage analysis covr package. allows measure effectiveness tests showing parts R code R directory actually tested. calculate code coverage, need install covr package. installed, can use following methods: covr::report(): method runs tests generates detailed coverage report HTML format. covr::package_coverage(): method provides simpler, text-based code coverage report. Alternatively, can use provided run_tests_with_coverage.sh script run Unbiased tests code coverage.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"configuring-sentry","dir":"","previous_headings":"Running Tests","what":"Configuring Sentry","title":"Unbiased: Production-Grade Randomization API","text":"Unbiased server offers robust error reporting capabilities integration Sentry service. activate Sentry, simply set SENTRY_DSN environment variable. Additionally, flexibility customize setup configuring following environment variables: SENTRY_ENVIRONMENT used set environment (e.g., “production”, “staging”, “development”). set, environment defaults “development”. SENTRY_RELEASE used set release Sentry. set, release defaults “unspecified”.","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/index.html","id":"purpose-and-scope-for-clinical-trial-randomization","dir":"","previous_headings":"","what":"Purpose and Scope for Clinical Trial Randomization","title":"Unbiased: Production-Grade Randomization API","text":"Randomization gold standard conducting clinical trials fundamental aspect clinical trials, studies comparing two arms. cases randomization desirable technique ensure patients randomly allocated defined groups. essential maintaining integrity trial ensuring results reliable, blinding research personnel. However, situations desirable studies balance patients terms numbers group , addition, achieve balance respect relevant factors, sex diabetes type. Adequate selection randomization methods allows intended randomization goals realized. Unbiased compared standard commonly used randomization methods, e.g. simple method block method, apart methods, additionally offers enhanced features flexible adaptive methods, based current information allocation patients trial. Compared , example, block randomization, adaptive randomization ensures relatively equal allocation patient groups, also allows groups balanced basis certain important covariates, key advantage. randomization requires predefined criteria, probability given patient assigned group based minimizing total imbalance, weights can assigned personally individual covariate. advanced algorithmic approach sets apart others minimizing selection bias improving overall efficiency randomization process clinical trials. Unbiased allows use simple, block adaptive minimization randomization methods relevant conduct clinical trials, package caters needs clinical trial randomization. … find differences randomization methods, read vignette Comparative Analysis Randomization Methods.","code":""},{"path":"https://ttscience.github.io/unbiased/index.html","id":"comparison-with-other-solutions","dir":"","previous_headings":"","what":"Comparison with other solutions","title":"Unbiased: Production-Grade Randomization API","text":"many packages perform specific randomization methods R. designed stratified randomization permuted blocks, blockrand randomizeR. also utilize options using minimization randomization - e.g. randpack Minirand. unique contribution landscape integration comprehensive API commitment rigorous testing. dual focus ensures unbiased supports practical needs clinical trials, also aligns technical requirements modern clinical research environments. prioritizing aspects, unbiased addresses critical gap market: need eCRF-compatible randomization solution dependable easily integrated existing workflows. , together implementation minimization techniques, sets unbiased apart novel, comprehensive tool.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":null,"dir":"Reference","previous_headings":"","what":"AuditLog Class — AuditLog","title":"AuditLog Class — AuditLog","text":"class used internally store audit logs request.","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"public-methods","dir":"Reference","previous_headings":"","what":"Public methods","title":"AuditLog Class — AuditLog","text":"AuditLog$new() AuditLog$disable() AuditLog$is_enabled() AuditLog$set_request_body() AuditLog$set_response_body() AuditLog$set_ip_address() AuditLog$set_user_agent() AuditLog$set_event_type() AuditLog$set_study_id() AuditLog$set_response_code() AuditLog$validate_log() AuditLog$persist() AuditLog$clone()","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$new(request_method, endpoint_url)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-1","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$disable()"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-2","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$is_enabled()"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-3","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_request_body(request_body)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-4","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_response_body(response_body)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-5","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_ip_address(ip_address)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-6","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_user_agent(user_agent)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-7","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_event_type(event_type)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-8","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_study_id(study_id)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-9","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$set_response_code(response_code)"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-10","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$validate_log()"},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-11","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$persist()"},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"method-clone-","dir":"Reference","previous_headings":"","what":"Method clone()","title":"AuditLog Class — AuditLog","text":"objects class cloneable method.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"usage-12","dir":"Reference","previous_headings":"","what":"Usage","title":"AuditLog Class — AuditLog","text":"","code":"AuditLog$clone(deep = FALSE)"},{"path":"https://ttscience.github.io/unbiased/reference/AuditLog.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"AuditLog Class — AuditLog","text":"deep Whether make deep clone.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_event_type.html","id":null,"dir":"Reference","previous_headings":"","what":"Set Audit Log Event Type — audit_log_set_event_type","title":"Set Audit Log Event Type — audit_log_set_event_type","text":"function sets event type audit log. retrieves audit log request's internal data, calls audit log's set_event_type method provided event type.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_event_type.html","id":"ref-usage","dir":"Reference","previous_headings":"","what":"Usage","title":"Set Audit Log Event Type — audit_log_set_event_type","text":"","code":"audit_log_set_event_type(event_type, req)"},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_event_type.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"Set Audit Log Event Type — audit_log_set_event_type","text":"event_type event type set audit log. req request object, contain audit log internal data.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_event_type.html","id":"value","dir":"Reference","previous_headings":"","what":"Value","title":"Set Audit Log Event Type — audit_log_set_event_type","text":"Returns nothing modifies audit log -place.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_study_id.html","id":null,"dir":"Reference","previous_headings":"","what":"Set Audit Log Study ID — audit_log_set_study_id","title":"Set Audit Log Study ID — audit_log_set_study_id","text":"function sets study ID audit log. retrieves audit log request's internal data, calls audit log's set_study_id method provided study ID.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_study_id.html","id":"ref-usage","dir":"Reference","previous_headings":"","what":"Usage","title":"Set Audit Log Study ID — audit_log_set_study_id","text":"","code":"audit_log_set_study_id(study_id, req)"},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_study_id.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"Set Audit Log Study ID — audit_log_set_study_id","text":"study_id study ID set audit log. req request object, contain audit log internal data.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/audit_log_set_study_id.html","id":"value","dir":"Reference","previous_headings":"","what":"Value","title":"Set Audit Log Study ID — audit_log_set_study_id","text":"Returns nothing modifies audit log -place.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":null,"dir":"Reference","previous_headings":"","what":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"randomize_dynamic function implements dynamic randomization algorithm using minimization method proposed Pocock (Pocock Simon, 1975). requires defining basic study parameters: number arms (K), number covariates (C), patient allocation ratios (\\(a_k\\)) (k = 1,2,…., K), weights covariates (\\(w_i\\)) (= 1,2,…., C), maximum probability (p) assigning patient group smallest total unbalance multiplied respective weights (\\(G_k\\)). total unbalance first patient regardless assigned arm, patient randomly allocated given arm. Subsequent patients randomized based calculation unbalance depending selected method: \"range\", \"var\" (variance), \"sd\" (standard deviation). case two arms, \"range\" method equivalent \"sd\" method.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"ref-usage","dir":"Reference","previous_headings":"","what":"Usage","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"","code":"randomize_minimisation_pocock( arms, current_state, weights, ratio, method = \"var\", p = 0.85 )"},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"arms character() Arm names. current_state tibble() table covariates current arm assignments column arm, last row contains new patient empty string arm weights numeric() vector positive weights, equal length number covariates, numbered covariates, defaults equal weights ratio integer() Vector positive integers (0 allowed), equal length number arms, named arms, defaults equal weight method character() Function used compute within-arm variability, must one : sd, var, range, defaults var p numeric() single value, proportion randomness (0, 1) randomization vs determinism, defaults 85% deterministic","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"value","dir":"Reference","previous_headings":"","what":"Value","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"character() name arm assigned patient","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"details","dir":"Reference","previous_headings":"","what":"Details","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"Initially, algorithm creates matrix results comparing newly randomized patient current balance patients based defined covariates. next step, arm specified covariate, various scenarios patient allocation calculated. existing results (n) updated new patient, , considering ratio coefficients, results divided specific allocation ratio. Depending method, total unbalance calculated, taking account weights, number covariates using one three methods (“sd”, “range”, “var”). Based number defined arms, minimum value (\\(G_k\\)) (defined weighted sum level-based imbalance) selects arm patient assigned predefined probability (p). probability patient assigned arm equal (1-p)/(K-1) remaining arms.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"note","dir":"Reference","previous_headings":"","what":"Note","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"function's implementation refactored adaptation codebase 'Minirand' package.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"references","dir":"Reference","previous_headings":"","what":"References","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"Pocock, S. J., & Simon, R. (1975). Minimization: new method assigning patients treatment control groups clinical trials. Minirand Package: Man Jin, Adam Polis, Jonathan Hartzel. (https://CRAN.R-project.org/package=Minirand)","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_minimisation_pocock.html","id":"ref-examples","dir":"Reference","previous_headings":"","what":"Examples","title":"Patient Randomization Using Minimization Method — randomize_minimisation_pocock","text":"","code":"n_at_the_moment <- 10 arms <- c(\"control\", \"active low\", \"active high\") sex <- sample(c(\"F\", \"M\"), n_at_the_moment + 1, replace = TRUE, prob = c(0.4, 0.6) ) diabetes <- sample(c(\"diabetes\", \"no diabetes\"), n_at_the_moment + 1, replace = TRUE, prob = c(0.2, 0.8) ) arm <- sample(arms, n_at_the_moment, replace = TRUE, prob = c(0.4, 0.4, 0.2) ) |> c(\"\") covar_df <- tibble::tibble(sex, diabetes, arm) covar_df #> # A tibble: 11 × 3 #> sex diabetes arm #> #> 1 M no diabetes \"control\" #> 2 F no diabetes \"active low\" #> 3 F no diabetes \"active high\" #> 4 M no diabetes \"control\" #> 5 M no diabetes \"active low\" #> 6 M no diabetes \"control\" #> 7 M no diabetes \"control\" #> 8 M no diabetes \"control\" #> 9 F diabetes \"active low\" #> 10 F no diabetes \"active low\" #> 11 F no diabetes \"\" randomize_minimisation_pocock(arms = arms, current_state = covar_df) #> [1] \"active high\" randomize_minimisation_pocock( arms = arms, current_state = covar_df, ratio = c( \"control\" = 1, \"active low\" = 2, \"active high\" = 2 ), weights = c( \"sex\" = 0.5, \"diabetes\" = 1 ) ) #> [1] \"active high\""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_simple.html","id":null,"dir":"Reference","previous_headings":"","what":"Simple randomization — randomize_simple","title":"Simple randomization — randomize_simple","text":"Randomly assigns patient one arms according specified ratios, regardless already performed assignments.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_simple.html","id":"ref-usage","dir":"Reference","previous_headings":"","what":"Usage","title":"Simple randomization — randomize_simple","text":"","code":"randomize_simple(arms, ratio)"},{"path":"https://ttscience.github.io/unbiased/reference/randomize_simple.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"Simple randomization — randomize_simple","text":"arms character() Arm names. ratio integer() Vector positive integers (0 allowed), equal length number arms, named arms, defaults equal weight","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_simple.html","id":"value","dir":"Reference","previous_headings":"","what":"Value","title":"Simple randomization — randomize_simple","text":"Selected arm assignment.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/randomize_simple.html","id":"ref-examples","dir":"Reference","previous_headings":"","what":"Examples","title":"Simple randomization — randomize_simple","text":"","code":"randomize_simple(c(\"active\", \"placebo\"), c(\"active\" = 2, \"placebo\" = 1)) #> [1] \"active\""},{"path":"https://ttscience.github.io/unbiased/reference/run_unbiased.html","id":null,"dir":"Reference","previous_headings":"","what":"Run API — run_unbiased","title":"Run API — run_unbiased","text":"Starts unbiased API.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/run_unbiased.html","id":"ref-usage","dir":"Reference","previous_headings":"","what":"Usage","title":"Run API — run_unbiased","text":"","code":"run_unbiased()"},{"path":"https://ttscience.github.io/unbiased/reference/run_unbiased.html","id":"arguments","dir":"Reference","previous_headings":"","what":"Arguments","title":"Run API — run_unbiased","text":"host character(1) Host URL. port integer(1) Port serve API .","code":""},{"path":"https://ttscience.github.io/unbiased/reference/run_unbiased.html","id":"value","dir":"Reference","previous_headings":"","what":"Value","title":"Run API — run_unbiased","text":"Function called serve API caller thread.","code":""},{"path":"https://ttscience.github.io/unbiased/reference/unbiased-package.html","id":null,"dir":"Reference","previous_headings":"","what":"unbiased: Unbiased: Production-Grade Randomization API — unbiased-package","title":"unbiased: Unbiased: Production-Grade Randomization API — unbiased-package","text":"Unbiased package delivers minimization-based randomization algorithm patient allocation clinical trials, fully integrated production-ready API. designed work seamlessly persistent PostgreSQL database, ensuring reliable data management integrity. Packaged precompiled Docker images, Unbiased simplifies deployment just running docker-compose , making exceptionally straightforward incorporate environment.","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/reference/unbiased-package.html","id":"author","dir":"Reference","previous_headings":"","what":"Author","title":"unbiased: Unbiased: Production-Grade Randomization API — unbiased-package","text":"Maintainer: Kamil Sijko kamil.sijko@ttsi.com.pl (ORCID) Authors: Kinga Sałata kinga.salata@ttsi.com.pl Aleksandra Duda aleksandra.duda@ttsi.com.pl Łukasz Wałejko lukasz.walejko@ttsi.com.pl Jagoda Głowacka-Walas jagoda.glowacka-walas@ttsi.com.pl (ORCID) Laura Bąkała contributors: Michał Seweryn michal.seweryn@biol.uni.lodz.pl (ORCID) [contributor] Transition Technologies Science Sp. z o.o. [funder, copyright holder]","code":""},{"path":[]},{"path":"https://ttscience.github.io/unbiased/news/index.html","id":"new-features-1-0-0","dir":"Changelog","previous_headings":"","what":"New Features","title":"unbiased 1.0.0","text":"Implemented support adaptive randomization using Pocock’s minimization algorithm, integrating new R function based Minirand package Man Jin, Adam Polis, Jonathan Hartzel (Minirand Package). Introduced new POST endpoints facilitating creation studies randomization subjects. Added new GET endpoints comprehensive access study overviews, -depth details, information randomized patients. Implemented audit trial mechanism systematically logs stores request database alongside unbiased’s response. Introduced new GET endpoint enabling users access complete audit trail specific study.","code":""},{"path":"https://ttscience.github.io/unbiased/news/index.html","id":"vignettes--articles-1-0-0","dir":"Changelog","previous_headings":"","what":"Vignettes / Articles","title":"unbiased 1.0.0","text":"Added new article benchmarking Pocock’s minimization algorithm permuted block randomization simple randomization, focusing balance covariates.","code":""},{"path":"https://ttscience.github.io/unbiased/news/index.html","id":"devops-integration-1-0-0","dir":"Changelog","previous_headings":"","what":"DevOps Integration","title":"unbiased 1.0.0","text":"Integrated project Sentry capture errors, empowering users provide credentials receive notifications event unexpected occurrences, including HTTP 500 instances. Integrated project CodeCov, achieving code coverage 95% higher, maintenance level improvement required merging. Integrated project pkgdown, hosting project site ttscience.github.io/unbiased/. Enforced use linter, ensuring errors present upon merging.","code":""},{"path":"https://ttscience.github.io/unbiased/news/index.html","id":"smaller-improvements-1-0-0","dir":"Changelog","previous_headings":"","what":"Smaller Improvements","title":"unbiased 1.0.0","text":"Improved handling malformed JSONs, now returning HTTP 400 instead default HTTP 500 (plumber behavior).","code":""}]