R Under development (unstable) (2024-01-25 r85826 ucrt) -- "Unsuffered Consequences" Copyright (C) 2024 The R Foundation for Statistical Computing Platform: x86_64-w64-mingw32/x64 R is free software and comes with ABSOLUTELY NO WARRANTY. You are welcome to redistribute it under certain conditions. Type 'license()' or 'licence()' for distribution details. R is a collaborative project with many contributors. Type 'contributors()' for more information and 'citation()' on how to cite R or R packages in publications. Type 'demo()' for some demos, 'help()' for on-line help, or 'help.start()' for an HTML browser interface to help. Type 'q()' to quit R. > # This test file tests the RadixForest class in R/r6_class.r > # 1) That insertion and deletion produce the correct results with random strings > # 2) That search (hamming, levenshtein and anchored) produce the same results as the internal `dist_matrix` and `dist_pairwise` functions > > library(seqtrie) > library(stringdist) > library(stringi) > library(dplyr) Attaching package: 'dplyr' The following objects are masked from 'package:stats': filter, lag The following objects are masked from 'package:base': intersect, setdiff, setequal, union > > # Use 2 threads on github actions and CRAN, 4 threads locally > IS_LOCAL <- Sys.getenv("IS_LOCAL") != "" > NTHREADS <- ifelse(IS_LOCAL, 4, 2) > NITER <- ifelse(IS_LOCAL, 4, 1) > NSEQS <- 10000 # must be larger than 1000 > MAXSEQLEN <- 200 > MAXDIST <- MAXSEQLEN * 0.05 > MAXFRAC <- 0.05 > CHARSET <- "ACGT" > > forest_equal <- function(x, y) { + xs <- gsub("[0-9]+", "#", x$to_string()) + ys <- gsub("[0-9]+", "#", y$to_string()) + if(!setequal(xs, ys)) return(FALSE) + if(x$size() != y$size()) return(FALSE) + return(TRUE) + } > > random_strings <- function(N, charset = "abcdefghijklmnopqrstuvwxyz") { + charset_stri <- paste0("[", charset, "]") + len <- sample(0:MAXSEQLEN, N, replace=TRUE) + result <- lapply(0:MAXSEQLEN, function(x) { + nx <- sum(len == x) + if(nx == 0) return(character()) + stringi::stri_rand_strings(nx, x, pattern = charset_stri) + }) + sample(unlist(result)) + } > > > mutate_strings <- function(x, prob = 0.025, indel_prob = 0.025, charset = "abcdefghijklmnopqrstuvwxyz") { + charset <- unlist(strsplit(charset, "")) + xsplit <- strsplit(x, "") + sapply(xsplit, function(a) { + r <- runif(length(a)) < prob + a[r] <- sample(charset, sum(r), replace=TRUE) + ins <- runif(length(a)) < indel_prob + a[ins] <- paste0(sample(charset, sum(ins), replace=TRUE), sample(charset, sum(ins), replace=TRUE)) + del <- runif(length(a)) < indel_prob + a[del] <- "" + paste0(a, collapse = "") + }) + } > > sd_search <- function(query, target, method = "lv") { + results <- stringdist::stringdistmatrix(query, target, method = method, nthread=NTHREADS) + results <- data.frame(query = rep(query, times=length(target)), + target = rep(target, each=length(query)), + distance = as.vector(results), stringsAsFactors = FALSE) + results <- dplyr::filter(results, is.finite(distance)) + results$distance <- as.integer(results$distance) + dplyr::arrange(results, query, target) + } > > tt <- "RadixForest" > for(. in 1:NITER) { + print(paste0("Checking correct insert/erase methods for ", tt)) + local({ + x <- RadixForest$new() + y <- RadixForest$new() + ins <- c(random_strings(NSEQS, CHARSET),"") + era <- c(sample(c(sample(ins, NSEQS/10), random_strings(NSEQS/10, CHARSET))),"") + x$insert(ins) + stopifnot(x$validate()) + stopifnot(x$size() == n_distinct(ins)) + x$erase(era) + stopifnot(x$validate()) + stopifnot(x$size() == n_distinct(ins[!ins %in% era])) + y$insert(ins[!ins %in% era]) + stopifnot(y$validate()) + stopifnot(forest_equal(x, y)) + }) + + print(paste0('Checking find for ', tt)) + local({ + x <- RadixForest$new() + ins <- c(random_strings(NSEQS, CHARSET),"") + era <- c(sample(c(sample(ins, NSEQS/10), random_strings(NSEQS/10, CHARSET)))) + fin <- c(sample(c(sample(ins, NSEQS/10), random_strings(NSEQS/10, CHARSET))),"") + expected <- fin %in% setdiff(ins, era) + x$insert(ins) + stopifnot(x$validate()) + x$erase(era) + stopifnot(x$validate()) + results <- x$find(fin) + stopifnot(identical(results, expected)) + }) + + print(paste0('Checking prefix_search for ', tt)) + local({ + x <- RadixForest$new() + ins <- c(random_strings(NSEQS, CHARSET),"") + era <- c(sample(c(sample(ins, NSEQS/10), random_strings(NSEQS/10, CHARSET)))) + fin <- c(sample(c(sample(ins, NSEQS/1000), random_strings(NSEQS/1000, CHARSET))),"") %>% substr(1,5) + fin <- c(fin, paste0(fin, substr(fin,1,1))) + ins2 <- setdiff(ins, era) + expected <- lapply(fin, function(f) { + ex <- grep(paste0("^", f), ins2, value=TRUE) + if(length(ex) == 0) return(NULL) + data.frame(query = f, target = ex, stringsAsFactors = F) + }) + expected <- do.call(rbind, expected) + expected <- dplyr::arrange(expected, query, target) + x$insert(ins) + stopifnot(x$validate()) + x$erase(era) + stopifnot(x$validate()) + results <- x$prefix_search(fin) %>% dplyr::arrange(query, target) + stopifnot(identical(results, expected)) + }) + + print(paste0("Checking multithreaded hamming search correctness for ", tt)) + local({ + x <- RadixForest$new() + target <- c(random_strings(NSEQS, CHARSET),"") %>% unique + query <- sample(c(sample(target, NSEQS/1000), random_strings(NSEQS/1000, CHARSET))) + query <- c(mutate_strings(query, indel_prob=0, charset = CHARSET), "") %>% unique + x$insert(target) + stopifnot(x$validate()) + results_dist <- x$search(query, max_distance = MAXDIST, mode = "hamming", nthreads=NTHREADS, show_progress=TRUE) %>% dplyr::arrange(query, target) + results_frac <- x$search(query, max_fraction = MAXFRAC, mode = "hamming", nthreads=NTHREADS, show_progress=TRUE) %>% dplyr::arrange(query, target) + sd_results <- sd_search(query, target, method = "hamming") + sd_dist <- dplyr::filter(sd_results, distance <= MAXDIST) + sd_frac <- dplyr::filter(sd_results, distance <= nchar(query) * MAXFRAC) + stopifnot(identical(results_dist, sd_dist)) + stopifnot(identical(results_frac, sd_frac)) + }) + + print(paste0("Checking multithreaded levenshtein search correctness for ", tt)) + local({ + x <- RadixForest$new() + target <- c(random_strings(NSEQS, CHARSET),"") %>% unique + query <- sample(c(sample(target, NSEQS/1000), random_strings(NSEQS/1000, CHARSET))) + query <- c(mutate_strings(query, charset = CHARSET), "") %>% unique + x$insert(target) + stopifnot(x$validate()) + results_dist <- x$search(query, max_distance = MAXDIST, mode = "levenshtein", nthreads=NTHREADS, show_progress=TRUE) %>% dplyr::arrange(query, target) + results_frac <- x$search(query, max_fraction = MAXFRAC, mode = "levenshtein", nthreads=NTHREADS, show_progress=TRUE) %>% dplyr::arrange(query, target) + sd_results <- sd_search(query, target, method = "lv") + sd_dist <- dplyr::filter(sd_results, distance <= MAXDIST) + sd_frac <- dplyr::filter(sd_results, distance <= nchar(query) * MAXFRAC) + stopifnot(identical(results_dist, sd_dist)) + stopifnot(identical(results_frac, sd_frac)) + }) + } [1] "Checking correct insert/erase methods for RadixForest" [1] "Checking find for RadixForest" [1] "Checking prefix_search for RadixForest" [1] "Checking multithreaded hamming search correctness for RadixForest" |----|----|----|----|----|----|----|----|----|----| *************************************************** |----|----|----|----|----|----|----|----|----|----| *************************************************** [1] "Checking multithreaded levenshtein search correctness for RadixForest" |----|----|----|----|----|----|----|----|----|----| *************************************************** |----|----|----|----|----|----|----|----|----|----| *************************************************** > > proc.time() user system elapsed 12.04 0.60 8.45