aboutsummaryrefslogtreecommitdiff
path: root/src/remove_pkg.c
blob: 14e1f4afed816ae22a3e3e2cb991b727d393490a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
#include <ftw.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <unistd.h>

#include "lua_state.h"
#include "remove_pkg.h"
#include "vars.h"

static int remove_installed(
  const char *src_path, const struct stat *sb,
  int typeflag, struct FTW *ftwbuf
) {
  (void)sb;
  (void)ftwbuf;

  if (typeflag == FTW_F) {
    const char *filename = src_path + ftwbuf->base;
    const char *ext = strrchr(filename, '.');
    if (!ext)
      ext = "";

    if (strncmp(ext, ".so", 3) == 0) {
      char dest[MAX_PATH_LEN];
      snprintf(dest, sizeof(dest), "%s/%s", get_install_dir("lib"), filename);
      if (file_exists(dest))
        remove(dest);
    } else if (access(src_path, X_OK) == 0) {
      if (strcmp(ext, ".sample") != 0 && strcmp(filename, "bldit") != 0 &&
          strcmp(filename, "build.sh") != 0 &&
          strcmp(filename, "compile.sh") != 0) {
        char dest[MAX_PATH_LEN];
        snprintf(dest, sizeof(dest), "%s/%s", get_install_dir("bin"), filename);
        if (file_exists(dest))
          remove(dest);
      }
    } else if (strcmp(ext, ".h") == 0) {
      char dest[MAX_PATH_LEN];
      snprintf(dest, sizeof(dest), "%s/%s", get_install_dir("include"),
               filename);
      if (file_exists(dest))
        remove(dest);
    }
  }

  return 0;
}

static int remove_tree(const char *fpath, const struct stat *sb, int typeflag,
                       struct FTW *ftwbuf) {
  (void)sb;
  (void)ftwbuf;
  if (typeflag == FTW_F || typeflag == FTW_SL) {
    unlink(fpath);
  } else if (typeflag == FTW_DP) {
    rmdir(fpath);
  }
  return 0;
}

void remove_pkg(Pkg pkg) {
  if (!file_exists(pkg.src)) {
    printf("%s %s is not installed!\n", print_pkgit, pkg.name);
    return;
  }
  chdir(pkg.src);

  bool uninstall_available = false;
  if (!uninstall_available) if (repo_uninstall(pkg.name, pkg.target)) 
    uninstall_available = true;
  if (!uninstall_available) if (bldit_uninstall(pkg.target))
    uninstall_available = true;
  if (!uninstall_available) if (config_uninstall(pkg.src, pkg.target))
    uninstall_available = true;

  if (!uninstall_available) printf(
    "%s no uninstall function availible for package: %s\n",
    print_error, pkg.name
  );

  nftw(pkg.src, remove_installed, 64, FTW_PHYS);
  const char *last_slash = strrchr(pkg.src, '/');
  char target[MAX_PATH_LEN];
  if (last_slash && last_slash != pkg.src) {
    size_t parent_len = last_slash - pkg.src;
    snprintf(target, sizeof(target), "%.*s", (int)parent_len, pkg.src);
  } else {
    snprintf(target, sizeof(target), "%s", pkg.src);
  }
  nftw(pkg.src, remove_tree, 64, FTW_DEPTH | FTW_PHYS);
  printf("%s removed %s\n", print_success, pkg.name);
}