Merge branch 'jc/maint-github-actions-update'
[git.git] / builtin / commit-tree.c
blobcc8d584be2f4a9d3b0c950a4b8cbc6c140904d8c
1 /*
2 * GIT - The information manager from hell
4 * Copyright (C) Linus Torvalds, 2005
5 */
6 #include "cache.h"
7 #include "config.h"
8 #include "object-store.h"
9 #include "repository.h"
10 #include "commit.h"
11 #include "tree.h"
12 #include "builtin.h"
13 #include "utf8.h"
14 #include "gpg-interface.h"
15 #include "parse-options.h"
17 static const char * const commit_tree_usage[] = {
18 N_("git commit-tree <tree> [(-p <parent>)...]"),
19 N_("git commit-tree [(-p <parent>)...] [-S[<keyid>]] [(-m <message>)...]\n"
20 " [(-F <file>)...] <tree>"),
21 NULL
24 static const char *sign_commit;
26 static void new_parent(struct commit *parent, struct commit_list **parents_p)
28 struct object_id *oid = &parent->object.oid;
29 struct commit_list *parents;
30 for (parents = *parents_p; parents; parents = parents->next) {
31 if (parents->item == parent) {
32 error(_("duplicate parent %s ignored"), oid_to_hex(oid));
33 return;
35 parents_p = &parents->next;
37 commit_list_insert(parent, parents_p);
40 static int commit_tree_config(const char *var, const char *value, void *cb)
42 int status = git_gpg_config(var, value, NULL);
43 if (status)
44 return status;
45 return git_default_config(var, value, cb);
48 static int parse_parent_arg_callback(const struct option *opt,
49 const char *arg, int unset)
51 struct object_id oid;
52 struct commit_list **parents = opt->value;
54 BUG_ON_OPT_NEG_NOARG(unset, arg);
56 if (get_oid_commit(arg, &oid))
57 die(_("not a valid object name %s"), arg);
59 assert_oid_type(&oid, OBJ_COMMIT);
60 new_parent(lookup_commit(the_repository, &oid), parents);
61 return 0;
64 static int parse_message_arg_callback(const struct option *opt,
65 const char *arg, int unset)
67 struct strbuf *buf = opt->value;
69 BUG_ON_OPT_NEG_NOARG(unset, arg);
71 if (buf->len)
72 strbuf_addch(buf, '\n');
73 strbuf_addstr(buf, arg);
74 strbuf_complete_line(buf);
76 return 0;
79 static int parse_file_arg_callback(const struct option *opt,
80 const char *arg, int unset)
82 int fd;
83 struct strbuf *buf = opt->value;
85 BUG_ON_OPT_NEG_NOARG(unset, arg);
87 if (buf->len)
88 strbuf_addch(buf, '\n');
89 if (!strcmp(arg, "-"))
90 fd = 0;
91 else {
92 fd = xopen(arg, O_RDONLY);
94 if (strbuf_read(buf, fd, 0) < 0)
95 die_errno(_("git commit-tree: failed to read '%s'"), arg);
96 if (fd && close(fd))
97 die_errno(_("git commit-tree: failed to close '%s'"), arg);
99 return 0;
102 int cmd_commit_tree(int argc, const char **argv, const char *prefix)
104 static struct strbuf buffer = STRBUF_INIT;
105 struct commit_list *parents = NULL;
106 struct object_id tree_oid;
107 struct object_id commit_oid;
109 struct option options[] = {
110 OPT_CALLBACK_F('p', NULL, &parents, N_("parent"),
111 N_("id of a parent commit object"), PARSE_OPT_NONEG,
112 parse_parent_arg_callback),
113 OPT_CALLBACK_F('m', NULL, &buffer, N_("message"),
114 N_("commit message"), PARSE_OPT_NONEG,
115 parse_message_arg_callback),
116 OPT_CALLBACK_F('F', NULL, &buffer, N_("file"),
117 N_("read commit log message from file"), PARSE_OPT_NONEG,
118 parse_file_arg_callback),
119 { OPTION_STRING, 'S', "gpg-sign", &sign_commit, N_("key-id"),
120 N_("GPG sign commit"), PARSE_OPT_OPTARG, NULL, (intptr_t) "" },
121 OPT_END()
124 git_config(commit_tree_config, NULL);
126 if (argc < 2 || !strcmp(argv[1], "-h"))
127 usage_with_options(commit_tree_usage, options);
129 argc = parse_options(argc, argv, prefix, options, commit_tree_usage, 0);
131 if (argc != 1)
132 die(_("must give exactly one tree"));
134 if (get_oid_tree(argv[0], &tree_oid))
135 die(_("not a valid object name %s"), argv[0]);
137 if (!buffer.len) {
138 if (strbuf_read(&buffer, 0, 0) < 0)
139 die_errno(_("git commit-tree: failed to read"));
142 if (commit_tree(buffer.buf, buffer.len, &tree_oid, parents, &commit_oid,
143 NULL, sign_commit)) {
144 strbuf_release(&buffer);
145 return 1;
148 printf("%s\n", oid_to_hex(&commit_oid));
149 strbuf_release(&buffer);
150 return 0;