Skip to content

Commit c58cac1

Browse files
committed
Introduce a glob-filtering iterator
If the backend doesn't provide support for it, the matching is done in refdb on top of a normal iterator.
1 parent 9bd89d9 commit c58cac1

File tree

5 files changed

+62
-12
lines changed

5 files changed

+62
-12
lines changed

include/git2/sys/refdb_backend.h

Lines changed: 13 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -30,10 +30,11 @@ GIT_BEGIN_DECL
3030
* ...
3131
* }
3232
*
33-
* and assing `iter->parent.backend` to your `git_refdb_backend`.
33+
* and assign `iter->parent.backend` to your `git_refdb_backend`.
3434
*/
3535
struct git_reference_iterator {
3636
git_refdb_backend *backend;
37+
char *glob;
3738
};
3839

3940
/** An instance for a custom backend */
@@ -67,6 +68,17 @@ struct git_refdb_backend {
6768
git_reference_iterator **iter,
6869
struct git_refdb_backend *backend);
6970

71+
/**
72+
* Allocate a glob-filtering iterator object for the backend.
73+
*
74+
* A refdb implementation may provide this function. If it's
75+
* not available, the glob matching will be done by the frontend.
76+
*/
77+
int (*iterator_glob)(
78+
git_reference_iterator **iter,
79+
struct git_refdb_backend *backend,
80+
const char *glob);
81+
7082
/**
7183
* Return the current value and advance the iterator.
7284
*

src/refdb.c

Lines changed: 36 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -126,29 +126,59 @@ int git_refdb_lookup(git_reference **out, git_refdb *db, const char *ref_name)
126126

127127
int git_refdb_iterator(git_reference_iterator **out, git_refdb *db)
128128
{
129-
git_reference_iterator *iter;
130-
131129
if (!db->backend || !db->backend->iterator) {
132130
giterr_set(GITERR_REFERENCE, "This backend doesn't support iterators");
133131
return -1;
134132
}
135133

136-
if (db->backend->iterator(&iter, db->backend) < 0) {
137-
git__free(iter);
134+
if (db->backend->iterator(out, db->backend) < 0)
135+
return -1;
136+
137+
return 0;
138+
}
139+
140+
int git_refdb_iterator_glob(git_reference_iterator **out, git_refdb *db, const char *glob)
141+
{
142+
if (!db->backend) {
143+
giterr_set(GITERR_REFERENCE, "There are no backends loaded");
144+
return -1;
145+
}
146+
147+
if (db->backend->iterator_glob)
148+
return db->backend->iterator_glob(out, db->backend, glob);
149+
150+
/* If the backend doesn't support glob-filtering themselves, we have to do it */
151+
if (db->backend->iterator(out, db->backend) < 0)
152+
return -1;
153+
154+
(*out)->glob = git__strdup(glob);
155+
if (!(*out)->glob) {
156+
db->backend->iterator_free(*out);
138157
return -1;
139158
}
140159

141-
*out = iter;
142160
return 0;
143161
}
144162

145163
int git_refdb_next(const char **out, git_reference_iterator *iter)
146164
{
147-
return iter->backend->next(out, iter);
165+
int error;
166+
167+
if (!iter->glob)
168+
return iter->backend->next(out, iter);
169+
170+
/* If the iterator has a glob, we need to filter */
171+
while ((error = iter->backend->next(out, iter)) == 0) {
172+
if (!p_fnmatch(iter->glob, *out, 0))
173+
break;
174+
}
175+
176+
return error;
148177
}
149178

150179
void git_refdb_iterator_free(git_reference_iterator *iter)
151180
{
181+
git__free(iter->glob);
152182
iter->backend->iterator_free(iter);
153183
}
154184

src/refdb.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ int git_refdb_lookup(
2727
const char *ref_name);
2828

2929
int git_refdb_iterator(git_reference_iterator **out, git_refdb *db);
30+
int git_refdb_iterator_glob(git_reference_iterator **out, git_refdb *db, const char *glob);
3031
int git_refdb_next(const char **out, git_reference_iterator *iter);
3132
void git_refdb_iterator_free(git_reference_iterator *iter);
3233

src/refs.c

Lines changed: 11 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -593,6 +593,16 @@ int git_reference_iterator_new(git_reference_iterator **out, git_repository *rep
593593
return git_refdb_iterator(out, refdb);
594594
}
595595

596+
int git_reference_iterator_glob_new(git_reference_iterator **out, git_repository *repo, const char *glob)
597+
{
598+
git_refdb *refdb;
599+
600+
if (git_repository_refdb__weakptr(&refdb, repo) < 0)
601+
return -1;
602+
603+
return git_refdb_iterator_glob(out, refdb, glob);
604+
}
605+
596606
int git_reference_next(const char **out, git_reference_iterator *iter)
597607
{
598608
return git_refdb_next(out, iter);
@@ -928,13 +938,10 @@ int git_reference_foreach_glob(
928938
const char *name;
929939
int error;
930940

931-
if (git_reference_iterator_new(&iter, repo) < 0)
941+
if (git_reference_iterator_glob_new(&iter, repo, glob) < 0)
932942
return -1;
933943

934944
while ((error = git_reference_next(&name, iter)) == 0) {
935-
if (p_fnmatch(glob, name, 0))
936-
continue;
937-
938945
if (callback(name, payload)) {
939946
error = GIT_EUSER;
940947
goto out;

tests-clar/refdb/testdb.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -123,7 +123,7 @@ static int refdb_test_backend__iterator(git_reference_iterator **out, git_refdb_
123123

124124
GIT_UNUSED(_backend);
125125

126-
iter = git__malloc(sizeof(refdb_test_iter));
126+
iter = git__calloc(1, sizeof(refdb_test_iter));
127127
GITERR_CHECK_ALLOC(iter);
128128

129129
iter->parent.backend = _backend;

0 commit comments

Comments
 (0)