#include #include #include #ifdef ERR #include #endif #ifdef GIT #include #include #include #endif #define P(X) fwrite(X, 1, 1, stdout) #ifdef ERR void l(const char *fmt, ...) { va_list ap; va_start(ap, fmt); vfprintf(stderr, fmt, ap); va_end(ap); if (fmt[0] && fmt[strlen(fmt) - 1] == ':') { fputc(' ', stderr); perror(NULL); } else { fputc('\n', stderr); } } #define L(...) l(__VA_ARGS__) #else #define L(...) #endif #ifdef GIT /** * @brief search all parent directories for a git repo * * @return absolute path to git repo */ char *find_git_repo() { char path[PATH_MAX] = ".", *rpath; struct stat s; int i, c; /* find the number of jumps to the root of the fs */ rpath = realpath(path, NULL); for (i = c = 0; i < strlen(rpath); i++) { if (rpath[i] == '/') { c++; } } free(rpath); /* start searching */ for (i = c; i > 0; i--) { strcat(path, "/.git"); /* if there seems to be a git directory return the directory it was found in */ if (stat(path, &s) == 0 && S_ISDIR(s.st_mode)) { return realpath(path, NULL); } /* reset contents of gpath, and go up a directory */ memset(&path[strlen(path) - 4], '.', 2); memset(&path[strlen(path) - 2], 0, 2); } return NULL; } /** * @brief open git repo if one is available at the current path * * @return a pointer to the git repo object */ git_repository *init_git() { char *buf; git_repository *repo; /* check for a repo before loading libgit2 */ if ((buf = find_git_repo()) == NULL) { return NULL; } /* disable a bunch of git options to hopefully speed things up */ git_libgit2_opts(GIT_OPT_ENABLE_CACHING, 0); git_libgit2_opts(GIT_OPT_SET_SEARCH_PATH, GIT_CONFIG_LEVEL_GLOBAL, ""); git_libgit2_opts(GIT_OPT_SET_SEARCH_PATH, GIT_CONFIG_LEVEL_XDG, ""); git_libgit2_opts(GIT_OPT_SET_SEARCH_PATH, GIT_CONFIG_LEVEL_SYSTEM, ""); git_libgit2_opts(GIT_OPT_SET_TEMPLATE_PATH, ""); git_libgit2_opts(GIT_OPT_DISABLE_PACK_KEEP_FILE_CHECKS, 1); /* initialize the git library and repository */ if (git_libgit2_init() < 0) { L("Failed to initalize libgit2, proceeding without git functionality enabled."); return NULL; } if (git_repository_open(&repo, buf) < 0) { L("Failed to open git repo: %s", git_error_last()->message); free(buf); return NULL; } /* get rid of object containing git repo path and return the repo */ free(buf); return repo; } /** * @brief check for any existing stashes in the provided git repo * * @param repo git repo to check for existing stashes * @return 1 if stashes found 0 otherwise */ int has_stashes(git_repository *repo) { git_reference *stash = NULL; int e; e = git_reference_lookup(&stash, repo, "refs/stash"); if (e == GIT_ENOTFOUND) { return 0; } else if (e < 0) { L("Error looking up stash reference: %s", git_error_last()->message); return 0; } else { e = 1; } git_reference_free(stash); return e; } /** * @brief check for any untracked changes in the current git repo * * @param repo git repository object * @return 1 if any untracked changes have been found 0 otherwise */ int has_untracked(git_repository *repo) { git_status_options opts = GIT_STATUS_OPTIONS_INIT; git_status_list *list = NULL; int r = 0; opts.show = GIT_STATUS_SHOW_INDEX_AND_WORKDIR; opts.flags = GIT_STATUS_OPT_INCLUDE_UNTRACKED | GIT_STATUS_OPT_RENAMES_HEAD_TO_INDEX; if (git_status_list_new(&list, repo, &opts) < 0) { L("Error checking for untracked changes: %s", git_error_last()->message); return 0; } /* if any changes are found return 1 */ if (git_status_list_entrycount(list) > 0) { r = 1; } git_status_list_free(list); return r; } /** * @brief check for staged changes * * @param repo git repository object * @return 1 if any staged changes found 0 otherwise */ int has_staged(git_repository *repo) { git_status_entry entry; git_status_list *list = NULL; git_status_options opts = GIT_STATUS_OPTIONS_INIT; int i, c, r = 0; opts.show = GIT_STATUS_SHOW_INDEX_ONLY; opts.flags = GIT_STATUS_INDEX_NEW; if (git_status_list_new(&list, repo, &opts) < 0) { L("Error checking for staged changes: %s", git_error_last()->message); return 0; } /* if any staged changes are found return 1 */ if ((c = git_status_list_entrycount(list)) > 0) { for (i = 0; i < c; i++) { entry = *git_status_byindex(list, i); if (entry.status & (GIT_STATUS_INDEX_NEW | GIT_STATUS_INDEX_DELETED | GIT_STATUS_INDEX_MODIFIED)) { r = 1; break; } } } git_status_list_free(list); return r; } #endif int main(int argc, char *argv[]) { int code = -1; /* print version information */ if (argc > 1 && strcmp(argv[1], "-v") == 0) { printf("XD [number] v%s\n", VERSION); return 0; } #ifdef GIT git_repository *repo; if ((repo = init_git())) { /* change the eyes depending on the current git repo's status */ if (has_stashes(repo)) { P("8"); /* goggle eyes if we have some stashed changes */ } else if (git_repository_is_empty(repo)) { P("B"); /* sunglasses if we're in a new repo with no HEAD */ } else { P(";"); /* wink when we're in a git repo */ } /* change the nose depending on the current git repo's status */ if (has_staged(repo)) { P("*"); /* change to broken nose for staged changes */ } else if (has_untracked(repo)) { P("^"); /* add a little nose when there are untracked changes in the repo */ } else if (git_repository_head_detached(repo)) { P("-"); /* add a minus nose when the HEAD is detached */ } git_repository_free(repo); git_libgit2_shutdown(); } else #endif if (1) { P(":"); } /* get exit code from user args */ if (argv[1]) { code = atoi(argv[1]); } /* change mouth based on exit code */ if (code >= 0) { switch (code) { case 0: P(")"); break; /* all good */ case 130: P("O"); break; /* Ctrl-c pressed (SIGTERM) */ case 126: P("P"); break; /* permission denied */ case 127: P("/"); break; /* command not found */ default: P("("); break; /* all other codes (usually the program saying it has failed) */ } } else { P("|"); /* no code info */ } }