/* * libgit2 "add" example - shows how to modify the index * * Written by the libgit2 contributors * * To the extent possible under law, the author(s) have dedicated all copyright * and related and neighboring rights to this software to the public domain * worldwide. This software is distributed without any warranty. * * You should have received a copy of the CC0 Public Domain Dedication along * with this software. If not, see * . */ #include "common.h" /** * The following example demonstrates how to add files with libgit2. * * It will use the repository in the current working directory, and act * on files passed as its parameters. * * Recognized options are: * -v/--verbose: show the file's status after acting on it. * -n/--dry-run: do not actually change the index. * -u/--update: update the index instead of adding to it. */ enum index_mode { INDEX_NONE, INDEX_ADD }; struct index_options { int dry_run; int verbose; git_repository *repo; enum index_mode mode; int add_update; }; /* Forward declarations for helpers */ static void parse_opts(const char **repo_path, struct index_options *opts, struct args_info *args); int print_matched_cb(const char *path, const char *matched_pathspec, void *payload); int lg2_add(git_repository *repo, int argc, char **argv) { git_index_matched_path_cb matched_cb = NULL; git_index *index; git_strarray array = {0}; struct index_options options = {0}; struct args_info args = ARGS_INFO_INIT; options.mode = INDEX_ADD; /* Parse the options & arguments. */ parse_opts(NULL, &options, &args); strarray_from_args(&array, &args); /* Grab the repository's index. */ check_lg2(git_repository_index(&index, repo), "Could not open repository index", NULL); /* Setup a callback if the requested options need it */ if (options.verbose || options.dry_run) { matched_cb = &print_matched_cb; } options.repo = repo; /* Perform the requested action with the index and files */ if (options.add_update) { git_index_update_all(index, &array, matched_cb, &options); } else { git_index_add_all(index, &array, 0, matched_cb, &options); } /* Cleanup memory */ git_index_write(index); git_index_free(index); return 0; } /* * This callback is called for each file under consideration by * git_index_(update|add)_all above. * It makes uses of the callback's ability to abort the action. */ int print_matched_cb(const char *path, const char *matched_pathspec, void *payload) { struct index_options *opts = (struct index_options *)(payload); int ret; unsigned status; (void)matched_pathspec; /* Get the file status */ if (git_status_file(&status, opts->repo, path) < 0) return -1; if ((status & GIT_STATUS_WT_MODIFIED) || (status & GIT_STATUS_WT_NEW)) { printf("add '%s'\n", path); ret = 0; } else { ret = 1; } if (opts->dry_run) ret = 1; return ret; } static void print_usage(void) { fprintf(stderr, "usage: add [options] [--] file-spec [file-spec] [...]\n\n"); fprintf(stderr, "\t-n, --dry-run dry run\n"); fprintf(stderr, "\t-v, --verbose be verbose\n"); fprintf(stderr, "\t-u, --update update tracked files\n"); exit(1); } static void parse_opts(const char **repo_path, struct index_options *opts, struct args_info *args) { if (args->argc <= 1) print_usage(); for (args->pos = 1; args->pos < args->argc; ++args->pos) { const char *curr = args->argv[args->pos]; if (curr[0] != '-') { if (!strcmp("add", curr)) { opts->mode = INDEX_ADD; continue; } else if (opts->mode == INDEX_NONE) { fprintf(stderr, "missing command: %s", curr); print_usage(); break; } else { /* We might be looking at a filename */ break; } } else if (match_bool_arg(&opts->verbose, args, "--verbose") || match_bool_arg(&opts->dry_run, args, "--dry-run") || match_str_arg(repo_path, args, "--git-dir") || (opts->mode == INDEX_ADD && match_bool_arg(&opts->add_update, args, "--update"))) { continue; } else if (match_bool_arg(NULL, args, "--help")) { print_usage(); break; } else if (match_arg_separator(args)) { break; } else { fprintf(stderr, "Unsupported option %s.\n", curr); print_usage(); } } }