|
|
|
#ifndef REFS_H
|
|
|
|
#define REFS_H
|
|
|
|
|
|
|
|
struct ref_lock {
|
Enable the packed refs file format
This actually "turns on" the packed ref file format, now that the
infrastructure to do so sanely exists (ie notably the change to make the
reference reading logic take refnames rather than pathnames to the loose
objects that no longer necessarily even exist).
In particular, when the ref lookup hits a refname that has no loose file
associated with it, it falls back on the packed-ref information. Also, the
ref-locking code, while still using a loose file for the locking itself
(and _creating_ a loose file for the new ref) no longer requires that the
old ref be in such an unpacked state.
Finally, this does a minimal hack to git-checkout.sh to rather than check
the ref-file directly, do a "git-rev-parse" on the "heads/$refname".
That's not really wonderful - we should rather really have a special
routine to verify the names as proper branch head names, but it is a
workable solution for now.
With this, I can literally do something like
git pack-refs
find .git/refs -type f -print0 | xargs -0 rm -f --
and the end result is a largely working repository (ie I've done two
commits - which creates _one_ unpacked ref file - done things like run
"gitk" and "git log" etc, and it all looks ok).
There are probably things missing, but I'm hoping that the missing things
are now of the "small and obvious" kind, and that somebody else might want
to start looking at this too. Hint hint ;)
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Signed-off-by: Junio C Hamano <junkio@cox.net>
19 years ago
|
|
|
char *ref_name;
|
|
|
|
char *orig_ref_name;
|
|
|
|
struct lock_file *lk;
|
|
|
|
unsigned char old_sha1[20];
|
|
|
|
int lock_fd;
|
|
|
|
int force_write;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define REF_ISSYMREF 01
|
|
|
|
#define REF_ISPACKED 02
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Calls the specified function for each ref file until it returns nonzero,
|
|
|
|
* and returns the value
|
|
|
|
*/
|
|
|
|
typedef int each_ref_fn(const char *refname, const unsigned char *sha1, int flags, void *cb_data);
|
|
|
|
extern int head_ref(each_ref_fn, void *);
|
|
|
|
extern int for_each_ref(each_ref_fn, void *);
|
|
|
|
extern int for_each_ref_in(const char *, each_ref_fn, void *);
|
|
|
|
extern int for_each_tag_ref(each_ref_fn, void *);
|
|
|
|
extern int for_each_branch_ref(each_ref_fn, void *);
|
|
|
|
extern int for_each_remote_ref(each_ref_fn, void *);
|
|
|
|
extern int for_each_replace_ref(each_ref_fn, void *);
|
|
|
|
extern int for_each_glob_ref(each_ref_fn, const char *pattern, void *);
|
|
|
|
extern int for_each_glob_ref_in(each_ref_fn, const char *pattern, const char* prefix, void *);
|
|
|
|
|
|
|
|
extern int head_ref_submodule(const char *submodule, each_ref_fn fn, void *cb_data);
|
|
|
|
extern int for_each_ref_submodule(const char *submodule, each_ref_fn fn, void *cb_data);
|
|
|
|
extern int for_each_ref_in_submodule(const char *submodule, const char *prefix,
|
|
|
|
each_ref_fn fn, void *cb_data);
|
|
|
|
extern int for_each_tag_ref_submodule(const char *submodule, each_ref_fn fn, void *cb_data);
|
|
|
|
extern int for_each_branch_ref_submodule(const char *submodule, each_ref_fn fn, void *cb_data);
|
|
|
|
extern int for_each_remote_ref_submodule(const char *submodule, each_ref_fn fn, void *cb_data);
|
|
|
|
|
ref namespaces: infrastructure
Add support for dividing the refs of a single repository into multiple
namespaces, each of which can have its own branches, tags, and HEAD.
Git can expose each namespace as an independent repository to pull from
and push to, while sharing the object store, and exposing all the refs
to operations such as git-gc.
Storing multiple repositories as namespaces of a single repository
avoids storing duplicate copies of the same objects, such as when
storing multiple branches of the same source. The alternates mechanism
provides similar support for avoiding duplicates, but alternates do not
prevent duplication between new objects added to the repositories
without ongoing maintenance, while namespaces do.
To specify a namespace, set the GIT_NAMESPACE environment variable to
the namespace. For each ref namespace, git stores the corresponding
refs in a directory under refs/namespaces/. For example,
GIT_NAMESPACE=foo will store refs under refs/namespaces/foo/. You can
also specify namespaces via the --namespace option to git.
Note that namespaces which include a / will expand to a hierarchy of
namespaces; for example, GIT_NAMESPACE=foo/bar will store refs under
refs/namespaces/foo/refs/namespaces/bar/. This makes paths in
GIT_NAMESPACE behave hierarchically, so that cloning with
GIT_NAMESPACE=foo/bar produces the same result as cloning with
GIT_NAMESPACE=foo and cloning from that repo with GIT_NAMESPACE=bar. It
also avoids ambiguity with strange namespace paths such as
foo/refs/heads/, which could otherwise generate directory/file conflicts
within the refs directory.
Add the infrastructure for ref namespaces: handle the GIT_NAMESPACE
environment variable and --namespace option, and support iterating over
refs in a namespace.
Signed-off-by: Josh Triplett <josh@joshtriplett.org>
Signed-off-by: Jamey Sharp <jamey@minilop.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
14 years ago
|
|
|
extern int head_ref_namespaced(each_ref_fn fn, void *cb_data);
|
|
|
|
extern int for_each_namespaced_ref(each_ref_fn fn, void *cb_data);
|
|
|
|
|
|
|
|
static inline const char *has_glob_specials(const char *pattern)
|
|
|
|
{
|
|
|
|
return strpbrk(pattern, "?*[");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* can be used to learn about broken ref and symref */
|
|
|
|
extern int for_each_rawref(each_ref_fn, void *);
|
|
|
|
|
|
|
|
extern void warn_dangling_symref(FILE *fp, const char *msg_fmt, const char *refname);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Extra refs will be listed by for_each_ref() before any actual refs
|
|
|
|
* for the duration of this process or until clear_extra_refs() is
|
|
|
|
* called. Only extra refs added before for_each_ref() is called will
|
|
|
|
* be listed on a given call of for_each_ref().
|
|
|
|
*/
|
|
|
|
extern void add_extra_ref(const char *refname, const unsigned char *sha1, int flags);
|
|
|
|
extern void clear_extra_refs(void);
|
|
|
|
extern int ref_exists(char *);
|
|
|
|
|
|
|
|
extern int peel_ref(const char *, unsigned char *);
|
|
|
|
|
|
|
|
/** Locks a "refs/" ref returning the lock on success and NULL on failure. **/
|
|
|
|
extern struct ref_lock *lock_ref_sha1(const char *ref, const unsigned char *old_sha1);
|
|
|
|
|
|
|
|
/** Locks any ref (for 'HEAD' type refs). */
|
|
|
|
#define REF_NODEREF 0x01
|
|
|
|
extern struct ref_lock *lock_any_ref_for_update(const char *ref, const unsigned char *old_sha1, int flags);
|
|
|
|
|
|
|
|
/** Close the file descriptor owned by a lock and return the status */
|
|
|
|
extern int close_ref(struct ref_lock *lock);
|
|
|
|
|
|
|
|
/** Close and commit the ref locked by the lock */
|
|
|
|
extern int commit_ref(struct ref_lock *lock);
|
|
|
|
|
|
|
|
/** Release any lock taken but not written. **/
|
|
|
|
extern void unlock_ref(struct ref_lock *lock);
|
|
|
|
|
|
|
|
/** Writes sha1 into the ref specified by the lock. **/
|
|
|
|
extern int write_ref_sha1(struct ref_lock *lock, const unsigned char *sha1, const char *msg);
|
|
|
|
|
|
|
|
/** Setup reflog before using. **/
|
|
|
|
int log_ref_setup(const char *ref_name, char *logfile, int bufsize);
|
|
|
|
|
|
|
|
/** Reads log for the value of ref during at_time. **/
|
|
|
|
extern int read_ref_at(const char *ref, unsigned long at_time, int cnt, unsigned char *sha1, char **msg, unsigned long *cutoff_time, int *cutoff_tz, int *cutoff_cnt);
|
|
|
|
|
|
|
|
/* iterate over reflog entries */
|
|
|
|
typedef int each_reflog_ent_fn(unsigned char *osha1, unsigned char *nsha1, const char *, unsigned long, int, const char *, void *);
|
|
|
|
int for_each_reflog_ent(const char *ref, each_reflog_ent_fn fn, void *cb_data);
|
|
|
|
int for_each_recent_reflog_ent(const char *ref, each_reflog_ent_fn fn, long, void *cb_data);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Calls the specified function for each reflog file until it returns nonzero,
|
|
|
|
* and returns the value
|
|
|
|
*/
|
|
|
|
extern int for_each_reflog(each_ref_fn, void *);
|
|
|
|
|
|
|
|
#define CHECK_REF_FORMAT_OK 0
|
|
|
|
#define CHECK_REF_FORMAT_ERROR (-1)
|
|
|
|
#define CHECK_REF_FORMAT_ONELEVEL (-2)
|
|
|
|
#define CHECK_REF_FORMAT_WILDCARD (-3)
|
|
|
|
extern int check_ref_format(const char *target);
|
|
|
|
|
|
|
|
extern const char *prettify_refname(const char *refname);
|
|
|
|
extern char *shorten_unambiguous_ref(const char *ref, int strict);
|
|
|
|
|
|
|
|
/** rename ref, return 0 on success **/
|
|
|
|
extern int rename_ref(const char *oldref, const char *newref, const char *logmsg);
|
|
|
|
|
|
|
|
/** resolve ref in nested "gitlink" repository */
|
|
|
|
extern int resolve_gitlink_ref(const char *name, const char *refname, unsigned char *result);
|
|
|
|
|
|
|
|
/** lock a ref and then write its file */
|
|
|
|
enum action_on_err { MSG_ON_ERR, DIE_ON_ERR, QUIET_ON_ERR };
|
|
|
|
int update_ref(const char *action, const char *refname,
|
|
|
|
const unsigned char *sha1, const unsigned char *oldval,
|
|
|
|
int flags, enum action_on_err onerr);
|
|
|
|
|
|
|
|
#endif /* REFS_H */
|