tests/testthat/test_Miscellaneous_Functions.R

# gsignal Miscellaneous Functions
library(gsignal)
library(testthat)

# -----------------------------------------------------------------------
# ifft() and imvfft()

test_that("parameters to ifft() are correct", {
  expect_error(ifft())
  expect_error(ifft('invalid'))
  expect_error(ifft(1, -2))
  expect_error(ifft(1, 2, 3, 4, 5))
})

test_that("ifft() tests are correct", {
  expect_equal(ifft(stats::fft(1:10)), 1:10)
  expect_equal(ifft(stats::fft(c(1+5i, 2+3i, 3+2i, 4+6i, 5+2i))), c(1+5i, 2+3i, 3+2i, 4+6i, 5+2i))
  expect_equal(imvfft(stats::mvfft(matrix(1:20, 4, 5))), matrix(1:20, 4, 5))
})

# -----------------------------------------------------------------------
# pad(), prepad(), postpad()

test_that("parameters to pad() are correct", {
  expect_error(pad())
  expect_error(pad('invalid'))
  expect_error(pad(1, -2))
  expect_error(pad(1, 2, 3, 4, 5, 6))
})

test_that("pad() tests are correct", {
  v <- 1:24
  expect_equal(postpad(v, 30), c(1:24, rep(0, 6)))
  expect_equal(postpad(v, 20), 1:20)
  expect_equal(prepad(v, 30), c(rep(0, 6), 1:24))
  expect_equal(prepad(v, 20), 5:24)
  
  m <- matrix(1:24, 4, 6)
  expect_equal(postpad(m, 8, 100), matrix(c(1:4, rep(100, 4), 5:8, rep(100, 4), 9:12, rep(100, 4),
                                            13:16, rep(100, 4), 17:20, rep(100, 4), 21:24, rep(100, 4)),
                                          8, 6, byrow = FALSE))
  expect_equal(postpad(m, 8, 100, MARGIN = 1), matrix(c(1:24, rep(100, 8)), 4, 8))
  expect_equal(prepad(m, 8, 100), matrix(c(rep(100, 4), 1:4, rep(100, 4), 5:8, rep(100, 4), 9:12, rep(100, 4),
                                            13:16, rep(100, 4), 17:20, rep(100, 4), 21:24),
                                          8, 6, byrow = FALSE))
  expect_equal(prepad(m, 8, 100, MARGIN = 1), matrix(c(rep(100, 8), 1:24), 4, 8))
  
  expect_equal(postpad(m, 2), matrix(c(1, 2, 5, 6, 9, 10, 13, 14, 17, 18, 21, 22), 2, 6))
  expect_equal(postpad(m, 2, MARGIN = 1), matrix(1:8, 4, 2))
  expect_equal(prepad(m, 2), matrix(c(3, 4, 7, 8, 11, 12, 15, 16, 19, 20, 23, 24), 2, 6))
  expect_equal(prepad(m, 2, MARGIN = 1), matrix(17:24, 4, 2))
})

# -----------------------------------------------------------------------
# poly() 

test_that("parameters to poly() are correct", {
  expect_error(poly())
  expect_error(poly('invalid'))
  expect_error(poly(1, 2))
  expect_error(poly(matrix(1:6, 2, 3)))
})

test_that("poly() tests are correct", {
  expect_equal(poly(0), c(1, 0))
  expect_equal(poly(1), c(1, -1))
  expect_equal(poly(-1), c(1, 1))
  expect_equal(poly(c(1, 2, 3)), c(1, -6, 11, -6))
  expect_equal(poly(matrix(1:4, 2, 2, byrow = TRUE)), c(1, -5, -2))
  expect_equal(poly(c(-1 + 1i)), c(1 + 0i, 1 - 1i))
})

# -----------------------------------------------------------------------
# filter() 

test_that("parameters to filter() are correct", {
  expect_error(filter())
  expect_error(filter(1, 2))
  expect_error(filter(1, 2, 'invalid'))
  expect_error(filter(1, 1, 1:10, 'invalid'))
  expect_error(filter(c(1, 1), 1, 1:10, c(0, 0)))
})

test_that("filter() tests are correct", {
  a <- c(1, 1)
  b <- c(1, 1)
  x <- c(1, rep(0L, 9))
  expect_equal(filter(b, 1, x), c(rep(1L, 2), rep(0L, 8)))
  filt <- Ma(b)
  expect_equal(filter(filt, x), c(rep(1L, 2), rep(0L, 8)))
  expect_equal(filter(1, a, x), rep(c(1L, -1L), 5))
  filt <- Arma(b, a)
  expect_equal(filter(filt, x), c(1L, rep(0L, 9)))

  # # complex input  
  # r <- sqrt (1/2) * (1 + 1i)
  # a <- a * r
  # b <- b * r
  # expect_equal(suppressWarnings(filter (b, 1, x)), Re(r * c(rep(1L, 2), rep(0L, 8))))
  # expect_equal(suppressWarnings(filter (b, a, x)), c(1L, rep(0L, 9)))
  
  a <- c(1, 1)
  b <- c(1, 1)
  x <- c(1, rep(0L, 9))
  lst <- filter (b, 1, x, -1)
  expect_equal(lst[['y']], c(0, 1, 0, 0, 0, 0, 0, 0, 0, 0))
  expect_equal(lst[['zf']], 0)
  
  b <- c(1, 1)
  x  <- y0 <- matrix(0L, 10, 3)
  x[1, 1] <- -1;  x[1, 2] <- 1
  y0[1:2, 1] <- -1;  y0[1:2, 2] <- 1
  y <- filter(b, 1, x)
  expect_equal(y, y0)
  
  expect_equal(filter(1, rep(1, 10) / 10, NULL), NULL)
  expect_equal(filter(1, rep(1, 10) / 10, rep(0,10)), rep(0,10))
  expect_equal(filter(1, rep(1, 10) / 10, 1:5), rep(10, 5))
  
  # Test using initial conditions
  expect_equal(filter(c(1, 1, 1), c(1, 1), c(1, 2), c(1, 1))[['y']], c(2, 2))
  expect_equal(filter(c(1, 3), 1, matrix(1:6, ncol = 2, byrow = TRUE), matrix(c(4, 5), ncol = 2))[['y']], 
               matrix(c(5, 7, 6, 10, 14, 18), ncol = 2, byrow = TRUE))
})

# -----------------------------------------------------------------------
# conv()

test_that("parameters to conv() are correct", {
  expect_error(conv())
  expect_error(conv(1))
  expect_error(conv(1, 2, 3, 4))
  expect_error(conv(1, 2, 'invalid'))
})

test_that("conv() tests are correct", {
  x <- rep(1L, 3); b <- 2; c <- 3
  expect_equal(conv(x, x), c(1, 2, 3, 2, 1))
  expect_equal(conv(x, b), rep(2L, 3))
  expect_equal(conv(b, x), rep(2L, 3))
  expect_equal(conv(x, c), rep(3L, 3))
  expect_equal(conv(c, x), rep(3L, 3))
  expect_equal(conv(b, c), 6)

  a <- 1:10; b <- 1:3
  expect_equal(length(conv (a,b)), length(a) + length(b) - 1)
  expect_equal(length(conv (b,a)), length(a) + length(b) - 1)
  expect_equal(conv(a, b, "full"), conv (a,b))
  expect_equal(conv(b, a, "full"), conv (b,a))
  expect_equal(conv(a, b, "same"), c(4, 10, 16, 22, 28, 34, 40, 46, 52, 47))
  expect_equal(conv(b, a, "same"), c(28, 34, 40))
  expect_equal(conv(a, b, "valid"), c(10, 16, 22, 28, 34, 40, 46, 52))
  expect_equal(conv(b, a, "valid"), NULL)
  expect_equal(conv(a, a, "valid"), 220L)
  expect_equal(conv(b, b, "valid"), 10L)
})

# -----------------------------------------------------------------------
# fftconv()

test_that("parameters to fftconv() are correct", {
  expect_error(fftconv())
  expect_error(fftconv(1))
  expect_error(fftconv(1, 2, 3, 4))
})

test_that("fftconv() tests are correct", {
  x <- rep(1L, 3); b <- 2; c <- 3
  expect_equal(fftconv(x, x), c(1, 2, 3, 2, 1))
  expect_equal(fftconv(x, b), rep(2L, 3))
  expect_equal(fftconv(b, x), rep(2L, 3))
  expect_equal(fftconv(x, c), rep(3L, 3))
  expect_equal(fftconv(c, x), rep(3L, 3))
  expect_equal(fftconv(b, c), 6)
  
  a <- 1:10; b <- 1:3
  expect_equal(length(fftconv (a,b)), length(a) + length(b) - 1)
  expect_equal(length(fftconv (b,a)), length(a) + length(b) - 1)
  expect_equal(fftconv(a, b, NULL), fftconv (a,b))
  expect_equal(fftconv(b, a, NULL), fftconv (b,a))
})


# -----------------------------------------------------------------------
# conv2()

test_that("parameters to conv2() are correct", {
  expect_error(conv2())
  expect_error(conv2(1))
  expect_error(conv2(1, 2, 3, 4))
  expect_error(conv2(matrix(1,1), matrix(2,1), 'invalid'))
})

test_that("conv2() tests are correct", {
  a <- matrix(1:16, 4, 4)
  b <- matrix(1:9, 3,3)
  ans <- matrix(c(1, 9, 36, 84, 115, 91,
                  4, 29, 99, 207, 263, 202,
                  10, 62, 192, 372, 446, 334,
                  16, 83, 237, 417, 485, 358,
                  17, 75, 198, 330, 365, 263,
                  12, 48, 120, 192, 204, 144),
                6, 6, byrow = TRUE)
  expect_equal(conv2(a, b), ans)
  expect_equal(conv2(a, b, 'same'), ans[2:5, 2:5])
  expect_equal(conv2(a, b, 'valid'), ans[3:4, 3:4])

  a <- matrix(c(1:5, 1:5), 2, 5, byrow = TRUE)
  b <- matrix(1:2, 1, 2)
  ans <- matrix(rep(c(1,4,7,10,13,10),2),2,6, byrow=T)
  expect_equal(conv2(a, b), ans)
  expect_equal(conv2(a, b, 'same'), ans[1:2, 2:6])
  expect_equal(conv2(a, b, 'valid'), ans[1:2, 2:5])
})

# -----------------------------------------------------------------------
# cplxpair()

test_that("parameters to cplxpair() are correct", {
  expect_error(cplxpair())
  expect_error(cplxpair(1, -1))
  expect_error(cplxpair(1, 2, 3, 4))
  expect_error(cplxpair(c(2000 * (1 + .Machine$double.eps) + 4i,
                          2000 * (1 - .Machine$double.eps) - 4i), 0))
  expect_error(cplxpair(c(2e6 + 1i, 2e6 - 1i, 1e-9 * (1 + 1i), 1e-9 * (1 - 2i))))
})

test_that("cplxpair() tests are correct", {
  expect_equal(cplxpair(1), 1)
  expect_equal(cplxpair(c(1 + 1i, 1-1i)), c(1 - 1i, 1 + 1i))
  expect_equal(cplxpair(c(1 + 1i, 1 + 1i, 1, 1 - 1i, 1 - 1i, 2)), 
               c(1 - 1i, 1 + 1i, 1 - 1i, 1 + 1i, 1, 2))
  expect_equal(cplxpair(c(0, 1, 2)), c(0, 1, 2))
  expect_equal(cplxpair(c(2000 * (1 + .Machine$double.eps) + 4i,
                          2000 * (1 - .Machine$double.eps) - 4i)),
               c(2000 - 4i, 2000 + 4i))
  z <- c(1 + 1i, 1 + 1i, 1, 1 - 1i, 1 - 1i, 2)
  ans <- cplxpair(z)
  m <- cbind(z, z)
  expect_equivalent(cplxpair(m, MARGIN = 2), cbind(ans, ans))
  expect_error(cplxpair(m, MARGIN = 1))
  
  # shared z,y
  z <- exp (2i * pi * c(4, 3, 5, 2, 6, 1, 0) / 7)
  z[2] <- Conj(z[1])
  z[4] <- Conj(z[3])
  z[6] <- Conj(z[5])
  expect_equal(cplxpair(z[pracma::randperm(7)]), z)
  expect_equal(cplxpair(z[pracma::randperm(7)]), z)
  expect_equal(cplxpair(z[pracma::randperm(7)]), z)
  expect_equal(cplxpair(cbind(z[pracma::randperm(7)], z[pracma::randperm(7)])), 
               cbind(z, z, deparse.level = 0))
  expect_equal(cplxpair(cbind(z[pracma::randperm(7)], z[pracma::randperm(7)])), 
               cbind(z, z, deparse.level = 0))
  y <- c(-1-1i,  -1+1i, -3, -2, 1, 2, 3)
  expect_equal(cplxpair(cbind(z[pracma::randperm(7)], y[pracma::randperm(7)], deparse.level = 0)),
                        cbind(z, y, deparse.level = 0))
  expect_equal(cplxpair(cbind(z[pracma::randperm(7)], 
                              y[pracma::randperm(7)],
                              z[pracma::randperm(7)], deparse.level = 0)),
               cbind(z, y, z, deparse.level = 0))
  
  # Test tolerance
  expect_equal(cplxpair(c(2000 * (1 + .Machine$double.eps) + 4i,
                       2000 * (1 - .Machine$double.eps) - 4i)),
               c(2000 - 4i,  2000 + 4i), 
               tolerance = 100 * .Machine$double.eps)
  expect_error(cplxpair(c(2000 * (1 + .Machine$double.eps) + 4i,
                          2000 * (1 - .Machine$double.eps) - 4i), tol = 0))
  expect_error(cplxpair(c(2e6+1i, 2e6-1i, 1e-9 * (1+1i), 1e-9 * (1-2i))))
})

# -----------------------------------------------------------------------
# unwrap() 

test_that("parameters to unwrap() are correct", {
  expect_error(unwrap())
  expect_error(unwrap('invalid'))
  expect_error(unwrap(1 + 0i, 0))
  expect_error(unwrap(matrix(1), 2, 3))
})

test_that("unwrap() tests are correct", {
  
  # shared i, t, r, w, tol
  i <- 0
  t <- NULL
  r <- 0:100
  w <- r - 2 * pi * floor((r + pi) / (2 * pi))
  tol <- 1e3 * .Machine$double.eps
  
  expect_equal(unwrap(w), r, tolerance = tol)
  expect_equivalent(unwrap(cbind(w, w)), cbind(r, r))

  ## Test that small values of tol have the same effect as tol = pi
  expect_equal(unwrap(w, 0.1), r)
  expect_equal(unwrap(w, tol), r)

  ## Test that phase changes larger than 2*pi unwrap properly
  expect_equal(unwrap(c(0, 1)), c(0, 1))
  expect_equal(unwrap(c(0, 4)), c(0, 4 - 2 * pi))
  expect_equal(unwrap(c(0, 7)), c(0, 7 - 2 * pi))
  expect_equal(unwrap(c(0, 10)), c(0, 10 - 4 * pi))
  expect_equal(unwrap(c(0, 13)), c(0, 13 - 4 * pi))
  expect_equal(unwrap(c(0, 16)), c(0, 16 - 6 * pi))
  expect_equal(unwrap(c(0, 19)), c(0, 19 - 6 * pi))
  expect_lt(max(abs(diff(unwrap(100 * pi * runif(1000, 1))))), pi)
 
  A <- c(pi*(-4), pi*(-2+1/6), pi/4, pi*(2+1/3), pi*(4+1/2), pi*(8+2/3), pi*(16+1), pi*(32+3/2), pi*64)
  expect_equal(unwrap(A), unwrap(A, pi))

})

# -----------------------------------------------------------------------
# mpoles()

test_that("parameters to mpoles() are correct", {
  expect_error(mpoles())
  expect_error(mpoles(1, 2, 3, 4, 5))
  expect_error(mpoles(1, 'invalid', TRUE, TRUE))
  expect_error(mpoles(1, 0.001, 'invalid', TRUE))
  expect_error(mpoles(1, 0.001, TRUE, 'invalid'))
})

test_that("mpoles() tests are correct", {
  res <- mpoles(c(0, 0), 0.01)
  expect_equal(res, c(1, 2))
  
  p <- c(2, 3, 1, 1, 2)
  res <- mpoles (p, index.return = TRUE)
  expect_equal(res$m, c(1, 1, 2, 1, 2))
  expect_equal(res$n, c(2, 5, 1, 4, 3))
  expect_equal(p[res$n], c(3, 2, 2, 1, 1))
})

# -----------------------------------------------------------------------
# polyreduce() 

test_that("parameters to polyreduce() are correct", {
  expect_error(polyreduce())
  expect_error(polyreduce(NULL))
  expect_error(polyreduce(1, 2))
})

test_that("polyreduce() tests are correct", {
  expect_equal(polyreduce(c(0, 0, 1, 2, 3)), c(1, 2, 3))
  expect_equal(polyreduce(c(1, 2, 3, 0, 0)), c(1, 2, 3, 0, 0))
  expect_equal(polyreduce(c(1, 0, 3)), c(1, 0, 3))
  expect_equal(polyreduce(c(0, 0, 0)), 0)
})

# -----------------------------------------------------------------------
# residue() and rresidue() 

test_that("parameters to residue() are correct", {
  expect_error(residue())
  expect_error(residue(NULL))
  expect_error(residue(1, 2, 3, 4))
  expect_error(rresidue())
  expect_error(rresidue(NULL))
  expect_error(rresidue(1, 2, 3, 4, 5))
})

test_that("residue() tests are correct", {
  tol <- 1e-6
  
  b <- c(1, 1, 1)
  a <- c(1, -5, 8, -4)
  rpk <- residue (b, a)
  expect_equal(Re(rpk$r), c(-2, 7, 3))
  expect_equal(Re(rpk$p), c(2, 2, 1))
  expect_null(rpk$k)
  ba <- rresidue (rpk$r, rpk$p, rpk$k)
  expect_equal(Re(ba$b), b)
  expect_equal(Re(ba$a), a)

  b <- c(1, 0, 1)
  a <- c(1, 0, 18, 0, 81)
  rpk <- residue (b, a)
  expect_equal(rpk$r, c(-5i, 12, +5i, 12) / 54, tolerance = tol)
  expect_equal(rpk$p, c(+3i, +3i, -3i, -3i))
  expect_null(rpk$k)
  ba <- rresidue (rpk$r, rpk$p, rpk$k)
  expect_equal(Re(ba$b), c(0, b))
  expect_equal(Re(ba$a), a)
  
  r <- c(7, 3, -2)
  p <- c(2, 1, 2)
  k <- c(1, 0)
  ba <- rresidue(r, p, k)
  expect_equal(Re(ba$b), c(1, -5, 18, -39, 28))
  expect_equal(Re(ba$a), c(1, -5, 8, -4))
  rpk <- residue(ba$b, ba$a)
  mn <- mpoles(p, index.return = TRUE)
  expect_equal(sort(Re(rpk$r)), sort(r[mn$n]))
  
  b <- 1
  a <- c(1, 10, 25)
  rpk <- residue (b, a)
  expect_equal(Re(rpk$r), c(0, 1))
  expect_equal(Re(rpk$p), c(-5, -5))
  expect_null(rpk$k)
  ba <- rresidue (rpk$r, rpk$p, rpk$k)
  expect_equal(Re(ba$b), b)
  expect_equal(Re(ba$a), a)
  
})
gjmvanboxtel/gsignal documentation built on Nov. 22, 2023, 8:19 p.m.