#' Source: pcalg package
#' Note that gInput here should be an adjacency matrix where
#' adj[i,j] = 1 if i -> j (as opposed to the input format of many other pcalg functions)
#' Note: the correctness of this function has not been verified
apply_orientation_rules <- function(gInput, verbose = FALSE) {
  res <- gInput
  if (!is.matrix(gInput)) {
    if (sum(gInput) > 0) {
      g <- as(gInput, "matrix")
      p <- as.numeric(dim(g)[1])
      pdag <- g
      ind <- which(g == 1, arr.ind = TRUE)
    } else {
      cat("Invalid or empty PDAG! This function only accepts graphNEL or adj mat!\n")
      return(NULL)
    }
  } else {
    if (length(res[1, ]) > 0) {
      g <- res
      p <- length(g[1, ])
      pdag <- g
      ind <- which(g == 1, arr.ind = TRUE)
    } else {
      cat("Invalid or empty PDAG! This function only accepts graphNEL or adj mat!\n")
      return(NULL)
    }
  }
  old_pdag <- matrix(0, p, p)
  while (!all(old_pdag == pdag)) {
    old_pdag <- pdag
    ind <- which((pdag == 1 & t(pdag) == 0), arr.ind = TRUE)
    for (i in seq_len(nrow(ind))) {
      a <- ind[i, 1]
      b <- ind[i, 2]
      indC <- which((pdag[b, ] == 1 & pdag[, b] == 1) & (pdag[a, ] == 0 & pdag[, a] == 0))
      if (length(indC) > 0) {
        pdag[b, indC] <- 1
        pdag[indC, b] <- 0
        if (verbose) cat("\nRule 1:", a, "->", b, " and ", b, "-", indC, " where ", a, " and ", indC, " not connected: ", b, "->", indC, "\n")
      }
    }
    ind <- which((pdag == 1 & t(pdag) == 1), arr.ind = TRUE)
    for (i in seq_len(nrow(ind))) {
      a <- ind[i, 1]
      b <- ind[i, 2]
      indC <- which((pdag[a, ] == 1 & pdag[, a] == 0) & (pdag[, b] == 1 & pdag[b, ] == 0))
      if (length(indC) > 0) {
        pdag[a, b] <- 1
        pdag[b, a] <- 0
        if (verbose) cat("\nRule 2: Kette ", a, "->", indC, "->", b, ":", a, "->", b, "\n")
      }
    }
    ind <- which((pdag == 1 & t(pdag) == 1), arr.ind = TRUE)
    for (i in seq_len(nrow(ind))) {
      a <- ind[i, 1]
      b <- ind[i, 2]
      indC <- which((pdag[a, ] == 1 & pdag[, a] == 1) & (pdag[, b] == 1 & pdag[b, ] == 0))
      if (length(indC) >= 2) {
        g2 <- pdag[indC, indC]
        if (length(g2) <= 1) {
          g2 <- 0
        } else {
          diag(g2) <- rep(1, length(indC))
        }
        g3 <- g2 + t(g2)
        if (any(g3 == 0)) {
          pdag[a, b] <- 1
          pdag[b, a] <- 0
          if (verbose) cat("\nRule 3:", a, "->", b, "\n")
        }
      }
    }
    ind <- which((pdag == 1 & t(pdag) == 1), arr.ind = TRUE)
    if (length(ind) > 0) {
      for (i in seq_len(nrow(ind))) {
        a <- ind[i, 1]
        b <- ind[i, 2]
        indC <- which((pdag[a, ] == 1 & pdag[, a] == 1) & (pdag[, b] == 0 & pdag[b, ] == 0))
        l.indC <- length(indC)
        if (l.indC > 0) {
          found <- FALSE
          ic <- 0
          while (!found & (ic < l.indC)) {
            ic <- ic + 1
            c <- indC[ic]
            indD <- which((pdag[c, ] == 1 & pdag[, c] == 0) & (pdag[, b] == 1 & pdag[b, ] == 0))
            if (length(indD) > 0) {
              found <- TRUE
              pdag[b, a] <- 0
              if (verbose) cat("Rule 4 applied \n")
            }
          }
        }
      }
    }
  }
  if (!is.matrix(res)) {
    res <- as(pdag, "graphNEL")
  } else {
    res <- pdag
  }
  return(res)
}
# This function orients an edge from u to v
orient_edge <- function(EH, u, v) {
  EH[u, v] <- 1
  EH[v, u] <- 0
  return(EH)
}

# This function removes an orientation of an edge
remove_orientation <- function(EH, u, v) {
  EH[u, v] <- EH[v, u] <- 0
  return(EH)
}

get_undirected_edges <- function(adj) {
  undir <- list()

  indices <- which(adj == 1, arr.ind = TRUE)
  triangular_indices <- indices[indices[,1] < indices[,2], ,drop = FALSE]
  for (i in seq_len(nrow(triangular_indices))) {
    row_idx <- triangular_indices[i, 1]
    col_idx <- triangular_indices[i, 2]
    if (adj[col_idx, row_idx] == 1) {
      undir <- c(undir, list(as.numeric(triangular_indices[i,])))
    }
  }
  return(undir)
}

# This is the recursive enumeration function
enumerate_cpdag <- function(EH, undirected = NULL) {
  results <- list()
  H <- apply_orientation_rules(EH)
  if (is.null(undirected)) {
    undirected <- get_undirected_edges(H)
  }

  if (length(undirected) == 0) {
    return(list(H))
  } else {
    for (i in seq_along(undirected)) {
      u <- undirected[[i]][1]
      v <- undirected[[i]][2]

      # Orient u -> v
      EH <- orient_edge(H, u, v)
      result_1 <- enumerate_cpdag(EH)

      # Undo orientation
      #EH <- remove_orientation(EH, u, v)

      # Orient v -> u and recurse
      EH <- orient_edge(H, v, u)
      result2 <- enumerate_cpdag(EH)
      # Undo orientation
      #EH <- remove_orientation(EH, v, u)

      # if output is matrix
      if (is.matrix(result_1)) {
        results <- append(results, list(result_1))
      } else {
        results <- append(results, result_1)
      }

      if (is.matrix(result2)) {
         results <- append(results, list(result2))
        } else {
            results <- append(results, result2)
      }

    }
  }

  results <- remove_duplicates(results)

  return(results)
}

remove_duplicates <- function(results) {
  new_list <- list()

  for (i in seq_along(results)) {
    identical_found <- FALSE
    for (j in seq_along(new_list)) {
      if (all(results[[i]] == new_list[[j]])) {
        identical_found <- TRUE
        break
      }
    }
    if (!identical_found) {
      new_list <- append(new_list, list(results[[i]]))
    }
  }
  return(new_list)
}

