|  | /* | 
|  | * libgit2 "remote" example - shows how to modify remotes for a repo | 
|  | * | 
|  | * 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 | 
|  | * <http://creativecommons.org/publicdomain/zero/1.0/>. | 
|  | */ | 
|  |  | 
|  | #include "common.h" | 
|  |  | 
|  | /** | 
|  | * This is a sample program that is similar to "git remote".  See the | 
|  | * documentation for that (try "git help remote") to understand what this | 
|  | * program is emulating. | 
|  | * | 
|  | * This demonstrates using the libgit2 APIs to modify remotes of a repository. | 
|  | */ | 
|  |  | 
|  | enum subcmd { | 
|  | subcmd_add, | 
|  | subcmd_remove, | 
|  | subcmd_rename, | 
|  | subcmd_seturl, | 
|  | subcmd_show, | 
|  | }; | 
|  |  | 
|  | struct opts { | 
|  | enum subcmd cmd; | 
|  |  | 
|  | /* for command-specific args */ | 
|  | int argc; | 
|  | char **argv; | 
|  | }; | 
|  |  | 
|  | static int cmd_add(git_repository *repo, struct opts *o); | 
|  | static int cmd_remove(git_repository *repo, struct opts *o); | 
|  | static int cmd_rename(git_repository *repo, struct opts *o); | 
|  | static int cmd_seturl(git_repository *repo, struct opts *o); | 
|  | static int cmd_show(git_repository *repo, struct opts *o); | 
|  |  | 
|  | static void parse_subcmd( | 
|  | struct opts *opt, int argc, char **argv); | 
|  | static void usage(const char *msg, const char *arg); | 
|  |  | 
|  | int main(int argc, char *argv[]) | 
|  | { | 
|  | int retval = 0; | 
|  | struct opts opt = {0}; | 
|  | git_buf buf = GIT_BUF_INIT_CONST(NULL, 0); | 
|  | git_repository *repo = NULL; | 
|  |  | 
|  | parse_subcmd(&opt, argc, argv); | 
|  |  | 
|  | git_libgit2_init(); | 
|  |  | 
|  | check_lg2(git_repository_discover(&buf, ".", 0, NULL), | 
|  | "Could not find repository", NULL); | 
|  |  | 
|  | check_lg2(git_repository_open(&repo, buf.ptr), | 
|  | "Could not open repository", NULL); | 
|  | git_buf_free(&buf); | 
|  |  | 
|  | switch (opt.cmd) | 
|  | { | 
|  | case subcmd_add: | 
|  | retval = cmd_add(repo, &opt); | 
|  | break; | 
|  | case subcmd_remove: | 
|  | retval = cmd_remove(repo, &opt); | 
|  | break; | 
|  | case subcmd_rename: | 
|  | retval = cmd_rename(repo, &opt); | 
|  | break; | 
|  | case subcmd_seturl: | 
|  | retval = cmd_seturl(repo, &opt); | 
|  | break; | 
|  | case subcmd_show: | 
|  | retval = cmd_show(repo, &opt); | 
|  | break; | 
|  | } | 
|  |  | 
|  | git_libgit2_shutdown(); | 
|  |  | 
|  | return retval; | 
|  | } | 
|  |  | 
|  | static int cmd_add(git_repository *repo, struct opts *o) | 
|  | { | 
|  | char *name, *url; | 
|  | git_remote *remote = {0}; | 
|  |  | 
|  | if (o->argc != 2) | 
|  | usage("you need to specify a name and URL", NULL); | 
|  |  | 
|  | name = o->argv[0]; | 
|  | url = o->argv[1]; | 
|  |  | 
|  | check_lg2(git_remote_create(&remote, repo, name, url), | 
|  | "could not create remote", NULL); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_remove(git_repository *repo, struct opts *o) | 
|  | { | 
|  | char *name; | 
|  |  | 
|  | if (o->argc != 1) | 
|  | usage("you need to specify a name", NULL); | 
|  |  | 
|  | name = o->argv[0]; | 
|  |  | 
|  | check_lg2(git_remote_delete(repo, name), | 
|  | "could not delete remote", name); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_rename(git_repository *repo, struct opts *o) | 
|  | { | 
|  | int i, retval; | 
|  | char *old, *new; | 
|  | git_strarray problems = {0}; | 
|  |  | 
|  | if (o->argc != 2) | 
|  | usage("you need to specify old and new remote name", NULL); | 
|  |  | 
|  | old = o->argv[0]; | 
|  | new = o->argv[1]; | 
|  |  | 
|  | retval = git_remote_rename(&problems, repo, old, new); | 
|  | if (!retval) | 
|  | return 0; | 
|  |  | 
|  | for (i = 0; i < (int) problems.count; i++) { | 
|  | puts(problems.strings[0]); | 
|  | } | 
|  |  | 
|  | git_strarray_free(&problems); | 
|  |  | 
|  | return retval; | 
|  | } | 
|  |  | 
|  | static int cmd_seturl(git_repository *repo, struct opts *o) | 
|  | { | 
|  | int i, retval, push = 0; | 
|  | char *name = NULL, *url = NULL; | 
|  |  | 
|  | for (i = 0; i < o->argc; i++) { | 
|  | char *arg = o->argv[i]; | 
|  |  | 
|  | if (!strcmp(arg, "--push")) { | 
|  | push = 1; | 
|  | } else if (arg[0] != '-' && name == NULL) { | 
|  | name = arg; | 
|  | } else if (arg[0] != '-' && url == NULL) { | 
|  | url = arg; | 
|  | } else { | 
|  | usage("invalid argument to set-url", arg); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (name == NULL || url == NULL) | 
|  | usage("you need to specify remote and the new URL", NULL); | 
|  |  | 
|  | if (push) | 
|  | retval = git_remote_set_pushurl(repo, name, url); | 
|  | else | 
|  | retval = git_remote_set_url(repo, name, url); | 
|  |  | 
|  | check_lg2(retval, "could not set URL", url); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int cmd_show(git_repository *repo, struct opts *o) | 
|  | { | 
|  | int i; | 
|  | const char *arg, *name, *fetch, *push; | 
|  | int verbose = 0; | 
|  | git_strarray remotes = {0}; | 
|  | git_remote *remote = {0}; | 
|  |  | 
|  | for (i = 0; i < o->argc; i++) { | 
|  | arg = o->argv[i]; | 
|  |  | 
|  | if (!strcmp(arg, "-v") || !strcmp(arg, "--verbose")) { | 
|  | verbose = 1; | 
|  | } | 
|  | } | 
|  |  | 
|  | check_lg2(git_remote_list(&remotes, repo), | 
|  | "could not retrieve remotes", NULL); | 
|  |  | 
|  | for (i = 0; i < (int) remotes.count; i++) { | 
|  | name = remotes.strings[i]; | 
|  | if (!verbose) { | 
|  | puts(name); | 
|  | continue; | 
|  | } | 
|  |  | 
|  | check_lg2(git_remote_lookup(&remote, repo, name), | 
|  | "could not look up remote", name); | 
|  |  | 
|  | fetch = git_remote_url(remote); | 
|  | if (fetch) | 
|  | printf("%s\t%s (fetch)\n", name, fetch); | 
|  | push = git_remote_pushurl(remote); | 
|  | /* use fetch URL if no distinct push URL has been set */ | 
|  | push = push ? push : fetch; | 
|  | if (push) | 
|  | printf("%s\t%s (push)\n", name, push); | 
|  |  | 
|  | git_remote_free(remote); | 
|  | } | 
|  |  | 
|  | git_strarray_free(&remotes); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void parse_subcmd( | 
|  | struct opts *opt, int argc, char **argv) | 
|  | { | 
|  | char *arg = argv[1]; | 
|  | enum subcmd cmd = 0; | 
|  |  | 
|  | if (argc < 2) | 
|  | usage("no command specified", NULL); | 
|  |  | 
|  | if (!strcmp(arg, "add")) { | 
|  | cmd = subcmd_add; | 
|  | } else if (!strcmp(arg, "remove")) { | 
|  | cmd = subcmd_remove; | 
|  | } else if (!strcmp(arg, "rename")) { | 
|  | cmd = subcmd_rename; | 
|  | } else if (!strcmp(arg, "set-url")) { | 
|  | cmd = subcmd_seturl; | 
|  | } else if (!strcmp(arg, "show")) { | 
|  | cmd = subcmd_show; | 
|  | } else { | 
|  | usage("command is not valid", arg); | 
|  | } | 
|  | opt->cmd = cmd; | 
|  |  | 
|  | opt->argc = argc - 2; /* executable and subcommand are removed */ | 
|  | opt->argv = argv + 2; | 
|  | } | 
|  |  | 
|  | static void usage(const char *msg, const char *arg) | 
|  | { | 
|  | fputs("usage: remote add <name> <url>\n", stderr); | 
|  | fputs("       remote remove <name>\n", stderr); | 
|  | fputs("       remote rename <old> <new>\n", stderr); | 
|  | fputs("       remote set-url [--push] <name> <newurl>\n", stderr); | 
|  | fputs("       remote show [-v|--verbose]\n", stderr); | 
|  |  | 
|  | if (msg && !arg) | 
|  | fprintf(stderr, "\n%s\n", msg); | 
|  | else if (msg && arg) | 
|  | fprintf(stderr, "\n%s: %s\n", msg, arg); | 
|  | exit(1); | 
|  | } |