#################################################
#################################################
## .BICemp:
##   Bayesian Information criteria for GGM estimation
##    as in Gao et al. (2012)
##
##  INPUTS:
##      mat       : sample covariance/correlation matrix.
##      inv_est   : estimated precision matrix/inverse correlation.
##      p         : total dimension.
##      n         : sample size.
##      threshold : degree count in BIC only considers 
##                    absolute entries over threshold.
##
##  OUTPUTS: 
##      ridge_eig : numeric vector. Shifted eigenvalues
##                   of the matrix.
##
.BICemp <- function(mat, inv_est, p, n, threshold = 0){
  ## After symmetrization, there is a legitimate minimizer. Consider this...
  inv_est_symm  <- (inv_est + t(inv_est)) / 2 
  
  .eig          <- eigen(inv_est_symm)
  .eig_val      <- Re(.eig$values)
  # print(paste0("No. of negative eigenvals: ",sum(.eig_val <= 0)))
  # print(paste0("No. of complex eigenvals: ",sum(Im(.eig_val) != 0)))
  .eig_val[.eig_val <= 0] <- 1e-5
  
  .bic <- 0
  .bic <- .bic - n * sum( log(.eig_val) )
  .bic <- .bic + n * Trace(inv_est %*% mat)
  
  .edgenum <- sum(abs(inv_est[upper.tri(inv_est, FALSE)]) > threshold)
  
  .bic <- .bic + log(n) * .edgenum
  return(.bic)
}


#################################################
#################################################
## BIChwglasso:
##  This function computes the HW-GLASSO estimator
##  for the given range of tuning parameters, 
##  and then compute the BIC of each estimator.
##  Outputs the fit with optimal BIC Tun. parameter.
##
##  INPUTS:
##      mat       : sample covariance/correlation matrix.
##      rho       : tuning parameter range vector.
##      p         : total dimension.
##      n         : sample size.
##      threshold : degree count in BIC only considers 
##                    absolute entries over threshold.
##      maxit     : Maximum number of iterations of GLASSO.
##      penalize.diagonal :
##                    if FALSE, off-diagonal L1 penalty.
##
##  OUTPUTS: 
##    OUTPUT        : list of GLASSO outputs for tun. pars. in rho.
##    BIC           : vector of BIC values for tun. par. in rho.
##    optimal.index : index corresponding to optimal rho.
##    optimal.rho   : optimal tuning parameter value. 
##    optimal.model : object glasso trained with optimal.rho
##    total.time    : total time for training.
##
BIChwglasso <- function(mat, rho, p, n,
                       threshold = 0, maxit = 200,
                       penalize.diagonal = TRUE){
  
  ## Calculate weighting matrix:
  .inv_mat <- solve(mat + 0.1 * diag(p))
  .W  <- matrix(rep(0, p * p), ncol = p)

  for (.i in (2:p)) {
    for(.j in (1:(.i-1))) {
      .a  <- abs(.inv_mat[.i, .j])
      .ai <- sum(abs(.inv_mat[.i, -.i])) 
      .aj <- sum(abs(.inv_mat[.j, -.j])) 
      
      .W[.i, .j] <- (.i != .j) / (.a * .ai * .aj)
    }
  }
  .W <- .W + t(.W)
  
  ## Fit weigthed GLASSO models:
  .rholength = length(rho)
  .BIC.HWGL = rep(NA, .rholength)
  .OUTPUT.HWGL = list()
  
  .start_time = Sys.time()
  for(.i in 1:.rholength){
    if(.i == 1){
      .OUTPUT.HWGL[[.i]] <- glasso(
        s = mat, rho = rho[.i]*.W,
        nobs = n, zero = NULL,
        thr = 1.0e-4, maxit = maxit,  approx = FALSE,
        penalize.diagonal = penalize.diagonal,
        start = "cold", w.init = NULL, wi.init = NULL,
        trace = FALSE)

    } else {
      .OUTPUT.HWGL[[.i]] <- glasso(
        s = mat, rho = rho[.i]*.W,
        nobs = n, zero = NULL, 
        thr = 1.0e-4, maxit = maxit,  approx = FALSE,
        penalize.diagonal = penalize.diagonal, 
        start = "warm",
        w.init  = .OUTPUT.HWGL[[.i-1]]$w, 
        wi.init = .OUTPUT.HWGL[[.i-1]]$wi,
        trace = FALSE)

    }
    .BIC.HWGL[.i] <- .BICemp(
      mat = mat, 
      inv_est = .OUTPUT.HWGL[[.i]]$wi, 
      n = n, p = p, threshold = threshold)

  }
  ## Select optimal model:
  .end_time = Sys.time()
  .total_time_HWGL <- .end_time - .start_time
  
  ## Return outputs:
  .OUTPUT = list(OUTPUT = .OUTPUT.HWGL, 
                 BIC = .BIC.HWGL, 
                 optimal.index = which.min(.BIC.HWGL),
                 optimal.rho = rho[which.min(.BIC.HWGL)],
                 optimal.model = .OUTPUT.HWGL[[which.min(.BIC.HWGL)]],
                 total.time = .total_time_HWGL)
  return(.OUTPUT)
}

