fetch.c 4.6 KB
Newer Older
J
Junio C Hamano 已提交
1
#include "fetch.h"
2 3 4 5

#include "cache.h"
#include "commit.h"
#include "tree.h"
D
Daniel Barkalow 已提交
6 7
#include "tag.h"
#include "blob.h"
8 9 10 11 12
#include "refs.h"

const char *write_ref = NULL;

const unsigned char *current_ref = NULL;
13 14 15 16

int get_tree = 0;
int get_history = 0;
int get_all = 0;
17
int get_verbosely = 0;
18
static unsigned char current_commit_sha1[20];
19

20 21
void pull_say(const char *fmt, const char *hex) 
{
22 23 24 25
	if (get_verbosely)
		fprintf(stderr, fmt, hex);
}

26 27 28 29 30 31 32 33 34 35
static void report_missing(const char *what, const unsigned char *missing)
{
	char missing_hex[41];

	strcpy(missing_hex, sha1_to_hex(missing));;
	fprintf(stderr,
		"Cannot obtain needed %s %s\nwhile processing commit %s.\n",
		what, missing_hex, sha1_to_hex(current_commit_sha1));
}

36
static int process(struct object *obj);
D
Daniel Barkalow 已提交
37

38
static int process_tree(struct tree *tree)
39
{
40
	struct tree_entry_list *entry;
41 42 43 44

	if (parse_tree(tree))
		return -1;

45 46 47 48
	entry = tree->entries;
	tree->entries = NULL;
	while (entry) {
		struct tree_entry_list *next = entry->next;
49
		if (process(entry->item.any))
50
			return -1;
51 52
		free(entry);
		entry = next;
53 54 55 56
	}
	return 0;
}

57 58 59
#define COMPLETE	(1U << 0)
#define SEEN		(1U << 1)
#define TO_SCAN		(1U << 2)
60

J
Junio C Hamano 已提交
61
static struct commit_list *complete = NULL;
62

63
static int process_commit(struct commit *commit)
64
{
65
	if (parse_commit(commit))
66 67
		return -1;

68
	while (complete && complete->item->date >= commit->date) {
J
Junio C Hamano 已提交
69
		pop_most_recent_commit(&complete, COMPLETE);
70 71
	}

J
Junio C Hamano 已提交
72
	if (commit->object.flags & COMPLETE)
73 74
		return 0;

75
	memcpy(current_commit_sha1, commit->object.sha1, 20);
76

77 78
	pull_say("walk %s\n", sha1_to_hex(commit->object.sha1));

79
	if (get_tree) {
80
		if (process(&commit->tree->object))
81 82 83 84 85
			return -1;
		if (!get_all)
			get_tree = 0;
	}
	if (get_history) {
86
		struct commit_list *parents = commit->parents;
87
		for (; parents; parents = parents->next) {
88
			if (process(&parents->item->object))
89 90 91 92 93 94
				return -1;
		}
	}
	return 0;
}

95
static int process_tag(struct tag *tag)
D
Daniel Barkalow 已提交
96
{
97
	if (parse_tag(tag))
D
Daniel Barkalow 已提交
98
		return -1;
99
	return process(tag->tagged);
D
Daniel Barkalow 已提交
100 101
}

102 103 104
static struct object_list *process_queue = NULL;
static struct object_list **process_queue_end = &process_queue;

105
static int process_object(struct object *obj)
D
Daniel Barkalow 已提交
106
{
107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
	if (obj->type == commit_type) {
		if (process_commit((struct commit *)obj))
			return -1;
		return 0;
	}
	if (obj->type == tree_type) {
		if (process_tree((struct tree *)obj))
			return -1;
		return 0;
	}
	if (obj->type == blob_type) {
		return 0;
	}
	if (obj->type == tag_type) {
		if (process_tag((struct tag *)obj))
			return -1;
D
Daniel Barkalow 已提交
123
		return 0;
124 125 126 127 128 129
	}
	return error("Unable to determine requirements "
		     "of type %s for %s",
		     obj->type, sha1_to_hex(obj->sha1));
}

130
static int process(struct object *obj)
131
{
132 133 134 135
	if (obj->flags & SEEN)
		return 0;
	obj->flags |= SEEN;

136
	if (has_sha1_file(obj->sha1)) {
137
		/* We already have it, so we should scan it now. */
138
		obj->flags |= TO_SCAN;
139 140 141 142
	} else {
		if (obj->flags & COMPLETE)
			return 0;
		prefetch(obj->sha1);
143
	}
144
		
145 146 147 148 149 150 151
	object_list_insert(obj, process_queue_end);
	process_queue_end = &(*process_queue_end)->next;
	return 0;
}

static int loop(void)
{
152 153
	struct object_list *elem;

154 155
	while (process_queue) {
		struct object *obj = process_queue->item;
156 157 158
		elem = process_queue;
		process_queue = elem->next;
		free(elem);
159 160 161
		if (!process_queue)
			process_queue_end = &process_queue;

162 163 164 165
		/* If we are not scanning this object, we placed it in
		 * the queue because we needed to fetch it first.
		 */
		if (! (obj->flags & TO_SCAN)) {
166
			if (!has_sha1_file(obj->sha1) && fetch(obj->sha1)) {
167 168 169 170 171 172
				report_missing(obj->type
					       ? obj->type
					       : "object", obj->sha1);
				return -1;
			}
		}
173 174
		if (!obj->type)
			parse_object(obj->sha1);
175 176
		if (process_object(obj))
			return -1;
177 178
	}
	return 0;
D
Daniel Barkalow 已提交
179 180
}

181 182 183 184 185 186 187 188 189 190 191 192
static int interpret_target(char *target, unsigned char *sha1)
{
	if (!get_sha1_hex(target, sha1))
		return 0;
	if (!check_ref_format(target)) {
		if (!fetch_ref(target, sha1)) {
			return 0;
		}
	}
	return -1;
}

193 194
static int mark_complete(const char *path, const unsigned char *sha1)
{
J
Junio C Hamano 已提交
195 196 197 198
	struct commit *commit = lookup_commit_reference_gently(sha1, 1);
	if (commit) {
		commit->object.flags |= COMPLETE;
		insert_by_date(commit, &complete);
199 200 201
	}
	return 0;
}
202

203 204 205
int pull(char *target)
{
	unsigned char sha1[20];
206 207
	int fd = -1;

208
	save_commit_buffer = 0;
209
	track_object_refs = 0;
210 211 212 213 214 215
	if (write_ref && current_ref) {
		fd = lock_ref_sha1(write_ref, current_ref);
		if (fd < 0)
			return -1;
	}

216 217
	for_each_ref(mark_complete);

218 219 220
	if (interpret_target(target, sha1))
		return error("Could not interpret %s as something to pull",
			     target);
221
	if (process(lookup_unknown_object(sha1)))
222 223
		return -1;
	if (loop())
224 225 226 227 228 229 230 231 232 233
		return -1;
	
	if (write_ref) {
		if (current_ref) {
			write_ref_sha1(write_ref, fd, sha1);
		} else {
			write_ref_sha1_unlocked(write_ref, sha1);
		}
	}
	return 0;
234
}