10 #include "list-objects.h"
11 #include "list-objects-filter.h"
12 #include "list-objects-filter-options.h"
15 #include "object-store.h"
17 /* Remember to update object flag allocation in object.h */
19 * FILTER_SHOWN_BUT_REVISIT -- we set this bit on tree objects
20 * that have been shown, but should be revisited if they appear
21 * in the traversal (until we mark it SEEN). This is a way to
22 * let us silently de-dup calls to show() in the caller. This
23 * is subtly different from the "revision.h:SHOWN" and the
24 * "sha1-name.c:ONELINE_SEEN" bits. And also different from
25 * the non-de-dup usage in pack-bitmap.c
27 #define FILTER_SHOWN_BUT_REVISIT (1<<21)
30 * A filter for list-objects to omit ALL blobs from the traversal.
31 * And to OPTIONALLY collect a list of the omitted OIDs.
33 struct filter_blobs_none_data
{
37 static enum list_objects_filter_result
filter_blobs_none(
39 enum list_objects_filter_situation filter_situation
,
45 struct filter_blobs_none_data
*filter_data
= filter_data_
;
47 switch (filter_situation
) {
49 BUG("unknown filter_situation: %d", filter_situation
);
52 assert(obj
->type
== OBJ_TREE
);
53 /* always include all tree objects */
54 return LOFR_MARK_SEEN
| LOFR_DO_SHOW
;
57 assert(obj
->type
== OBJ_TREE
);
61 assert(obj
->type
== OBJ_BLOB
);
62 assert((obj
->flags
& SEEN
) == 0);
64 if (filter_data
->omits
)
65 oidset_insert(filter_data
->omits
, &obj
->oid
);
66 return LOFR_MARK_SEEN
; /* but not LOFR_DO_SHOW (hard omit) */
70 static void *filter_blobs_none__init(
71 struct oidset
*omitted
,
72 struct list_objects_filter_options
*filter_options
,
73 filter_object_fn
*filter_fn
,
74 filter_free_fn
*filter_free_fn
)
76 struct filter_blobs_none_data
*d
= xcalloc(1, sizeof(*d
));
79 *filter_fn
= filter_blobs_none
;
80 *filter_free_fn
= free
;
85 * A filter for list-objects to omit ALL trees and blobs from the traversal.
86 * Can OPTIONALLY collect a list of the omitted OIDs.
88 struct filter_trees_depth_data
{
92 * Maps trees to the minimum depth at which they were seen. It is not
93 * necessary to re-traverse a tree at deeper or equal depths than it has
94 * already been traversed.
96 * We can't use LOFR_MARK_SEEN for tree objects since this will prevent
97 * it from being traversed at shallower depths.
99 struct oidmap seen_at_depth
;
101 unsigned long exclude_depth
;
102 unsigned long current_depth
;
105 struct seen_map_entry
{
106 struct oidmap_entry base
;
110 /* Returns 1 if the oid was in the omits set before it was invoked. */
111 static int filter_trees_update_omits(
113 struct filter_trees_depth_data
*filter_data
,
116 if (!filter_data
->omits
)
120 return oidset_remove(filter_data
->omits
, &obj
->oid
);
122 return oidset_insert(filter_data
->omits
, &obj
->oid
);
125 static enum list_objects_filter_result
filter_trees_depth(
126 struct repository
*r
,
127 enum list_objects_filter_situation filter_situation
,
129 const char *pathname
,
130 const char *filename
,
133 struct filter_trees_depth_data
*filter_data
= filter_data_
;
134 struct seen_map_entry
*seen_info
;
135 int include_it
= filter_data
->current_depth
<
136 filter_data
->exclude_depth
;
141 * Note that we do not use _MARK_SEEN in order to allow re-traversal in
142 * case we encounter a tree or blob again at a shallower depth.
145 switch (filter_situation
) {
147 BUG("unknown filter_situation: %d", filter_situation
);
150 assert(obj
->type
== OBJ_TREE
);
151 filter_data
->current_depth
--;
155 filter_trees_update_omits(obj
, filter_data
, include_it
);
156 return include_it
? LOFR_MARK_SEEN
| LOFR_DO_SHOW
: LOFR_ZERO
;
158 case LOFS_BEGIN_TREE
:
159 seen_info
= oidmap_get(
160 &filter_data
->seen_at_depth
, &obj
->oid
);
162 seen_info
= xcalloc(1, sizeof(*seen_info
));
163 oidcpy(&seen_info
->base
.oid
, &obj
->oid
);
164 seen_info
->depth
= filter_data
->current_depth
;
165 oidmap_put(&filter_data
->seen_at_depth
, seen_info
);
169 filter_data
->current_depth
>= seen_info
->depth
;
173 filter_res
= LOFR_SKIP_TREE
;
175 int been_omitted
= filter_trees_update_omits(
176 obj
, filter_data
, include_it
);
177 seen_info
->depth
= filter_data
->current_depth
;
180 filter_res
= LOFR_DO_SHOW
;
181 else if (filter_data
->omits
&& !been_omitted
)
183 * Must update omit information of children
184 * recursively; they have not been omitted yet.
186 filter_res
= LOFR_ZERO
;
188 filter_res
= LOFR_SKIP_TREE
;
191 filter_data
->current_depth
++;
196 static void filter_trees_free(void *filter_data
) {
197 struct filter_trees_depth_data
*d
= filter_data
;
200 oidmap_free(&d
->seen_at_depth
, 1);
204 static void *filter_trees_depth__init(
205 struct oidset
*omitted
,
206 struct list_objects_filter_options
*filter_options
,
207 filter_object_fn
*filter_fn
,
208 filter_free_fn
*filter_free_fn
)
210 struct filter_trees_depth_data
*d
= xcalloc(1, sizeof(*d
));
212 oidmap_init(&d
->seen_at_depth
, 0);
213 d
->exclude_depth
= filter_options
->tree_exclude_depth
;
214 d
->current_depth
= 0;
216 *filter_fn
= filter_trees_depth
;
217 *filter_free_fn
= filter_trees_free
;
222 * A filter for list-objects to omit large blobs.
223 * And to OPTIONALLY collect a list of the omitted OIDs.
225 struct filter_blobs_limit_data
{
226 struct oidset
*omits
;
227 unsigned long max_bytes
;
230 static enum list_objects_filter_result
filter_blobs_limit(
231 struct repository
*r
,
232 enum list_objects_filter_situation filter_situation
,
234 const char *pathname
,
235 const char *filename
,
238 struct filter_blobs_limit_data
*filter_data
= filter_data_
;
239 unsigned long object_length
;
242 switch (filter_situation
) {
244 BUG("unknown filter_situation: %d", filter_situation
);
246 case LOFS_BEGIN_TREE
:
247 assert(obj
->type
== OBJ_TREE
);
248 /* always include all tree objects */
249 return LOFR_MARK_SEEN
| LOFR_DO_SHOW
;
252 assert(obj
->type
== OBJ_TREE
);
256 assert(obj
->type
== OBJ_BLOB
);
257 assert((obj
->flags
& SEEN
) == 0);
259 t
= oid_object_info(r
, &obj
->oid
, &object_length
);
260 if (t
!= OBJ_BLOB
) { /* probably OBJ_NONE */
262 * We DO NOT have the blob locally, so we cannot
263 * apply the size filter criteria. Be conservative
264 * and force show it (and let the caller deal with
270 if (object_length
< filter_data
->max_bytes
)
273 if (filter_data
->omits
)
274 oidset_insert(filter_data
->omits
, &obj
->oid
);
275 return LOFR_MARK_SEEN
; /* but not LOFR_DO_SHOW (hard omit) */
279 if (filter_data
->omits
)
280 oidset_remove(filter_data
->omits
, &obj
->oid
);
281 return LOFR_MARK_SEEN
| LOFR_DO_SHOW
;
284 static void *filter_blobs_limit__init(
285 struct oidset
*omitted
,
286 struct list_objects_filter_options
*filter_options
,
287 filter_object_fn
*filter_fn
,
288 filter_free_fn
*filter_free_fn
)
290 struct filter_blobs_limit_data
*d
= xcalloc(1, sizeof(*d
));
292 d
->max_bytes
= filter_options
->blob_limit_value
;
294 *filter_fn
= filter_blobs_limit
;
295 *filter_free_fn
= free
;
300 * A filter driven by a sparse-checkout specification to only
301 * include blobs that a sparse checkout would populate.
303 * The sparse-checkout spec can be loaded from a blob with the
304 * given OID or from a local pathname. We allow an OID because
305 * the repo may be bare or we may be doing the filtering on the
310 * defval is the usual default include/exclude value that
311 * should be inherited as we recurse into directories based
312 * upon pattern matching of the directory itself or of a
313 * containing directory.
318 * 1 if the directory (recursively) contains any provisionally
321 * 0 if everything (recursively) contained in this directory
322 * has been explicitly included (SHOWN) in the result and
323 * the directory may be short-cut later in the traversal.
325 unsigned child_prov_omit
: 1;
328 struct filter_sparse_data
{
329 struct oidset
*omits
;
330 struct exclude_list el
;
333 struct frame
*array_frame
;
336 static enum list_objects_filter_result
filter_sparse(
337 struct repository
*r
,
338 enum list_objects_filter_situation filter_situation
,
340 const char *pathname
,
341 const char *filename
,
344 struct filter_sparse_data
*filter_data
= filter_data_
;
348 switch (filter_situation
) {
350 BUG("unknown filter_situation: %d", filter_situation
);
352 case LOFS_BEGIN_TREE
:
353 assert(obj
->type
== OBJ_TREE
);
355 val
= is_excluded_from_list(pathname
, strlen(pathname
),
356 filename
, &dtype
, &filter_data
->el
,
359 val
= filter_data
->array_frame
[filter_data
->nr
- 1].defval
;
361 ALLOC_GROW(filter_data
->array_frame
, filter_data
->nr
+ 1,
363 filter_data
->array_frame
[filter_data
->nr
].defval
= val
;
364 filter_data
->array_frame
[filter_data
->nr
].child_prov_omit
= 0;
368 * A directory with this tree OID may appear in multiple
369 * places in the tree. (Think of a directory move or copy,
370 * with no other changes, so the OID is the same, but the
371 * full pathnames of objects within this directory are new
372 * and may match is_excluded() patterns differently.)
373 * So we cannot mark this directory as SEEN (yet), since
374 * that will prevent process_tree() from revisiting this
375 * tree object with other pathname prefixes.
377 * Only _DO_SHOW the tree object the first time we visit
380 * We always show all tree objects. A future optimization
381 * may want to attempt to narrow this.
383 if (obj
->flags
& FILTER_SHOWN_BUT_REVISIT
)
385 obj
->flags
|= FILTER_SHOWN_BUT_REVISIT
;
389 assert(obj
->type
== OBJ_TREE
);
390 assert(filter_data
->nr
> 1);
392 frame
= &filter_data
->array_frame
[--filter_data
->nr
];
395 * Tell our parent directory if any of our children were
396 * provisionally omitted.
398 filter_data
->array_frame
[filter_data
->nr
- 1].child_prov_omit
|=
399 frame
->child_prov_omit
;
402 * If there are NO provisionally omitted child objects (ALL child
403 * objects in this folder were INCLUDED), then we can mark the
404 * folder as SEEN (so we will not have to revisit it again).
406 if (!frame
->child_prov_omit
)
407 return LOFR_MARK_SEEN
;
411 assert(obj
->type
== OBJ_BLOB
);
412 assert((obj
->flags
& SEEN
) == 0);
414 frame
= &filter_data
->array_frame
[filter_data
->nr
- 1];
417 val
= is_excluded_from_list(pathname
, strlen(pathname
),
418 filename
, &dtype
, &filter_data
->el
,
423 if (filter_data
->omits
)
424 oidset_remove(filter_data
->omits
, &obj
->oid
);
425 return LOFR_MARK_SEEN
| LOFR_DO_SHOW
;
429 * Provisionally omit it. We've already established that
430 * this pathname is not in the sparse-checkout specification
431 * with the CURRENT pathname, so we *WANT* to omit this blob.
433 * However, a pathname elsewhere in the tree may also
434 * reference this same blob, so we cannot reject it yet.
435 * Leave the LOFR_ bits unset so that if the blob appears
436 * again in the traversal, we will be asked again.
438 if (filter_data
->omits
)
439 oidset_insert(filter_data
->omits
, &obj
->oid
);
442 * Remember that at least 1 blob in this tree was
443 * provisionally omitted. This prevents us from short
444 * cutting the tree in future iterations.
446 frame
->child_prov_omit
= 1;
452 static void filter_sparse_free(void *filter_data
)
454 struct filter_sparse_data
*d
= filter_data
;
455 free(d
->array_frame
);
459 static void *filter_sparse_oid__init(
460 struct oidset
*omitted
,
461 struct list_objects_filter_options
*filter_options
,
462 filter_object_fn
*filter_fn
,
463 filter_free_fn
*filter_free_fn
)
465 struct filter_sparse_data
*d
= xcalloc(1, sizeof(*d
));
467 if (add_excludes_from_blob_to_list(filter_options
->sparse_oid_value
,
468 NULL
, 0, &d
->el
) < 0)
469 die("could not load filter specification");
471 ALLOC_GROW(d
->array_frame
, d
->nr
+ 1, d
->alloc
);
472 d
->array_frame
[d
->nr
].defval
= 0; /* default to include */
473 d
->array_frame
[d
->nr
].child_prov_omit
= 0;
476 *filter_fn
= filter_sparse
;
477 *filter_free_fn
= filter_sparse_free
;
481 typedef void *(*filter_init_fn
)(
482 struct oidset
*omitted
,
483 struct list_objects_filter_options
*filter_options
,
484 filter_object_fn
*filter_fn
,
485 filter_free_fn
*filter_free_fn
);
488 * Must match "enum list_objects_filter_choice".
490 static filter_init_fn s_filters
[] = {
492 filter_blobs_none__init
,
493 filter_blobs_limit__init
,
494 filter_trees_depth__init
,
495 filter_sparse_oid__init
,
498 void *list_objects_filter__init(
499 struct oidset
*omitted
,
500 struct list_objects_filter_options
*filter_options
,
501 filter_object_fn
*filter_fn
,
502 filter_free_fn
*filter_free_fn
)
504 filter_init_fn init_fn
;
506 assert((sizeof(s_filters
) / sizeof(s_filters
[0])) == LOFC__COUNT
);
508 if (filter_options
->choice
>= LOFC__COUNT
)
509 BUG("invalid list-objects filter choice: %d",
510 filter_options
->choice
);
512 init_fn
= s_filters
[filter_options
->choice
];
514 return init_fn(omitted
, filter_options
,
515 filter_fn
, filter_free_fn
);
517 *filter_free_fn
= NULL
;