Allow helper to map private ref names into normal names
This allows a helper to say that, when it handles "import refs/heads/topic", the script it outputs will actually write to refs/svn/origin/branches/topic; therefore, transport-helper should read it from the latter location after git-fast-import completes. Signed-off-by: Daniel Barkalow <barkalow@iabervon.org> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
e65e91ed4a
commit
72ff894308
@ -46,7 +46,11 @@ Supported if the helper has the "fetch" capability.
|
|||||||
'import' <name>::
|
'import' <name>::
|
||||||
Produces a fast-import stream which imports the current value
|
Produces a fast-import stream which imports the current value
|
||||||
of the named ref. It may additionally import other refs as
|
of the named ref. It may additionally import other refs as
|
||||||
needed to construct the history efficiently.
|
needed to construct the history efficiently. The script writes
|
||||||
|
to a helper-specific private namespace. The value of the named
|
||||||
|
ref should be written to a location in this namespace derived
|
||||||
|
by applying the refspecs from the "refspec" capability to the
|
||||||
|
name of the ref.
|
||||||
+
|
+
|
||||||
Supported if the helper has the "import" capability.
|
Supported if the helper has the "import" capability.
|
||||||
|
|
||||||
@ -67,6 +71,16 @@ CAPABILITIES
|
|||||||
'import'::
|
'import'::
|
||||||
This helper supports the 'import' command.
|
This helper supports the 'import' command.
|
||||||
|
|
||||||
|
'refspec' 'spec'::
|
||||||
|
When using the import command, expect the source ref to have
|
||||||
|
been written to the destination ref. The earliest applicable
|
||||||
|
refspec takes precedence. For example
|
||||||
|
"refs/heads/*:refs/svn/origin/branches/*" means that, after an
|
||||||
|
"import refs/heads/name", the script has written to
|
||||||
|
refs/svn/origin/branches/name. If this capability is used at
|
||||||
|
all, it must cover all refs reported by the list command; if
|
||||||
|
it is not used, it is effectively "*:*"
|
||||||
|
|
||||||
REF LIST ATTRIBUTES
|
REF LIST ATTRIBUTES
|
||||||
-------------------
|
-------------------
|
||||||
|
|
||||||
|
27
remote.c
27
remote.c
@ -673,6 +673,16 @@ static struct refspec *parse_push_refspec(int nr_refspec, const char **refspec)
|
|||||||
return parse_refspec_internal(nr_refspec, refspec, 0, 0);
|
return parse_refspec_internal(nr_refspec, refspec, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void free_refspec(int nr_refspec, struct refspec *refspec)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < nr_refspec; i++) {
|
||||||
|
free(refspec[i].src);
|
||||||
|
free(refspec[i].dst);
|
||||||
|
}
|
||||||
|
free(refspec);
|
||||||
|
}
|
||||||
|
|
||||||
static int valid_remote_nick(const char *name)
|
static int valid_remote_nick(const char *name)
|
||||||
{
|
{
|
||||||
if (!name[0] || is_dot_or_dotdot(name))
|
if (!name[0] || is_dot_or_dotdot(name))
|
||||||
@ -811,6 +821,23 @@ static int match_name_with_pattern(const char *key, const char *name,
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
char *apply_refspecs(struct refspec *refspecs, int nr_refspec,
|
||||||
|
const char *name)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
char *ret = NULL;
|
||||||
|
for (i = 0; i < nr_refspec; i++) {
|
||||||
|
struct refspec *refspec = refspecs + i;
|
||||||
|
if (refspec->pattern) {
|
||||||
|
if (match_name_with_pattern(refspec->src, name,
|
||||||
|
refspec->dst, &ret))
|
||||||
|
return ret;
|
||||||
|
} else if (!strcmp(refspec->src, name))
|
||||||
|
return strdup(refspec->dst);
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
int remote_find_tracking(struct remote *remote, struct refspec *refspec)
|
int remote_find_tracking(struct remote *remote, struct refspec *refspec)
|
||||||
{
|
{
|
||||||
int find_src = refspec->src == NULL;
|
int find_src = refspec->src == NULL;
|
||||||
|
5
remote.h
5
remote.h
@ -91,6 +91,11 @@ void ref_remove_duplicates(struct ref *ref_map);
|
|||||||
int valid_fetch_refspec(const char *refspec);
|
int valid_fetch_refspec(const char *refspec);
|
||||||
struct refspec *parse_fetch_refspec(int nr_refspec, const char **refspec);
|
struct refspec *parse_fetch_refspec(int nr_refspec, const char **refspec);
|
||||||
|
|
||||||
|
void free_refspec(int nr_refspec, struct refspec *refspec);
|
||||||
|
|
||||||
|
char *apply_refspecs(struct refspec *refspecs, int nr_refspec,
|
||||||
|
const char *name);
|
||||||
|
|
||||||
int match_refs(struct ref *src, struct ref **dst,
|
int match_refs(struct ref *src, struct ref **dst,
|
||||||
int nr_refspec, const char **refspec, int all);
|
int nr_refspec, const char **refspec, int all);
|
||||||
|
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
#include "commit.h"
|
#include "commit.h"
|
||||||
#include "diff.h"
|
#include "diff.h"
|
||||||
#include "revision.h"
|
#include "revision.h"
|
||||||
|
#include "remote.h"
|
||||||
|
|
||||||
struct helper_data
|
struct helper_data
|
||||||
{
|
{
|
||||||
@ -12,6 +13,9 @@ struct helper_data
|
|||||||
struct child_process *helper;
|
struct child_process *helper;
|
||||||
unsigned fetch : 1;
|
unsigned fetch : 1;
|
||||||
unsigned import : 1;
|
unsigned import : 1;
|
||||||
|
/* These go from remote name (as in "list") to private name */
|
||||||
|
struct refspec *refspecs;
|
||||||
|
int refspec_nr;
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct child_process *get_helper(struct transport *transport)
|
static struct child_process *get_helper(struct transport *transport)
|
||||||
@ -20,6 +24,9 @@ static struct child_process *get_helper(struct transport *transport)
|
|||||||
struct strbuf buf = STRBUF_INIT;
|
struct strbuf buf = STRBUF_INIT;
|
||||||
struct child_process *helper;
|
struct child_process *helper;
|
||||||
FILE *file;
|
FILE *file;
|
||||||
|
const char **refspecs = NULL;
|
||||||
|
int refspec_nr = 0;
|
||||||
|
int refspec_alloc = 0;
|
||||||
|
|
||||||
if (data->helper)
|
if (data->helper)
|
||||||
return data->helper;
|
return data->helper;
|
||||||
@ -51,6 +58,21 @@ static struct child_process *get_helper(struct transport *transport)
|
|||||||
data->fetch = 1;
|
data->fetch = 1;
|
||||||
if (!strcmp(buf.buf, "import"))
|
if (!strcmp(buf.buf, "import"))
|
||||||
data->import = 1;
|
data->import = 1;
|
||||||
|
if (!data->refspecs && !prefixcmp(buf.buf, "refspec ")) {
|
||||||
|
ALLOC_GROW(refspecs,
|
||||||
|
refspec_nr + 1,
|
||||||
|
refspec_alloc);
|
||||||
|
refspecs[refspec_nr++] = strdup(buf.buf + strlen("refspec "));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (refspecs) {
|
||||||
|
int i;
|
||||||
|
data->refspec_nr = refspec_nr;
|
||||||
|
data->refspecs = parse_fetch_refspec(refspec_nr, refspecs);
|
||||||
|
for (i = 0; i < refspec_nr; i++) {
|
||||||
|
free((char *)refspecs[i]);
|
||||||
|
}
|
||||||
|
free(refspecs);
|
||||||
}
|
}
|
||||||
return data->helper;
|
return data->helper;
|
||||||
}
|
}
|
||||||
@ -72,6 +94,9 @@ static int disconnect_helper(struct transport *transport)
|
|||||||
|
|
||||||
static int release_helper(struct transport *transport)
|
static int release_helper(struct transport *transport)
|
||||||
{
|
{
|
||||||
|
struct helper_data *data = transport->data;
|
||||||
|
free_refspec(data->refspec_nr, data->refspecs);
|
||||||
|
data->refspecs = NULL;
|
||||||
disconnect_helper(transport);
|
disconnect_helper(transport);
|
||||||
free(transport->data);
|
free(transport->data);
|
||||||
return 0;
|
return 0;
|
||||||
@ -119,6 +144,7 @@ static int fetch_with_import(struct transport *transport,
|
|||||||
{
|
{
|
||||||
struct child_process fastimport;
|
struct child_process fastimport;
|
||||||
struct child_process *helper = get_helper(transport);
|
struct child_process *helper = get_helper(transport);
|
||||||
|
struct helper_data *data = transport->data;
|
||||||
int i;
|
int i;
|
||||||
struct ref *posn;
|
struct ref *posn;
|
||||||
struct strbuf buf = STRBUF_INIT;
|
struct strbuf buf = STRBUF_INIT;
|
||||||
@ -139,10 +165,16 @@ static int fetch_with_import(struct transport *transport,
|
|||||||
finish_command(&fastimport);
|
finish_command(&fastimport);
|
||||||
|
|
||||||
for (i = 0; i < nr_heads; i++) {
|
for (i = 0; i < nr_heads; i++) {
|
||||||
|
char *private;
|
||||||
posn = to_fetch[i];
|
posn = to_fetch[i];
|
||||||
if (posn->status & REF_STATUS_UPTODATE)
|
if (posn->status & REF_STATUS_UPTODATE)
|
||||||
continue;
|
continue;
|
||||||
read_ref(posn->name, posn->old_sha1);
|
if (data->refspecs)
|
||||||
|
private = apply_refspecs(data->refspecs, data->refspec_nr, posn->name);
|
||||||
|
else
|
||||||
|
private = strdup(posn->name);
|
||||||
|
read_ref(private, posn->old_sha1);
|
||||||
|
free(private);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user