2 * Check-out files from the "current cache directory"
4 * Copyright (C) 2005 Linus Torvalds
6 * Careful: order of argument flags does matter. For example,
8 * checkout-cache -a -f file.c
10 * Will first check out all files listed in the cache (but not
11 * overwrite any old ones), and then force-checkout "file.c" a
12 * second time (ie that one _will_ overwrite any old contents
13 * with the same filename).
15 * Also, just doing "checkout-cache" does nothing. You probably
16 * meant "checkout-cache -a". And if you want to force it, you
17 * want "checkout-cache -f -a".
19 * Intuitiveness is not the goal here. Repeatability is. The
20 * reason for the "no arguments means no work" thing is that
21 * from scripts you are supposed to be able to do things like
23 * find . -name '*.h' -print0 | xargs -0 checkout-cache -f --
25 * which will force all existing *.h files to be replaced with
26 * their cached copies. If an empty command line implied "all",
27 * then this would force-refresh everything in the cache, which
30 * Oh, and the "--" is just a good idea when you know the rest
31 * will be filenames. Just so that you wouldn't have a filename
32 * of "-a" causing problems (not possible in the above example,
33 * but get used to it in scripting!).
37 static int force
= 0, quiet
= 0;
39 static void create_directories(const char *path
)
41 int len
= strlen(path
);
42 char *buf
= malloc(len
+ 1);
43 const char *slash
= path
;
45 while ((slash
= strchr(slash
+1, '/')) != NULL
) {
47 memcpy(buf
, path
, len
);
53 static int create_file(const char *path
, unsigned int mode
)
57 mode
= (mode
& 0100) ? 0777 : 0666;
58 fd
= open(path
, O_WRONLY
| O_TRUNC
| O_CREAT
, mode
);
60 if (errno
== ENOENT
) {
61 create_directories(path
);
62 fd
= open(path
, O_WRONLY
| O_TRUNC
| O_CREAT
, mode
);
68 static int write_entry(struct cache_entry
*ce
, const char *path
)
76 new = read_sha1_file(ce
->sha1
, type
, &size
);
77 if (!new || strcmp(type
, "blob")) {
78 return error("checkout-cache: unable to read sha1 file of %s (%s)",
79 path
, sha1_to_hex(ce
->sha1
));
81 fd
= create_file(path
, ntohl(ce
->ce_mode
));
84 return error("checkout-cache: unable to create %s (%s)",
85 path
, strerror(errno
));
87 wrote
= write(fd
, new, size
);
91 return error("checkout-cache: unable to write %s", path
);
95 static int checkout_entry(struct cache_entry
*ce
, const char *base_dir
)
98 static char path
[MAXPATHLEN
+1];
99 int len
= strlen(base_dir
);
101 memcpy(path
, base_dir
, len
);
102 strcpy(path
+ len
, ce
->name
);
104 if (!stat(path
, &st
)) {
105 unsigned changed
= cache_match_stat(ce
, &st
);
110 fprintf(stderr
, "checkout-cache: %s already exists\n", path
);
115 * We unlink the old file, to get the new one with the
116 * right permissions (including umask, which is nasty
117 * to emulate by hand - much easier to let the system
118 * just do the right thing)
122 return write_entry(ce
, path
);
125 static int checkout_file(const char *name
, const char *base_dir
)
127 int pos
= cache_name_pos(name
, strlen(name
));
132 "checkout-cache: %s is %s.\n",
135 !strcmp(active_cache
[pos
]->name
, name
)) ?
136 "unmerged" : "not in the cache");
140 return checkout_entry(active_cache
[pos
], base_dir
);
143 static int checkout_all(const char *base_dir
)
147 for (i
= 0; i
< active_nr
; i
++) {
148 struct cache_entry
*ce
= active_cache
[i
];
151 if (checkout_entry(ce
, base_dir
) < 0)
157 int main(int argc
, char **argv
)
159 int i
, force_filename
= 0;
160 const char *base_dir
= "";
162 if (read_cache() < 0) {
163 die("invalid cache");
166 for (i
= 1; i
< argc
; i
++) {
167 const char *arg
= argv
[i
];
168 if (!force_filename
) {
169 if (!strcmp(arg
, "-a")) {
170 checkout_all(base_dir
);
173 if (!strcmp(arg
, "--")) {
177 if (!strcmp(arg
, "-f")) {
181 if (!strcmp(arg
, "-q")) {
185 if (!memcmp(arg
, "--prefix=", 9)) {
190 checkout_file(arg
, base_dir
);