submodule.h 5.77 KB
Newer Older
Russell Belfer committed
1
/*
Edward Thomson committed
2
 * Copyright (C) the libgit2 contributors. All rights reserved.
Russell Belfer committed
3 4 5 6 7 8 9
 *
 * This file is part of libgit2, distributed under the GNU GPL v2 with
 * a Linking Exception. For full terms see the included COPYING file.
 */
#ifndef INCLUDE_submodule_h__
#define INCLUDE_submodule_h__

10 11
#include "common.h"

12 13
#include "git2/submodule.h"
#include "git2/repository.h"
14
#include "futils.h"
15

Russell Belfer committed
16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52
/* Notes:
 *
 * Submodule information can be in four places: the index, the config files
 * (both .git/config and .gitmodules), the HEAD tree, and the working
 * directory.
 *
 * In the index:
 * - submodule is found by path
 * - may be missing, present, or of the wrong type
 * - will have an oid if present
 *
 * In the HEAD tree:
 * - submodule is found by path
 * - may be missing, present, or of the wrong type
 * - will have an oid if present
 *
 * In the config files:
 * - submodule is found by submodule "name" which is usually the path
 * - may be missing or present
 * - will have a name, path, url, and other properties
 *
 * In the working directory:
 * - submodule is found by path
 * - may be missing, an empty directory, a checked out directory,
 *   or of the wrong type
 * - if checked out, will have a HEAD oid
 * - if checked out, will have git history that can be used to compare oids
 * - if checked out, may have modified files and/or untracked files
 */

/**
 * Description of submodule
 *
 * This record describes a submodule found in a repository.  There should be
 * an entry for every submodule found in the HEAD and index, and for every
 * submodule described in .gitmodules.  The fields are as follows:
 *
53 54 55 56
 * - `rc` tracks the refcount of how many hash table entries in the
 *   git_submodule_cache there are for this submodule.  It only comes into
 *   play if the name and path of the submodule differ.
 *
Russell Belfer committed
57 58 59 60 61
 * - `name` is the name of the submodule from .gitmodules.
 * - `path` is the path to the submodule from the repo root.  It is almost
 *    always the same as `name`.
 * - `url` is the url for the submodule.
 * - `update` is a git_submodule_update_t value - see gitmodules(5) update.
62
 * - `update_default` is the update value from the config
Russell Belfer committed
63
 * - `ignore` is a git_submodule_ignore_t value - see gitmodules(5) ignore.
64
 * - `ignore_default` is the ignore value from the config
65 66 67
 * - `fetch_recurse` is a git_submodule_recurse_t value - see gitmodules(5)
 *    fetchRecurseSubmodules.
 * - `fetch_recurse_default` is the recurse value from the config
68
 *
69
 * - `repo` is the parent repository that contains this submodule.
70 71 72 73 74
 * - `flags` after for internal use, tracking where this submodule has been
 *   found (head, index, config, workdir) and known status info, etc.
 * - `head_oid` is the SHA1 for the submodule path in the repo HEAD.
 * - `index_oid` is the SHA1 for the submodule recorded in the index.
 * - `wd_oid` is the SHA1 for the HEAD of the checked out submodule.
Russell Belfer committed
75 76
 *
 * If the submodule has been added to .gitmodules but not yet git added,
77 78 79
 * then the `index_oid` will be zero but still marked valid.  If the
 * submodule has been deleted, but the delete has not been committed yet,
 * then the `index_oid` will be set, but the `url` will be NULL.
Russell Belfer committed
80 81
 */
struct git_submodule {
82 83
	git_refcount rc;

84
	/* information from config */
Russell Belfer committed
85
	char *name;
86
	char *path; /* important: may just point to "name" string */
Russell Belfer committed
87
	char *url;
88
	char *branch;
Russell Belfer committed
89 90 91 92
	git_submodule_update_t update;
	git_submodule_update_t update_default;
	git_submodule_ignore_t ignore;
	git_submodule_ignore_t ignore_default;
93
	git_submodule_recurse_t fetch_recurse;
94
	git_submodule_recurse_t fetch_recurse_default;
95

Russell Belfer committed
96
	/* internal information */
97
	git_repository *repo;
98 99 100 101
	uint32_t flags;
	git_oid head_oid;
	git_oid index_oid;
	git_oid wd_oid;
Russell Belfer committed
102 103 104
};

/* Additional flags on top of public GIT_SUBMODULE_STATUS values */
105 106 107 108 109 110 111 112 113 114 115 116 117
enum {
	GIT_SUBMODULE_STATUS__WD_SCANNED          = (1u << 20),
	GIT_SUBMODULE_STATUS__HEAD_OID_VALID      = (1u << 21),
	GIT_SUBMODULE_STATUS__INDEX_OID_VALID     = (1u << 22),
	GIT_SUBMODULE_STATUS__WD_OID_VALID        = (1u << 23),
	GIT_SUBMODULE_STATUS__HEAD_NOT_SUBMODULE  = (1u << 24),
	GIT_SUBMODULE_STATUS__INDEX_NOT_SUBMODULE = (1u << 25),
	GIT_SUBMODULE_STATUS__WD_NOT_SUBMODULE    = (1u << 26),
	GIT_SUBMODULE_STATUS__INDEX_MULTIPLE_ENTRIES = (1u << 27),
};

#define GIT_SUBMODULE_STATUS__CLEAR_INTERNAL(S) \
	((S) & ~(0xFFFFFFFFu << 20))
Russell Belfer committed
118

119 120 121 122 123 124
/* Initialize an external submodule cache for the provided repo. */
extern int git_submodule_cache_init(git_strmap **out, git_repository *repo);

/* Release the resources of the submodule cache. */
extern int git_submodule_cache_free(git_strmap *cache);

125 126 127
/* Submodule lookup with an explicit cache */
extern int git_submodule__lookup_with_cache(
	git_submodule **out, git_repository *repo, const char *path, git_strmap *cache);
128

129 130 131 132 133 134 135 136 137 138 139 140 141 142
/* Internal status fn returns status and optionally the various OIDs */
extern int git_submodule__status(
	unsigned int *out_status,
	git_oid *out_head_id,
	git_oid *out_index_id,
	git_oid *out_wd_id,
	git_submodule *sm,
	git_submodule_ignore_t ign);

/* Open submodule repository as bare repo for quick HEAD check, etc. */
extern int git_submodule_open_bare(
	git_repository **repo,
	git_submodule *submodule);

143 144 145 146 147
extern int git_submodule_parse_ignore(
	git_submodule_ignore_t *out, const char *value);
extern int git_submodule_parse_update(
	git_submodule_update_t *out, const char *value);

148 149 150
extern int git_submodule__map(
	git_repository *repo,
	git_strmap *map);
151 152 153 154 155 156 157 158 159 160 161

/**
 * Check whether a submodule's name is valid.
 *
 * Check the path against the path validity rules, either the filesystem
 * defaults (like checkout does) or whichever you want to compare against.
 *
 * @param repo the repository which contains the submodule
 * @param name the name to check
 * @param flags the `GIT_PATH` flags to use for the check (0 to use filesystem defaults)
 */
162
extern int git_submodule_name_is_valid(git_repository *repo, const char *name, int flags);
163

Russell Belfer committed
164
#endif