apply: demonstrate a problem applying svn diffs
[git] / repository.c
1 #include "cache.h"
2 #include "repository.h"
3 #include "config.h"
4 #include "submodule-config.h"
5
6 /* The main repository */
7 static struct repository the_repo;
8 struct repository *the_repository = &the_repo;
9
10 static char *git_path_from_env(const char *envvar, const char *git_dir,
11                                const char *path, int fromenv)
12 {
13         if (fromenv) {
14                 const char *value = getenv(envvar);
15                 if (value)
16                         return xstrdup(value);
17         }
18
19         return xstrfmt("%s/%s", git_dir, path);
20 }
21
22 static int find_common_dir(struct strbuf *sb, const char *gitdir, int fromenv)
23 {
24         if (fromenv) {
25                 const char *value = getenv(GIT_COMMON_DIR_ENVIRONMENT);
26                 if (value) {
27                         strbuf_addstr(sb, value);
28                         return 1;
29                 }
30         }
31
32         return get_common_dir_noenv(sb, gitdir);
33 }
34
35 static void repo_setup_env(struct repository *repo)
36 {
37         struct strbuf sb = STRBUF_INIT;
38
39         repo->different_commondir = find_common_dir(&sb, repo->gitdir,
40                                                     !repo->ignore_env);
41         repo->commondir = strbuf_detach(&sb, NULL);
42         repo->objectdir = git_path_from_env(DB_ENVIRONMENT, repo->commondir,
43                                             "objects", !repo->ignore_env);
44         repo->graft_file = git_path_from_env(GRAFT_ENVIRONMENT, repo->commondir,
45                                              "info/grafts", !repo->ignore_env);
46         repo->index_file = git_path_from_env(INDEX_ENVIRONMENT, repo->gitdir,
47                                              "index", !repo->ignore_env);
48 }
49
50 void repo_set_gitdir(struct repository *repo, const char *path)
51 {
52         const char *gitfile = read_gitfile(path);
53
54         /*
55          * NEEDSWORK: Eventually we want to be able to free gitdir and the rest
56          * of the environment before reinitializing it again, but we have some
57          * crazy code paths where we try to set gitdir with the current gitdir
58          * and we don't want to free gitdir before copying the passed in value.
59          */
60         repo->gitdir = xstrdup(gitfile ? gitfile : path);
61
62         repo_setup_env(repo);
63 }
64
65 /*
66  * Attempt to resolve and set the provided 'gitdir' for repository 'repo'.
67  * Return 0 upon success and a non-zero value upon failure.
68  */
69 static int repo_init_gitdir(struct repository *repo, const char *gitdir)
70 {
71         int ret = 0;
72         int error = 0;
73         char *abspath = NULL;
74         const char *resolved_gitdir;
75
76         abspath = real_pathdup(gitdir, 0);
77         if (!abspath) {
78                 ret = -1;
79                 goto out;
80         }
81
82         /* 'gitdir' must reference the gitdir directly */
83         resolved_gitdir = resolve_gitdir_gently(abspath, &error);
84         if (!resolved_gitdir) {
85                 ret = -1;
86                 goto out;
87         }
88
89         repo_set_gitdir(repo, resolved_gitdir);
90
91 out:
92         free(abspath);
93         return ret;
94 }
95
96 void repo_set_worktree(struct repository *repo, const char *path)
97 {
98         repo->worktree = real_pathdup(path, 1);
99 }
100
101 static int read_and_verify_repository_format(struct repository_format *format,
102                                              const char *commondir)
103 {
104         int ret = 0;
105         struct strbuf sb = STRBUF_INIT;
106
107         strbuf_addf(&sb, "%s/config", commondir);
108         read_repository_format(format, sb.buf);
109         strbuf_reset(&sb);
110
111         if (verify_repository_format(format, &sb) < 0) {
112                 warning("%s", sb.buf);
113                 ret = -1;
114         }
115
116         strbuf_release(&sb);
117         return ret;
118 }
119
120 /*
121  * Initialize 'repo' based on the provided 'gitdir'.
122  * Return 0 upon success and a non-zero value upon failure.
123  */
124 int repo_init(struct repository *repo, const char *gitdir, const char *worktree)
125 {
126         struct repository_format format;
127         memset(repo, 0, sizeof(*repo));
128
129         repo->ignore_env = 1;
130
131         if (repo_init_gitdir(repo, gitdir))
132                 goto error;
133
134         if (read_and_verify_repository_format(&format, repo->commondir))
135                 goto error;
136
137         if (worktree)
138                 repo_set_worktree(repo, worktree);
139
140         return 0;
141
142 error:
143         repo_clear(repo);
144         return -1;
145 }
146
147 /*
148  * Initialize 'submodule' as the submodule given by 'path' in parent repository
149  * 'superproject'.
150  * Return 0 upon success and a non-zero value upon failure.
151  */
152 int repo_submodule_init(struct repository *submodule,
153                         struct repository *superproject,
154                         const char *path)
155 {
156         const struct submodule *sub;
157         struct strbuf gitdir = STRBUF_INIT;
158         struct strbuf worktree = STRBUF_INIT;
159         int ret = 0;
160
161         sub = submodule_from_cache(superproject, null_sha1, path);
162         if (!sub) {
163                 ret = -1;
164                 goto out;
165         }
166
167         strbuf_repo_worktree_path(&gitdir, superproject, "%s/.git", path);
168         strbuf_repo_worktree_path(&worktree, superproject, "%s", path);
169
170         if (repo_init(submodule, gitdir.buf, worktree.buf)) {
171                 /*
172                  * If initilization fails then it may be due to the submodule
173                  * not being populated in the superproject's worktree.  Instead
174                  * we can try to initilize the submodule by finding it's gitdir
175                  * in the superproject's 'modules' directory.  In this case the
176                  * submodule would not have a worktree.
177                  */
178                 strbuf_reset(&gitdir);
179                 strbuf_repo_git_path(&gitdir, superproject,
180                                      "modules/%s", sub->name);
181
182                 if (repo_init(submodule, gitdir.buf, NULL)) {
183                         ret = -1;
184                         goto out;
185                 }
186         }
187
188         submodule->submodule_prefix = xstrfmt("%s%s/",
189                                               superproject->submodule_prefix ?
190                                               superproject->submodule_prefix :
191                                               "", path);
192
193 out:
194         strbuf_release(&gitdir);
195         strbuf_release(&worktree);
196         return ret;
197 }
198
199 void repo_clear(struct repository *repo)
200 {
201         free(repo->gitdir);
202         repo->gitdir = NULL;
203         free(repo->commondir);
204         repo->commondir = NULL;
205         free(repo->objectdir);
206         repo->objectdir = NULL;
207         free(repo->graft_file);
208         repo->graft_file = NULL;
209         free(repo->index_file);
210         repo->index_file = NULL;
211         free(repo->worktree);
212         repo->worktree = NULL;
213         free(repo->submodule_prefix);
214         repo->submodule_prefix = NULL;
215
216         if (repo->config) {
217                 git_configset_clear(repo->config);
218                 free(repo->config);
219                 repo->config = NULL;
220         }
221
222         if (repo->submodule_cache) {
223                 submodule_cache_free(repo->submodule_cache);
224                 repo->submodule_cache = NULL;
225         }
226
227         if (repo->index) {
228                 discard_index(repo->index);
229                 free(repo->index);
230                 repo->index = NULL;
231         }
232 }
233
234 int repo_read_index(struct repository *repo)
235 {
236         if (!repo->index)
237                 repo->index = xcalloc(1, sizeof(*repo->index));
238         else
239                 discard_index(repo->index);
240
241         return read_index_from(repo->index, repo->index_file);
242 }