| | |
| | | import java.text.MessageFormat;
|
| | | import java.util.ArrayList;
|
| | | import java.util.Arrays;
|
| | | import java.util.Collection;
|
| | | import java.util.Collections;
|
| | | import java.util.Date;
|
| | | import java.util.HashMap;
|
| | |
| | | * Returns a list of repository names in the specified folder.
|
| | | *
|
| | | * @param repositoriesFolder
|
| | | * @param exportAll
|
| | | * if true, all repositories are listed. If false only the
|
| | | * repositories with a "git-daemon-export-ok" file are included
|
| | | * @param onlyBare
|
| | | * if true, only bare repositories repositories are listed. If
|
| | | * false all repositories are included.
|
| | | * @param searchSubfolders
|
| | | * recurse into subfolders to find grouped repositories
|
| | | * @return list of repository names
|
| | | */
|
| | | public static List<String> getRepositoryList(File repositoriesFolder, boolean exportAll,
|
| | | public static List<String> getRepositoryList(File repositoriesFolder, boolean onlyBare,
|
| | | boolean searchSubfolders) {
|
| | | List<String> list = new ArrayList<String>();
|
| | | if (repositoriesFolder == null || !repositoriesFolder.exists()) {
|
| | | return list;
|
| | | }
|
| | | list.addAll(getRepositoryList(repositoriesFolder.getAbsolutePath(), repositoriesFolder,
|
| | | exportAll, searchSubfolders));
|
| | | onlyBare, searchSubfolders));
|
| | | StringUtils.sortRepositorynames(list);
|
| | | return list;
|
| | | }
|
| | |
| | | * basePath is stripped from the repository name as repositories
|
| | | * are relative to this path
|
| | | * @param searchFolder
|
| | | * @param exportAll
|
| | | * if true all repositories are listed. If false only the
|
| | | * repositories with a "git-daemon-export-ok" file are included
|
| | | * @param onlyBare
|
| | | * if true only bare repositories will be listed. if false all
|
| | | * repositories are included.
|
| | | * @param searchSubfolders
|
| | | * recurse into subfolders to find grouped repositories
|
| | | * @return
|
| | | */
|
| | | private static List<String> getRepositoryList(String basePath, File searchFolder,
|
| | | boolean exportAll, boolean searchSubfolders) {
|
| | | boolean onlyBare, boolean searchSubfolders) {
|
| | | List<String> list = new ArrayList<String>();
|
| | | for (File file : searchFolder.listFiles()) {
|
| | | if (file.isDirectory()) {
|
| | | File gitDir = FileKey.resolve(new File(searchFolder, file.getName()), FS.DETECTED);
|
| | | if (gitDir != null) {
|
| | | boolean exportRepository = exportAll
|
| | | || new File(gitDir, "git-daemon-export-ok").exists();
|
| | |
|
| | | if (!exportRepository) {
|
| | | if (onlyBare && gitDir.getName().equals(".git")) {
|
| | | continue;
|
| | | }
|
| | | // determine repository name relative to base path
|
| | | String repository = StringUtils.getRelativePath(basePath,
|
| | | file.getAbsolutePath());
|
| | | list.add(repository);
|
| | | } else if (searchSubfolders) {
|
| | | } else if (searchSubfolders && file.canRead()) {
|
| | | // look for repositories in subfolders
|
| | | list.addAll(getRepositoryList(basePath, file, exportAll, searchSubfolders));
|
| | | list.addAll(getRepositoryList(basePath, file, onlyBare, searchSubfolders));
|
| | | }
|
| | | }
|
| | | }
|
| | |
| | | * last modified date of the repository folder is returned.
|
| | | *
|
| | | * @param repository
|
| | | * @param branch
|
| | | * if unspecified, all branches are checked.
|
| | | * @return
|
| | | */
|
| | | public static Date getLastChange(Repository repository, String branch) {
|
| | | public static Date getLastChange(Repository repository) {
|
| | | if (!hasCommits(repository)) {
|
| | | // null repository
|
| | | if (repository == null) {
|
| | |
| | | // fresh repository
|
| | | return new Date(repository.getDirectory().lastModified());
|
| | | }
|
| | | if (StringUtils.isEmpty(branch)) {
|
| | | List<RefModel> branchModels = getLocalBranches(repository, true, -1);
|
| | | if (branchModels.size() > 0) {
|
| | | // find most recent branch update
|
| | | Date lastChange = new Date(0);
|
| | | for (RefModel branchModel : branchModels) {
|
| | | if (branchModel.getDate().after(lastChange)) {
|
| | | lastChange = branchModel.getDate();
|
| | | }
|
| | | }
|
| | | return lastChange;
|
| | | } else {
|
| | | // try to find head
|
| | | branch = Constants.HEAD;
|
| | | }
|
| | | }
|
| | |
|
| | | // lookup specified branch
|
| | | RevCommit commit = getCommit(repository, branch);
|
| | | return getCommitDate(commit);
|
| | | List<RefModel> branchModels = getLocalBranches(repository, true, -1);
|
| | | if (branchModels.size() > 0) {
|
| | | // find most recent branch update
|
| | | Date lastChange = new Date(0);
|
| | | for (RefModel branchModel : branchModels) {
|
| | | if (branchModel.getDate().after(lastChange)) {
|
| | | lastChange = branchModel.getDate();
|
| | | }
|
| | | }
|
| | | return lastChange;
|
| | | }
|
| | | |
| | | // default to the repository folder modification date
|
| | | return new Date(repository.getDirectory().lastModified());
|
| | | }
|
| | |
|
| | | /**
|
| | |
| | | }
|
| | |
|
| | | /**
|
| | | * Returns the list of files in the repository that match one of the
|
| | | * specified extensions. This is a CASE-SENSITIVE search. If the repository
|
| | | * does not exist or is empty, an empty list is returned.
|
| | | * Returns the list of files in the repository on the default branch that
|
| | | * match one of the specified extensions. This is a CASE-SENSITIVE search.
|
| | | * If the repository does not exist or is empty, an empty list is returned.
|
| | | *
|
| | | * @param repository
|
| | | * @param extensions
|
| | | * @return list of files in repository with a matching extension
|
| | | */
|
| | | public static List<PathModel> getDocuments(Repository repository, List<String> extensions) {
|
| | | return getDocuments(repository, extensions, null);
|
| | | }
|
| | |
|
| | | /**
|
| | | * Returns the list of files in the repository in the specified commit that
|
| | | * match one of the specified extensions. This is a CASE-SENSITIVE search.
|
| | | * If the repository does not exist or is empty, an empty list is returned.
|
| | | * |
| | | * @param repository
|
| | | * @param extensions
|
| | | * @param objectId
|
| | | * @return list of files in repository with a matching extension
|
| | | */
|
| | | public static List<PathModel> getDocuments(Repository repository, List<String> extensions,
|
| | | String objectId) {
|
| | | List<PathModel> list = new ArrayList<PathModel>();
|
| | | if (!hasCommits(repository)) {
|
| | | return list;
|
| | | }
|
| | | RevCommit commit = getCommit(repository, null);
|
| | | RevCommit commit = getCommit(repository, objectId);
|
| | | final TreeWalk tw = new TreeWalk(repository);
|
| | | try {
|
| | | tw.addTree(commit.getTree());
|
| | | if (extensions != null && extensions.size() > 0) {
|
| | | Collection<TreeFilter> suffixFilters = new ArrayList<TreeFilter>();
|
| | | List<TreeFilter> suffixFilters = new ArrayList<TreeFilter>();
|
| | | for (String extension : extensions) {
|
| | | if (extension.charAt(0) == '.') {
|
| | | suffixFilters.add(PathSuffixFilter.create("\\" + extension));
|
| | |
| | | suffixFilters.add(PathSuffixFilter.create("\\." + extension));
|
| | | }
|
| | | }
|
| | | TreeFilter filter = OrTreeFilter.create(suffixFilters);
|
| | | TreeFilter filter;
|
| | | if (suffixFilters.size() == 1) {
|
| | | filter = suffixFilters.get(0);
|
| | | } else {
|
| | | filter = OrTreeFilter.create(suffixFilters);
|
| | | }
|
| | | tw.setFilter(filter);
|
| | | tw.setRecursive(true);
|
| | | }
|
| | |
| | | branchObject = getDefaultBranch(repository);
|
| | | } else {
|
| | | branchObject = repository.resolve(objectId);
|
| | | }
|
| | | if (branchObject == null) {
|
| | | return list;
|
| | | }
|
| | |
|
| | | RevWalk rw = new RevWalk(repository);
|
| | |
| | | }
|
| | |
|
| | | /**
|
| | | * Returns the default HEAD for a repository. Normally returns the ref HEAD points to, but if HEAD points to nothing
|
| | | * it returns null.
|
| | | * Returns the target of the symbolic HEAD reference for a repository.
|
| | | * Normally returns a branch reference name, but when HEAD is detached,
|
| | | * the commit is matched against the known tags. The most recent matching
|
| | | * tag ref name will be returned if it references the HEAD commit. If
|
| | | * no match is found, the SHA1 is returned.
|
| | | *
|
| | | * @param repository
|
| | | * @return the refmodel for HEAD or null
|
| | | * @return the ref name or the SHA1 for a detached HEAD
|
| | | */
|
| | | public static RefModel getDefaultHead(Repository repository) {
|
| | | RefModel ref = null;
|
| | | public static String getHEADRef(Repository repository) {
|
| | | String target = null;
|
| | | try {
|
| | | Ref head = repository.getRef(Constants.HEAD);
|
| | | if (head != null) {
|
| | | Ref target = head.getTarget();
|
| | | RevWalk rw = new RevWalk(repository);
|
| | | ObjectId targetId = target.getObjectId();
|
| | | if (targetId != null) {
|
| | | RevObject object = rw.parseAny(targetId);
|
| | | ref = new RefModel(target.getName(), target, object);
|
| | | target = repository.getFullBranch();
|
| | | if (!target.startsWith(Constants.R_HEADS)) {
|
| | | // refers to an actual commit, probably a tag
|
| | | // find latest tag that matches the commit, if any
|
| | | List<RefModel> tagModels = getTags(repository, true, -1);
|
| | | if (tagModels.size() > 0) {
|
| | | RefModel tag = null;
|
| | | Date lastDate = new Date(0);
|
| | | for (RefModel tagModel : tagModels) {
|
| | | if (tagModel.getReferencedObjectId().getName().equals(target) &&
|
| | | tagModel.getDate().after(lastDate)) {
|
| | | tag = tagModel;
|
| | | lastDate = tag.getDate();
|
| | | }
|
| | | }
|
| | | target = tag.getName();
|
| | | }
|
| | | rw.dispose();
|
| | | }
|
| | | } catch (Throwable t) {
|
| | | LOGGER.error("Failed to get default head!", t);
|
| | | error(t, repository, "{0} failed to get symbolic HEAD target");
|
| | | }
|
| | | return ref;
|
| | | return target;
|
| | | }
|
| | |
|
| | | |
| | | /**
|
| | | * Sets the default HEAD symbolic ref for a repository.
|
| | | * Sets the symbolic ref HEAD to the specified target ref. The
|
| | | * HEAD will be detached if the target ref is not a branch.
|
| | | *
|
| | | * @param repository
|
| | | * @param ref
|
| | | * @param targetRef
|
| | | * @return true if successful
|
| | | */
|
| | | public static void setDefaultHead(Repository repository, Ref ref) {
|
| | | public static boolean setHEADtoRef(Repository repository, String targetRef) {
|
| | | try {
|
| | | boolean detach = !ref.getName().startsWith(Constants.R_HEADS); // detach if not a branch
|
| | | // detach HEAD if target ref is not a branch
|
| | | boolean detach = !targetRef.startsWith(Constants.R_HEADS);
|
| | | RefUpdate.Result result;
|
| | | RefUpdate head = repository.updateRef(Constants.HEAD, detach);
|
| | | if (detach) { // Tag
|
| | | RevCommit commit = getCommit(repository, ref.getObjectId().getName());
|
| | | RevCommit commit = getCommit(repository, targetRef);
|
| | | head.setNewObjectId(commit.getId());
|
| | | result = head.forceUpdate();
|
| | | } else {
|
| | | result = head.link(ref.getName());
|
| | | result = head.link(targetRef);
|
| | | }
|
| | | switch (result) {
|
| | | case NEW:
|
| | | case FORCED:
|
| | | case NO_CHANGE:
|
| | | case FAST_FORWARD:
|
| | | break;
|
| | | return true; |
| | | default:
|
| | | LOGGER.error(MessageFormat.format("{0} failed to set default head to {1} ({2})",
|
| | | repository.getDirectory().getAbsolutePath(), ref.getName(), result));
|
| | | LOGGER.error(MessageFormat.format("{0} HEAD update to {1} returned result {2}",
|
| | | repository.getDirectory().getAbsolutePath(), targetRef, result));
|
| | | }
|
| | | } catch (Throwable t) {
|
| | | error(t, repository, "{0} failed to set default head to {1}", ref.getName());
|
| | | error(t, repository, "{0} failed to set HEAD to {1}", targetRef);
|
| | | }
|
| | | return false;
|
| | | }
|
| | | |
| | | /**
|
| | | * Get the full branch and tag ref names for any potential HEAD targets.
|
| | | *
|
| | | * @param repository
|
| | | * @return a list of ref names
|
| | | */
|
| | | public static List<String> getAvailableHeadTargets(Repository repository) {
|
| | | List<String> targets = new ArrayList<String>();
|
| | | for (RefModel branchModel : JGitUtils.getLocalBranches(repository, true, -1)) {
|
| | | targets.add(branchModel.getName());
|
| | | }
|
| | |
|
| | | for (RefModel tagModel : JGitUtils.getTags(repository, true, -1)) {
|
| | | targets.add(tagModel.getName());
|
| | | }
|
| | | return targets;
|
| | | }
|
| | |
|
| | | /**
|
| | |
| | |
|
| | | // Create a tree object to reference from a commit
|
| | | TreeFormatter tree = new TreeFormatter();
|
| | | tree.append("NEWBRANCH", FileMode.REGULAR_FILE, blobId);
|
| | | tree.append(".branch", FileMode.REGULAR_FILE, blobId);
|
| | | ObjectId treeId = odi.insert(tree);
|
| | |
|
| | | // Create a commit object
|