Linus Torvalds | 8bc9a0c | 2005-04-07 15:16:10 -0700 | [diff] [blame] | 1 | /* |
| 2 | * GIT - The information manager from hell |
| 3 | * |
| 4 | * Copyright (C) Linus Torvalds, 2005 |
| 5 | */ |
Linus Torvalds | e83c516 | 2005-04-07 15:13:13 -0700 | [diff] [blame] | 6 | #include "cache.h" |
Timo Hirvonen | c3c8835 | 2006-05-19 13:03:57 +0300 | [diff] [blame] | 7 | #include "builtin.h" |
Johannes Sixt | a47d181 | 2007-11-13 21:05:04 +0100 | [diff] [blame] | 8 | #include "exec_cmd.h" |
Linus Torvalds | e83c516 | 2005-04-07 15:13:13 -0700 | [diff] [blame] | 9 | |
Junio C Hamano | d3af621 | 2005-08-06 12:50:14 -0700 | [diff] [blame] | 10 | #ifndef DEFAULT_GIT_TEMPLATE_DIR |
Johannes Sixt | d52fd42 | 2007-06-11 11:10:47 +0200 | [diff] [blame] | 11 | #define DEFAULT_GIT_TEMPLATE_DIR "/usr/share/git-core/templates" |
Junio C Hamano | d3af621 | 2005-08-06 12:50:14 -0700 | [diff] [blame] | 12 | #endif |
| 13 | |
Shawn O. Pearce | c869753 | 2006-12-30 23:53:55 -0500 | [diff] [blame] | 14 | #ifdef NO_TRUSTABLE_FILEMODE |
| 15 | #define TEST_FILEMODE 0 |
| 16 | #else |
| 17 | #define TEST_FILEMODE 1 |
| 18 | #endif |
| 19 | |
Deskin Miller | 0a2c7ee | 2008-10-07 01:37:48 -0400 | [diff] [blame] | 20 | static int init_is_bare_repository = 0; |
| 21 | static int init_shared_repository = -1; |
| 22 | |
Johannes Schindelin | af6e277 | 2005-12-22 23:19:37 +0100 | [diff] [blame] | 23 | static void safe_create_dir(const char *dir, int share) |
Zach Welch | cb126d8 | 2005-04-19 21:48:15 -0700 | [diff] [blame] | 24 | { |
Junio C Hamano | f312de0 | 2005-07-06 01:21:46 -0700 | [diff] [blame] | 25 | if (mkdir(dir, 0777) < 0) { |
Zach Welch | cb126d8 | 2005-04-19 21:48:15 -0700 | [diff] [blame] | 26 | if (errno != EEXIST) { |
| 27 | perror(dir); |
| 28 | exit(1); |
| 29 | } |
| 30 | } |
Johannes Schindelin | af6e277 | 2005-12-22 23:19:37 +0100 | [diff] [blame] | 31 | else if (share && adjust_shared_perm(dir)) |
Alexander Potashev | d753070 | 2009-01-04 21:38:41 +0300 | [diff] [blame] | 32 | die("Could not make %s writable by group", dir); |
Zach Welch | cb126d8 | 2005-04-19 21:48:15 -0700 | [diff] [blame] | 33 | } |
| 34 | |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 35 | static void copy_templates_1(char *path, int baselen, |
| 36 | char *template, int template_baselen, |
| 37 | DIR *dir) |
| 38 | { |
| 39 | struct dirent *de; |
| 40 | |
| 41 | /* Note: if ".git/hooks" file exists in the repository being |
| 42 | * re-initialized, /etc/core-git/templates/hooks/update would |
Heikki Orsila | f18d244 | 2008-09-13 20:18:36 +0300 | [diff] [blame] | 43 | * cause "git init" to fail here. I think this is sane but |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 44 | * it means that the set of templates we ship by default, along |
| 45 | * with the way the namespace under .git/ is organized, should |
| 46 | * be really carefully chosen. |
| 47 | */ |
Johannes Schindelin | af6e277 | 2005-12-22 23:19:37 +0100 | [diff] [blame] | 48 | safe_create_dir(path, 1); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 49 | while ((de = readdir(dir)) != NULL) { |
| 50 | struct stat st_git, st_template; |
| 51 | int namelen; |
| 52 | int exists = 0; |
| 53 | |
| 54 | if (de->d_name[0] == '.') |
| 55 | continue; |
| 56 | namelen = strlen(de->d_name); |
| 57 | if ((PATH_MAX <= baselen + namelen) || |
| 58 | (PATH_MAX <= template_baselen + namelen)) |
| 59 | die("insanely long template name %s", de->d_name); |
| 60 | memcpy(path + baselen, de->d_name, namelen+1); |
| 61 | memcpy(template + template_baselen, de->d_name, namelen+1); |
| 62 | if (lstat(path, &st_git)) { |
| 63 | if (errno != ENOENT) |
Thomas Rast | 0721c31 | 2009-06-27 17:58:47 +0200 | [diff] [blame] | 64 | die_errno("cannot stat '%s'", path); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 65 | } |
| 66 | else |
| 67 | exists = 1; |
| 68 | |
| 69 | if (lstat(template, &st_template)) |
Thomas Rast | 0721c31 | 2009-06-27 17:58:47 +0200 | [diff] [blame] | 70 | die_errno("cannot stat template '%s'", template); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 71 | |
| 72 | if (S_ISDIR(st_template.st_mode)) { |
| 73 | DIR *subdir = opendir(template); |
| 74 | int baselen_sub = baselen + namelen; |
| 75 | int template_baselen_sub = template_baselen + namelen; |
| 76 | if (!subdir) |
Thomas Rast | 0721c31 | 2009-06-27 17:58:47 +0200 | [diff] [blame] | 77 | die_errno("cannot opendir '%s'", template); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 78 | path[baselen_sub++] = |
| 79 | template[template_baselen_sub++] = '/'; |
| 80 | path[baselen_sub] = |
| 81 | template[template_baselen_sub] = 0; |
| 82 | copy_templates_1(path, baselen_sub, |
| 83 | template, template_baselen_sub, |
| 84 | subdir); |
| 85 | closedir(subdir); |
| 86 | } |
| 87 | else if (exists) |
| 88 | continue; |
| 89 | else if (S_ISLNK(st_template.st_mode)) { |
| 90 | char lnk[256]; |
| 91 | int len; |
| 92 | len = readlink(template, lnk, sizeof(lnk)); |
| 93 | if (len < 0) |
Thomas Rast | 0721c31 | 2009-06-27 17:58:47 +0200 | [diff] [blame] | 94 | die_errno("cannot readlink '%s'", template); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 95 | if (sizeof(lnk) <= len) |
| 96 | die("insanely long symlink %s", template); |
| 97 | lnk[len] = 0; |
| 98 | if (symlink(lnk, path)) |
Thomas Rast | 0721c31 | 2009-06-27 17:58:47 +0200 | [diff] [blame] | 99 | die_errno("cannot symlink '%s' '%s'", lnk, path); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 100 | } |
| 101 | else if (S_ISREG(st_template.st_mode)) { |
| 102 | if (copy_file(path, template, st_template.st_mode)) |
Thomas Rast | 0721c31 | 2009-06-27 17:58:47 +0200 | [diff] [blame] | 103 | die_errno("cannot copy '%s' to '%s'", template, |
| 104 | path); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 105 | } |
| 106 | else |
| 107 | error("ignoring template %s", template); |
| 108 | } |
| 109 | } |
| 110 | |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 111 | static void copy_templates(const char *template_dir) |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 112 | { |
| 113 | char path[PATH_MAX]; |
| 114 | char template_path[PATH_MAX]; |
Junio C Hamano | d3af621 | 2005-08-06 12:50:14 -0700 | [diff] [blame] | 115 | int template_len; |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 116 | DIR *dir; |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 117 | const char *git_dir = get_git_dir(); |
| 118 | int len = strlen(git_dir); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 119 | |
Johannes Sixt | a47d181 | 2007-11-13 21:05:04 +0100 | [diff] [blame] | 120 | if (!template_dir) |
Junio C Hamano | d4ebc36 | 2006-12-19 01:28:15 -0800 | [diff] [blame] | 121 | template_dir = getenv(TEMPLATE_DIR_ENVIRONMENT); |
Steffen Prohaska | 2de9de5 | 2008-07-13 22:31:18 +0200 | [diff] [blame] | 122 | if (!template_dir) |
| 123 | template_dir = system_path(DEFAULT_GIT_TEMPLATE_DIR); |
Jeff King | 172035f | 2008-07-28 02:02:04 -0400 | [diff] [blame] | 124 | if (!template_dir[0]) |
| 125 | return; |
Frank Lichtenheld | 32d1776 | 2009-04-18 16:14:02 +0200 | [diff] [blame] | 126 | template_len = strlen(template_dir); |
| 127 | if (PATH_MAX <= (template_len+strlen("/config"))) |
| 128 | die("insanely long template path %s", template_dir); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 129 | strcpy(template_path, template_dir); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 130 | if (template_path[template_len-1] != '/') { |
| 131 | template_path[template_len++] = '/'; |
| 132 | template_path[template_len] = 0; |
| 133 | } |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 134 | dir = opendir(template_path); |
Junio C Hamano | d3af621 | 2005-08-06 12:50:14 -0700 | [diff] [blame] | 135 | if (!dir) { |
Miklos Vajna | 2fd8c0a | 2009-03-24 02:09:13 +0100 | [diff] [blame] | 136 | warning("templates not found %s", template_dir); |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 137 | return; |
Junio C Hamano | d3af621 | 2005-08-06 12:50:14 -0700 | [diff] [blame] | 138 | } |
| 139 | |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 140 | /* Make sure that template is from the correct vintage */ |
| 141 | strcpy(template_path + template_len, "config"); |
| 142 | repository_format_version = 0; |
| 143 | git_config_from_file(check_repository_format_version, |
Johannes Schindelin | ef90d6d | 2008-05-14 18:46:53 +0100 | [diff] [blame] | 144 | template_path, NULL); |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 145 | template_path[template_len] = 0; |
| 146 | |
| 147 | if (repository_format_version && |
| 148 | repository_format_version != GIT_REPO_VERSION) { |
Miklos Vajna | 2fd8c0a | 2009-03-24 02:09:13 +0100 | [diff] [blame] | 149 | warning("not copying templates of " |
| 150 | "a wrong format version %d from '%s'", |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 151 | repository_format_version, |
| 152 | template_dir); |
| 153 | closedir(dir); |
| 154 | return; |
| 155 | } |
| 156 | |
Junio C Hamano | d3af621 | 2005-08-06 12:50:14 -0700 | [diff] [blame] | 157 | memcpy(path, git_dir, len); |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 158 | if (len && path[len - 1] != '/') |
| 159 | path[len++] = '/'; |
Petr Baudis | 1f961c1 | 2005-09-20 02:19:50 +0200 | [diff] [blame] | 160 | path[len] = 0; |
Junio C Hamano | 8d5afef | 2005-08-02 16:45:21 -0700 | [diff] [blame] | 161 | copy_templates_1(path, len, |
| 162 | template_path, template_len, |
| 163 | dir); |
| 164 | closedir(dir); |
| 165 | } |
| 166 | |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 167 | static int create_default_files(const char *template_path) |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 168 | { |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 169 | const char *git_dir = get_git_dir(); |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 170 | unsigned len = strlen(git_dir); |
| 171 | static char path[PATH_MAX]; |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 172 | struct stat st1; |
| 173 | char repo_version_string[10]; |
Johannes Schindelin | 5cc8f37 | 2008-03-24 16:14:52 +0100 | [diff] [blame] | 174 | char junk[2]; |
Shawn O. Pearce | ef0a89a | 2006-12-15 00:44:58 -0500 | [diff] [blame] | 175 | int reinit; |
Shawn O. Pearce | c869753 | 2006-12-30 23:53:55 -0500 | [diff] [blame] | 176 | int filemode; |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 177 | |
| 178 | if (len > sizeof(path)-50) |
| 179 | die("insane git directory %s", git_dir); |
| 180 | memcpy(path, git_dir, len); |
| 181 | |
| 182 | if (len && path[len-1] != '/') |
| 183 | path[len++] = '/'; |
| 184 | |
| 185 | /* |
| 186 | * Create .git/refs/{heads,tags} |
| 187 | */ |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 188 | safe_create_dir(git_path("refs"), 1); |
| 189 | safe_create_dir(git_path("refs/heads"), 1); |
| 190 | safe_create_dir(git_path("refs/tags"), 1); |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 191 | |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 192 | /* First copy the templates -- we might have the default |
| 193 | * config file there, in which case we would want to read |
| 194 | * from it after installing. |
| 195 | */ |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 196 | copy_templates(template_path); |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 197 | |
Johannes Schindelin | ef90d6d | 2008-05-14 18:46:53 +0100 | [diff] [blame] | 198 | git_config(git_default_config, NULL); |
Deskin Miller | 0a2c7ee | 2008-10-07 01:37:48 -0400 | [diff] [blame] | 199 | is_bare_repository_cfg = init_is_bare_repository; |
Junio C Hamano | 5a688fe | 2009-03-25 16:19:36 -0700 | [diff] [blame] | 200 | |
| 201 | /* reading existing config may have overwrote it */ |
Deskin Miller | 0a2c7ee | 2008-10-07 01:37:48 -0400 | [diff] [blame] | 202 | if (init_shared_repository != -1) |
| 203 | shared_repository = init_shared_repository; |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 204 | |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 205 | /* |
Junio C Hamano | 138086a | 2006-06-09 22:07:23 -0700 | [diff] [blame] | 206 | * We would have created the above under user's umask -- under |
| 207 | * shared-repository settings, we would need to fix them up. |
| 208 | */ |
| 209 | if (shared_repository) { |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 210 | adjust_shared_perm(get_git_dir()); |
| 211 | adjust_shared_perm(git_path("refs")); |
| 212 | adjust_shared_perm(git_path("refs/heads")); |
| 213 | adjust_shared_perm(git_path("refs/tags")); |
Junio C Hamano | 138086a | 2006-06-09 22:07:23 -0700 | [diff] [blame] | 214 | } |
| 215 | |
| 216 | /* |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 217 | * Create the default symlink from ".git/HEAD" to the "master" |
Junio C Hamano | 8098a17 | 2005-09-30 14:26:57 -0700 | [diff] [blame] | 218 | * branch, if it does not exist yet. |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 219 | */ |
| 220 | strcpy(path + len, "HEAD"); |
Johannes Schindelin | 5cc8f37 | 2008-03-24 16:14:52 +0100 | [diff] [blame] | 221 | reinit = (!access(path, R_OK) |
| 222 | || readlink(path, junk, sizeof(junk)-1) != -1); |
Shawn O. Pearce | ef0a89a | 2006-12-15 00:44:58 -0500 | [diff] [blame] | 223 | if (!reinit) { |
Nicolas Pitre | 8b5157e | 2007-01-26 17:26:10 -0500 | [diff] [blame] | 224 | if (create_symref("HEAD", "refs/heads/master", NULL) < 0) |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 225 | exit(1); |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 226 | } |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 227 | |
| 228 | /* This forces creation of new config file */ |
| 229 | sprintf(repo_version_string, "%d", GIT_REPO_VERSION); |
| 230 | git_config_set("core.repositoryformatversion", repo_version_string); |
| 231 | |
Junio C Hamano | 8098a17 | 2005-09-30 14:26:57 -0700 | [diff] [blame] | 232 | path[len] = 0; |
Johannes Schindelin | e24317b | 2005-10-26 01:43:03 +0200 | [diff] [blame] | 233 | strcpy(path + len, "config"); |
Johannes Schindelin | e24317b | 2005-10-26 01:43:03 +0200 | [diff] [blame] | 234 | |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 235 | /* Check filemode trustability */ |
Shawn O. Pearce | c869753 | 2006-12-30 23:53:55 -0500 | [diff] [blame] | 236 | filemode = TEST_FILEMODE; |
| 237 | if (TEST_FILEMODE && !lstat(path, &st1)) { |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 238 | struct stat st2; |
Shawn O. Pearce | c869753 | 2006-12-30 23:53:55 -0500 | [diff] [blame] | 239 | filemode = (!chmod(path, st1.st_mode ^ S_IXUSR) && |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 240 | !lstat(path, &st2) && |
| 241 | st1.st_mode != st2.st_mode); |
Johannes Schindelin | e24317b | 2005-10-26 01:43:03 +0200 | [diff] [blame] | 242 | } |
Shawn O. Pearce | c869753 | 2006-12-30 23:53:55 -0500 | [diff] [blame] | 243 | git_config_set("core.filemode", filemode ? "true" : "false"); |
Shawn O. Pearce | ef0a89a | 2006-12-15 00:44:58 -0500 | [diff] [blame] | 244 | |
Johannes Schindelin | e90fdc3 | 2007-08-01 01:30:14 +0100 | [diff] [blame] | 245 | if (is_bare_repository()) |
Junio C Hamano | 7d1864c | 2007-01-07 02:00:28 -0800 | [diff] [blame] | 246 | git_config_set("core.bare", "true"); |
Junio C Hamano | 7d1864c | 2007-01-07 02:00:28 -0800 | [diff] [blame] | 247 | else { |
Johannes Schindelin | e90fdc3 | 2007-08-01 01:30:14 +0100 | [diff] [blame] | 248 | const char *work_tree = get_git_work_tree(); |
Junio C Hamano | 7d1864c | 2007-01-07 02:00:28 -0800 | [diff] [blame] | 249 | git_config_set("core.bare", "false"); |
Alex Riesen | 196055c | 2007-01-23 16:51:18 +0100 | [diff] [blame] | 250 | /* allow template config file to override the default */ |
| 251 | if (log_all_ref_updates == -1) |
| 252 | git_config_set("core.logallrefupdates", "true"); |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 253 | if (prefixcmp(git_dir, work_tree) || |
| 254 | strcmp(git_dir + strlen(work_tree), "/.git")) { |
Johannes Schindelin | e90fdc3 | 2007-08-01 01:30:14 +0100 | [diff] [blame] | 255 | git_config_set("core.worktree", work_tree); |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 256 | } |
Junio C Hamano | 7d1864c | 2007-01-07 02:00:28 -0800 | [diff] [blame] | 257 | } |
Junio C Hamano | 75d2449 | 2007-08-31 00:25:04 -0700 | [diff] [blame] | 258 | |
Junio C Hamano | 75d2449 | 2007-08-31 00:25:04 -0700 | [diff] [blame] | 259 | if (!reinit) { |
Dmitry Potapov | 2455406 | 2008-05-11 18:16:39 +0200 | [diff] [blame] | 260 | /* Check if symlink is supported in the work tree */ |
Junio C Hamano | 75d2449 | 2007-08-31 00:25:04 -0700 | [diff] [blame] | 261 | path[len] = 0; |
| 262 | strcpy(path + len, "tXXXXXX"); |
| 263 | if (!close(xmkstemp(path)) && |
| 264 | !unlink(path) && |
| 265 | !symlink("testing", path) && |
| 266 | !lstat(path, &st1) && |
| 267 | S_ISLNK(st1.st_mode)) |
| 268 | unlink(path); /* good */ |
| 269 | else |
| 270 | git_config_set("core.symlinks", "false"); |
Dmitry Potapov | 2455406 | 2008-05-11 18:16:39 +0200 | [diff] [blame] | 271 | |
| 272 | /* Check if the filesystem is case-insensitive */ |
| 273 | path[len] = 0; |
| 274 | strcpy(path + len, "CoNfIg"); |
| 275 | if (!access(path, F_OK)) |
| 276 | git_config_set("core.ignorecase", "true"); |
Junio C Hamano | 75d2449 | 2007-08-31 00:25:04 -0700 | [diff] [blame] | 277 | } |
| 278 | |
Shawn O. Pearce | ef0a89a | 2006-12-15 00:44:58 -0500 | [diff] [blame] | 279 | return reinit; |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 280 | } |
| 281 | |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 282 | int init_db(const char *template_dir, unsigned int flags) |
Junio C Hamano | 6adcca3 | 2007-08-27 00:58:06 -0700 | [diff] [blame] | 283 | { |
Junio C Hamano | d19938a | 2005-05-09 17:57:56 -0700 | [diff] [blame] | 284 | const char *sha1_dir; |
Timo Hirvonen | c3c8835 | 2006-05-19 13:03:57 +0300 | [diff] [blame] | 285 | char *path; |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 286 | int len, reinit; |
Linus Torvalds | e83c516 | 2005-04-07 15:13:13 -0700 | [diff] [blame] | 287 | |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 288 | safe_create_dir(get_git_dir(), 0); |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 289 | |
Deskin Miller | 0a2c7ee | 2008-10-07 01:37:48 -0400 | [diff] [blame] | 290 | init_is_bare_repository = is_bare_repository(); |
| 291 | |
Junio C Hamano | 4f62953 | 2005-11-25 16:03:56 -0800 | [diff] [blame] | 292 | /* Check to see if the repository version is right. |
| 293 | * Note that a newly created repository does not have |
| 294 | * config file, so this will not fail. What we are catching |
| 295 | * is an attempt to reinitialize new repository with an old tool. |
| 296 | */ |
| 297 | check_repository_format(); |
| 298 | |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 299 | reinit = create_default_files(template_dir); |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 300 | |
Linus Torvalds | cad88fd | 2005-05-30 10:20:44 -0700 | [diff] [blame] | 301 | sha1_dir = get_object_directory(); |
Linus Torvalds | e83c516 | 2005-04-07 15:13:13 -0700 | [diff] [blame] | 302 | len = strlen(sha1_dir); |
Christopher Li | 812666c | 2005-04-26 12:00:58 -0700 | [diff] [blame] | 303 | path = xmalloc(len + 40); |
Linus Torvalds | e83c516 | 2005-04-07 15:13:13 -0700 | [diff] [blame] | 304 | memcpy(path, sha1_dir, len); |
Zach Welch | cb126d8 | 2005-04-19 21:48:15 -0700 | [diff] [blame] | 305 | |
Johannes Schindelin | af6e277 | 2005-12-22 23:19:37 +0100 | [diff] [blame] | 306 | safe_create_dir(sha1_dir, 1); |
Linus Torvalds | f49fb35 | 2005-06-27 18:26:11 -0700 | [diff] [blame] | 307 | strcpy(path+len, "/pack"); |
Johannes Schindelin | af6e277 | 2005-12-22 23:19:37 +0100 | [diff] [blame] | 308 | safe_create_dir(path, 1); |
Junio C Hamano | d57306c | 2005-08-20 02:05:31 -0700 | [diff] [blame] | 309 | strcpy(path+len, "/info"); |
Johannes Schindelin | af6e277 | 2005-12-22 23:19:37 +0100 | [diff] [blame] | 310 | safe_create_dir(path, 1); |
| 311 | |
Junio C Hamano | 94df250 | 2006-06-09 23:09:49 -0700 | [diff] [blame] | 312 | if (shared_repository) { |
| 313 | char buf[10]; |
| 314 | /* We do not spell "group" and such, so that |
| 315 | * the configuration can be read by older version |
Heikki Orsila | 06cbe85 | 2008-04-16 11:34:24 +0300 | [diff] [blame] | 316 | * of git. Note, we use octal numbers for new share modes, |
| 317 | * and compatibility values for PERM_GROUP and |
| 318 | * PERM_EVERYBODY. |
Junio C Hamano | 94df250 | 2006-06-09 23:09:49 -0700 | [diff] [blame] | 319 | */ |
Junio C Hamano | 5a688fe | 2009-03-25 16:19:36 -0700 | [diff] [blame] | 320 | if (shared_repository < 0) |
| 321 | /* force to the mode value */ |
| 322 | sprintf(buf, "0%o", -shared_repository); |
| 323 | else if (shared_repository == PERM_GROUP) |
Heikki Orsila | 06cbe85 | 2008-04-16 11:34:24 +0300 | [diff] [blame] | 324 | sprintf(buf, "%d", OLD_PERM_GROUP); |
| 325 | else if (shared_repository == PERM_EVERYBODY) |
| 326 | sprintf(buf, "%d", OLD_PERM_EVERYBODY); |
| 327 | else |
Junio C Hamano | 5a688fe | 2009-03-25 16:19:36 -0700 | [diff] [blame] | 328 | die("oops"); |
Junio C Hamano | 94df250 | 2006-06-09 23:09:49 -0700 | [diff] [blame] | 329 | git_config_set("core.sharedrepository", buf); |
Johannes Schindelin | 11031d7 | 2006-09-21 01:07:54 +0200 | [diff] [blame] | 330 | git_config_set("receive.denyNonFastforwards", "true"); |
Junio C Hamano | 94df250 | 2006-06-09 23:09:49 -0700 | [diff] [blame] | 331 | } |
Johannes Schindelin | af6e277 | 2005-12-22 23:19:37 +0100 | [diff] [blame] | 332 | |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 333 | if (!(flags & INIT_DB_QUIET)) |
Jeffrey C. Ollie | 4576518 | 2007-06-07 07:50:29 -0500 | [diff] [blame] | 334 | printf("%s%s Git repository in %s/\n", |
| 335 | reinit ? "Reinitialized existing" : "Initialized empty", |
| 336 | shared_repository ? " shared" : "", |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 337 | get_git_dir()); |
Shawn O. Pearce | ef0a89a | 2006-12-15 00:44:58 -0500 | [diff] [blame] | 338 | |
Linus Torvalds | e83c516 | 2005-04-07 15:13:13 -0700 | [diff] [blame] | 339 | return 0; |
| 340 | } |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 341 | |
| 342 | static int guess_repository_type(const char *git_dir) |
| 343 | { |
| 344 | char cwd[PATH_MAX]; |
| 345 | const char *slash; |
| 346 | |
| 347 | /* |
| 348 | * "GIT_DIR=. git init" is always bare. |
| 349 | * "GIT_DIR=`pwd` git init" too. |
| 350 | */ |
| 351 | if (!strcmp(".", git_dir)) |
| 352 | return 1; |
| 353 | if (!getcwd(cwd, sizeof(cwd))) |
Thomas Rast | 0721c31 | 2009-06-27 17:58:47 +0200 | [diff] [blame] | 354 | die_errno("cannot tell cwd"); |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 355 | if (!strcmp(git_dir, cwd)) |
| 356 | return 1; |
| 357 | /* |
| 358 | * "GIT_DIR=.git or GIT_DIR=something/.git is usually not. |
| 359 | */ |
| 360 | if (!strcmp(git_dir, ".git")) |
| 361 | return 0; |
| 362 | slash = strrchr(git_dir, '/'); |
| 363 | if (slash && !strcmp(slash, "/.git")) |
| 364 | return 0; |
| 365 | |
| 366 | /* |
| 367 | * Otherwise it is often bare. At this point |
| 368 | * we are just guessing. |
| 369 | */ |
| 370 | return 1; |
| 371 | } |
| 372 | |
| 373 | static const char init_db_usage[] = |
Stephan Beyer | 1b1dd23 | 2008-07-13 15:36:15 +0200 | [diff] [blame] | 374 | "git init [-q | --quiet] [--bare] [--template=<template-directory>] [--shared[=<permissions>]]"; |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 375 | |
| 376 | /* |
| 377 | * If you want to, you can share the DB area with any number of branches. |
| 378 | * That has advantages: you can save space by sharing all the SHA1 objects. |
| 379 | * On the other hand, it might just make lookup slower and messier. You |
| 380 | * be the judge. The default case is to have one DB per managed directory. |
| 381 | */ |
| 382 | int cmd_init_db(int argc, const char **argv, const char *prefix) |
| 383 | { |
| 384 | const char *git_dir; |
| 385 | const char *template_dir = NULL; |
| 386 | unsigned int flags = 0; |
| 387 | int i; |
| 388 | |
| 389 | for (i = 1; i < argc; i++, argv++) { |
| 390 | const char *arg = argv[1]; |
| 391 | if (!prefixcmp(arg, "--template=")) |
| 392 | template_dir = arg+11; |
Luciano Rocha | 74d3b23 | 2008-05-28 19:53:57 +0100 | [diff] [blame] | 393 | else if (!strcmp(arg, "--bare")) { |
| 394 | static char git_dir[PATH_MAX+1]; |
| 395 | is_bare_repository_cfg = 1; |
| 396 | setenv(GIT_DIR_ENVIRONMENT, getcwd(git_dir, |
| 397 | sizeof(git_dir)), 0); |
| 398 | } else if (!strcmp(arg, "--shared")) |
Deskin Miller | 0a2c7ee | 2008-10-07 01:37:48 -0400 | [diff] [blame] | 399 | init_shared_repository = PERM_GROUP; |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 400 | else if (!prefixcmp(arg, "--shared=")) |
Deskin Miller | 0a2c7ee | 2008-10-07 01:37:48 -0400 | [diff] [blame] | 401 | init_shared_repository = git_config_perm("arg", arg+9); |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 402 | else if (!strcmp(arg, "-q") || !strcmp(arg, "--quiet")) |
| 403 | flags |= INIT_DB_QUIET; |
| 404 | else |
| 405 | usage(init_db_usage); |
| 406 | } |
| 407 | |
Junio C Hamano | 5a688fe | 2009-03-25 16:19:36 -0700 | [diff] [blame] | 408 | if (init_shared_repository != -1) |
| 409 | shared_repository = init_shared_repository; |
| 410 | |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 411 | /* |
| 412 | * GIT_WORK_TREE makes sense only in conjunction with GIT_DIR |
| 413 | * without --bare. Catch the error early. |
| 414 | */ |
| 415 | git_dir = getenv(GIT_DIR_ENVIRONMENT); |
| 416 | if ((!git_dir || is_bare_repository_cfg == 1) |
| 417 | && getenv(GIT_WORK_TREE_ENVIRONMENT)) |
| 418 | die("%s (or --work-tree=<directory>) not allowed without " |
| 419 | "specifying %s (or --git-dir=<directory>)", |
| 420 | GIT_WORK_TREE_ENVIRONMENT, |
| 421 | GIT_DIR_ENVIRONMENT); |
| 422 | |
| 423 | /* |
| 424 | * Set up the default .git directory contents |
| 425 | */ |
| 426 | if (!git_dir) |
| 427 | git_dir = DEFAULT_GIT_DIR_ENVIRONMENT; |
| 428 | |
| 429 | if (is_bare_repository_cfg < 0) |
| 430 | is_bare_repository_cfg = guess_repository_type(git_dir); |
| 431 | |
| 432 | if (!is_bare_repository_cfg) { |
| 433 | if (git_dir) { |
| 434 | const char *git_dir_parent = strrchr(git_dir, '/'); |
| 435 | if (git_dir_parent) { |
| 436 | char *rel = xstrndup(git_dir, git_dir_parent - git_dir); |
| 437 | git_work_tree_cfg = xstrdup(make_absolute_path(rel)); |
| 438 | free(rel); |
| 439 | } |
| 440 | } |
| 441 | if (!git_work_tree_cfg) { |
| 442 | git_work_tree_cfg = xcalloc(PATH_MAX, 1); |
| 443 | if (!getcwd(git_work_tree_cfg, PATH_MAX)) |
Thomas Rast | 0721c31 | 2009-06-27 17:58:47 +0200 | [diff] [blame] | 444 | die_errno ("Cannot access current working directory"); |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 445 | } |
| 446 | if (access(get_git_work_tree(), X_OK)) |
Thomas Rast | 0721c31 | 2009-06-27 17:58:47 +0200 | [diff] [blame] | 447 | die_errno ("Cannot access work tree '%s'", |
| 448 | get_git_work_tree()); |
Daniel Barkalow | f225aeb | 2008-04-27 13:39:27 -0400 | [diff] [blame] | 449 | } |
| 450 | |
| 451 | set_git_dir(make_absolute_path(git_dir)); |
| 452 | |
| 453 | return init_db(template_dir, flags); |
| 454 | } |