annotate mercurial-server/src/jetbrains/buildServer/buildTriggers/vcs/mercurial/MercurialVcsSupport.java @ 121:e8125034f6a9

fix cleanup
author Pavel.Sher
date Fri, 24 Sep 2010 17:37:27 +0400
parents 8587a9c22d55
children 62fe3e69cee6 43dd4142b0f5
rev   line source
25
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
1 /*
95
6c1cff1f61cc copyright =>2010
Pavel.Sher
parents: 90
diff changeset
2 * Copyright 2000-2010 JetBrains s.r.o.
25
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
3 *
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
4 * Licensed under the Apache License, Version 2.0 (the "License");
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
5 * you may not use this file except in compliance with the License.
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
6 * You may obtain a copy of the License at
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
7 *
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
8 * http://www.apache.org/licenses/LICENSE-2.0
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
9 *
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
10 * Unless required by applicable law or agreed to in writing, software
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
11 * distributed under the License is distributed on an "AS IS" BASIS,
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
13 * See the License for the specific language governing permissions and
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
14 * limitations under the License.
7047f643747f license added, preamble added to source code
Pavel.Sher
parents: 23
diff changeset
15 */
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
16 package jetbrains.buildServer.buildTriggers.vcs.mercurial;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
17
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
18 import jetbrains.buildServer.BuildAgent;
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
19 import jetbrains.buildServer.Used;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
20 import jetbrains.buildServer.buildTriggers.vcs.AbstractVcsPropertiesProcessor;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
21 import jetbrains.buildServer.buildTriggers.vcs.mercurial.command.*;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
22 import jetbrains.buildServer.log.Loggers;
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
23 import jetbrains.buildServer.serverSide.*;
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
24 import jetbrains.buildServer.util.EventDispatcher;
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
25 import jetbrains.buildServer.util.FileUtil;
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
26 import jetbrains.buildServer.util.StringUtil;
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
27 import jetbrains.buildServer.util.filters.Filter;
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
28 import jetbrains.buildServer.util.filters.FilterUtil;
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
29 import jetbrains.buildServer.vcs.*;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
30 import jetbrains.buildServer.vcs.patches.PatchBuilder;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
31 import org.jetbrains.annotations.NotNull;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
32 import org.jetbrains.annotations.Nullable;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
33
1
56e7f34ca887 added TeamCity libs, fix tests
Pavel.Sher
parents: 0
diff changeset
34 import java.io.File;
56e7f34ca887 added TeamCity libs, fix tests
Pavel.Sher
parents: 0
diff changeset
35 import java.io.FileFilter;
56e7f34ca887 added TeamCity libs, fix tests
Pavel.Sher
parents: 0
diff changeset
36 import java.io.FileInputStream;
56e7f34ca887 added TeamCity libs, fix tests
Pavel.Sher
parents: 0
diff changeset
37 import java.io.IOException;
56e7f34ca887 added TeamCity libs, fix tests
Pavel.Sher
parents: 0
diff changeset
38 import java.util.*;
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
39 import java.util.concurrent.ConcurrentHashMap;
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
40 import java.util.concurrent.ConcurrentMap;
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
41 import java.util.concurrent.locks.Lock;
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
42 import java.util.concurrent.locks.ReentrantLock;
1
56e7f34ca887 added TeamCity libs, fix tests
Pavel.Sher
parents: 0
diff changeset
43
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
44 /**
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
45 * Mercurial VCS plugin for TeamCity works as follows:
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
46 * <ul>
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
47 * <li>clones repository to internal storage
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
48 * <li>before any operation with working copy of repository pulls changes from the original repository
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
49 * <li>executes corresponding hg command
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
50 * </ul>
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
51 *
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
52 * <p>Working copy of repository is created in the $TEAMCITY_DATA_PATH/system/caches/hg_&lt;hash code> folder.
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
53 * <p>Personal builds (remote runs) are not yet supported, they require corresponding functionality from the IDE.
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
54 */
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
55 public class MercurialVcsSupport extends ServerVcsSupport implements LabelingSupport, VcsFileContentProvider {
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
56 private ConcurrentMap<String, Lock> myWorkDirLocks= new ConcurrentHashMap<String, Lock>();
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
57 private VcsManager myVcsManager;
27
7944e8985ebd prepare modules structure for agent side checkout
Pavel.Sher
parents: 25
diff changeset
58 private File myDefaultWorkFolderParent;
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
59
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
60 public MercurialVcsSupport(@NotNull final VcsManager vcsManager,
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
61 @NotNull ServerPaths paths,
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
62 @NotNull final SBuildServer server,
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
63 @NotNull EventDispatcher<BuildServerListener> dispatcher) {
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
64 myVcsManager = vcsManager;
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
65 myDefaultWorkFolderParent = new File(paths.getCachesDir(), "mercurial");
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
66 dispatcher.addListener(new BuildServerAdapter() {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
67 @Override
121
e8125034f6a9 fix cleanup
Pavel.Sher
parents: 106
diff changeset
68 public void cleanupFinished() {
e8125034f6a9 fix cleanup
Pavel.Sher
parents: 106
diff changeset
69 super.cleanupFinished();
e8125034f6a9 fix cleanup
Pavel.Sher
parents: 106
diff changeset
70 server.getExecutor().submit(new Runnable() {
e8125034f6a9 fix cleanup
Pavel.Sher
parents: 106
diff changeset
71 public void run() {
e8125034f6a9 fix cleanup
Pavel.Sher
parents: 106
diff changeset
72 removeOldWorkFolders();
e8125034f6a9 fix cleanup
Pavel.Sher
parents: 106
diff changeset
73 }
e8125034f6a9 fix cleanup
Pavel.Sher
parents: 106
diff changeset
74 });
e8125034f6a9 fix cleanup
Pavel.Sher
parents: 106
diff changeset
75 }
e8125034f6a9 fix cleanup
Pavel.Sher
parents: 106
diff changeset
76
e8125034f6a9 fix cleanup
Pavel.Sher
parents: 106
diff changeset
77 @Override
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
78 public void sourcesVersionReleased(@NotNull final BuildAgent agent) {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
79 super.sourcesVersionReleased(agent);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
80 server.getExecutor().submit(new Runnable() {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
81 public void run() {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
82 Set<File> clonedRepos = getAllClonedRepos();
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
83 if (clonedRepos == null) return;
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
84 for (File f: clonedRepos) {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
85 lockWorkDir(f);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
86 try {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
87 FileUtil.delete(f);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
88 } finally {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
89 unlockWorkDir(f);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
90 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
91 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
92 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
93 });
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
94 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
95 });
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
96 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
97
90
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
98 private Collection<ModifiedFile> computeModifiedFilesForMergeCommit(final Settings settings, final ChangeSet cur) throws VcsException {
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
99 if (!cur.containsFiles()) return Collections.emptyList();
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
100
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
101 ChangedFilesCommand cfc = new ChangedFilesCommand(settings);
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
102 cfc.setRevId(cur.getId());
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
103 return cfc.execute();
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
104 }
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
105
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
106 private List<VcsChange> toVcsChanges(final List<ModifiedFile> modifiedFiles, String prevVer, String curVer, final IncludeRule includeRule) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
107 List<VcsChange> files = new ArrayList<VcsChange>();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
108 for (ModifiedFile mf: modifiedFiles) {
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
109 String normalizedPath = PathUtil.normalizeSeparator(mf.getPath());
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
110 if (!normalizedPath.startsWith(includeRule.getFrom())) continue; // skip files which do not match include rule
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
111 String relPath = StringUtil.removeLeadingSlash(normalizedPath.substring(includeRule.getFrom().length()));
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
112
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
113 VcsChangeInfo.Type changeType = getChangeType(mf.getStatus());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
114 if (changeType == null) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
115 Loggers.VCS.warn("Unable to convert status: " + mf.getStatus() + " to VCS change type");
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
116 changeType = VcsChangeInfo.Type.NOT_CHANGED;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
117 }
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
118 files.add(new VcsChange(changeType, mf.getStatus().getName(), normalizedPath, relPath, prevVer, curVer));
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
119 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
120 return files;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
121 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
122
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
123 private VcsChangeInfo.Type getChangeType(final ModifiedFile.Status status) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
124 switch (status) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
125 case ADDED:return VcsChangeInfo.Type.ADDED;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
126 case MODIFIED:return VcsChangeInfo.Type.CHANGED;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
127 case REMOVED:return VcsChangeInfo.Type.REMOVED;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
128 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
129 return null;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
130 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
131
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
132 @NotNull
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
133 public byte[] getContent(@NotNull final VcsModification vcsModification,
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
134 @NotNull final VcsChangeInfo change,
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
135 @NotNull final VcsChangeInfo.ContentType contentType,
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
136 @NotNull final VcsRoot vcsRoot) throws VcsException {
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
137 syncClonedRepository(vcsRoot);
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
138 String version = contentType == VcsChangeInfo.ContentType.AFTER_CHANGE ? change.getAfterChangeRevisionNumber() : change.getBeforeChangeRevisionNumber();
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
139 return getContent(change.getRelativeFileName(), vcsRoot, version);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
140 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
141
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
142 @NotNull
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
143 public byte[] getContent(@NotNull final String filePath, @NotNull final VcsRoot vcsRoot, @NotNull final String version) throws VcsException {
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
144 syncClonedRepository(vcsRoot);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
145 Settings settings = createSettings(vcsRoot);
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
146 CatCommand cc = new CatCommand(settings);
59
30cc10fe9479 accept version in two formats: with rev number and without
Pavel.Sher
parents: 58
diff changeset
147 cc.setRevId(new ChangeSet(version).getId());
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
148 File parentDir = cc.execute(Collections.singletonList(filePath));
30
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
149 try {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
150 File file = new File(parentDir, filePath);
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
151 if (file.isFile()) {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
152 try {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
153 return FileUtil.loadFileBytes(file);
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
154 } catch (IOException e) {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
155 throw new VcsException("Failed to load content of file: " + file.getAbsolutePath(), e);
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
156 }
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
157 } else {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
158 Loggers.VCS.warn("Unable to obtain content of the file: " + filePath);
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
159 }
30
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
160 } finally {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
161 FileUtil.delete(parentDir);
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
162 }
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
163 return new byte[0];
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
164 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
165
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
166 @NotNull
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
167 public String getName() {
28
a7cab5083ada libraries moved on top level, dummy implementation of agent side checkout interface
Pavel.Sher
parents: 27
diff changeset
168 return Constants.VCS_NAME;
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
169 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
170
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
171 @NotNull
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
172 @Used("jsp")
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
173 public String getDisplayName() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
174 return "Mercurial";
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
175 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
176
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
177 @Nullable
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
178 public PropertiesProcessor getVcsPropertiesProcessor() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
179 return new AbstractVcsPropertiesProcessor() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
180 public Collection<InvalidProperty> process(final Map<String, String> properties) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
181 List<InvalidProperty> result = new ArrayList<InvalidProperty>();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
182 if (isEmpty(properties.get(Constants.HG_COMMAND_PATH_PROP))) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
183 result.add(new InvalidProperty(Constants.HG_COMMAND_PATH_PROP, "Path to 'hg' command must be specified"));
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
184 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
185 if (isEmpty(properties.get(Constants.REPOSITORY_PROP))) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
186 result.add(new InvalidProperty(Constants.REPOSITORY_PROP, "Repository must be specified"));
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
187 }
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
188 if (isEmpty(properties.get(Constants.SERVER_CLONE_PATH_PROP))) {
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
189 properties.put(Constants.SERVER_CLONE_PATH_PROP, myDefaultWorkFolderParent.getAbsolutePath());
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
190 }
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
191 return result;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
192 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
193 };
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
194 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
195
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
196 @NotNull
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
197 public String getVcsSettingsJspFilePath() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
198 return "mercurialSettings.jsp";
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
199 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
200
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
201 @NotNull
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
202 public String getCurrentVersion(@NotNull final VcsRoot root) throws VcsException {
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
203 // we will return full version of the most recent change as current version
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
204 syncClonedRepository(root);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
205 Settings settings = createSettings(root);
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
206 BranchesCommand branches = new BranchesCommand(settings);
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
207 Map<String, ChangeSet> result = branches.execute();
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
208 if (!result.containsKey(settings.getBranchName())) {
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
209 throw new VcsException("Unable to find current version for the branch: " + settings.getBranchName());
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
210 }
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
211
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
212 return result.get(settings.getBranchName()).getFullVersion();
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
213 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
214
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
215 public boolean sourcesUpdatePossibleIfChangesNotFound(@NotNull final VcsRoot root) {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
216 return false;
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
217 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
218
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
219 @NotNull
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
220 public String describeVcsRoot(final VcsRoot vcsRoot) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
221 return "mercurial: " + vcsRoot.getProperty(Constants.REPOSITORY_PROP);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
222 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
223
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
224 @Override
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
225 public TestConnectionSupport getTestConnectionSupport() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
226 return new TestConnectionSupport() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
227 public String testConnection(@NotNull final VcsRoot vcsRoot) throws VcsException {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
228 Settings settings = createSettings(vcsRoot);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
229 IdentifyCommand id = new IdentifyCommand(settings);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
230 StringBuilder res = new StringBuilder();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
231 res.append(quoteIfNeeded(settings.getHgCommandPath()));
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
232 res.append(" identify ");
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
233 final String obfuscatedUrl = CommandUtil.removePrivateData(settings.getRepositoryUrl(), Collections.singleton(settings.getPassword()));
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
234 res.append(quoteIfNeeded(obfuscatedUrl));
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
235 res.append('\n').append(id.execute());
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
236 return res.toString();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
237 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
238 };
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
239 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
240
17
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
241 private String quoteIfNeeded(@NotNull String str) {
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
242 if (str.indexOf(' ') != -1) {
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
243 return "\"" + str + "\"";
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
244 }
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
245
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
246 return str;
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
247 }
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
248
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
249 @Nullable
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
250 public Map<String, String> getDefaultVcsProperties() {
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
251 Map<String, String> defaults = new HashMap<String, String>();
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
252 defaults.put(Constants.HG_COMMAND_PATH_PROP, "hg");
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
253 defaults.put(Constants.SERVER_CLONE_PATH_PROP, myDefaultWorkFolderParent.getAbsolutePath());
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
254 return defaults;
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
255 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
256
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
257 public String getVersionDisplayName(@NotNull final String version, @NotNull final VcsRoot root) throws VcsException {
59
30cc10fe9479 accept version in two formats: with rev number and without
Pavel.Sher
parents: 58
diff changeset
258 return new ChangeSet(version).getId();
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
259 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
260
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
261 @NotNull
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
262 public Comparator<String> getVersionComparator() {
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
263 // comparator is called when TeamCity needs to sort modifications in the order of their appearance,
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
264 // currently we sort changes by revision number, not sure however that this is a good idea,
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
265 // probably it would be better to sort them by timestamp (and to add timestamp into the version).
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
266 return new Comparator<String>() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
267 public int compare(final String o1, final String o2) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
268 try {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
269 return new ChangeSet(o1).getRevNumber() - new ChangeSet(o2).getRevNumber();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
270 } catch (Exception e) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
271 return 1;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
272 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
273 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
274 };
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
275 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
276
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
277 // builds patch from version to version
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
278 private void buildIncrementalPatch(final Settings settings, @NotNull final ChangeSet fromVer, @NotNull final ChangeSet toVer, final PatchBuilder builder, final CheckoutRules checkoutRules)
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
279 throws VcsException, IOException {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
280 StatusCommand st = new StatusCommand(settings);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
281 st.setFromRevId(fromVer.getId());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
282 st.setToRevId(toVer.getId());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
283 List<ModifiedFile> modifiedFiles = st.execute();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
284 List<String> notDeletedFiles = new ArrayList<String>();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
285 for (ModifiedFile f: modifiedFiles) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
286 if (f.getStatus() != ModifiedFile.Status.REMOVED) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
287 notDeletedFiles.add(f.getPath());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
288 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
289 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
290
47
c785bc4c5f39 minor fix: do not call cat command if there are no files to export
Pavel.Sher
parents: 46
diff changeset
291 if (notDeletedFiles.isEmpty()) return;
c785bc4c5f39 minor fix: do not call cat command if there are no files to export
Pavel.Sher
parents: 46
diff changeset
292
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
293 CatCommand cc = new CatCommand(settings);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
294 cc.setRevId(toVer.getId());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
295 File parentDir = cc.execute(notDeletedFiles);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
296
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
297 try {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
298 for (ModifiedFile f: modifiedFiles) {
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
299 String mappedPath = checkoutRules.map(f.getPath());
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
300 if (mappedPath == null) continue; // skip
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
301 final File virtualFile = new File(mappedPath);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
302 if (f.getStatus() == ModifiedFile.Status.REMOVED) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
303 builder.deleteFile(virtualFile, true);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
304 } else {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
305 File realFile = new File(parentDir, f.getPath());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
306 FileInputStream is = new FileInputStream(realFile);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
307 try {
11
568707240741 fix patch
Pavel.Sher
parents: 9
diff changeset
308 builder.changeOrCreateBinaryFile(virtualFile, null, is, realFile.length());
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
309 } finally {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
310 is.close();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
311 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
312 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
313 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
314 } finally {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
315 FileUtil.delete(parentDir);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
316 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
317 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
318
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
319 // builds patch by exporting files using specified version
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
320 private void buildFullPatch(final Settings settings, @NotNull final ChangeSet toVer, final PatchBuilder builder, final CheckoutRules checkoutRules)
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
321 throws IOException, VcsException {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
322 CloneCommand cl = new CloneCommand(settings);
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
323 // clone from the local repository
67
e6971dc6b17c always use url with credentials if username/password are specified
Pavel.Sher
parents: 62
diff changeset
324 cl.setRepository(settings.getLocalRepositoryDir().getAbsolutePath());
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
325 cl.setToId(toVer.getId());
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
326 cl.setUpdateWorkingDir(false);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
327 File tempDir = FileUtil.createTempDirectory("mercurial", toVer.getId());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
328 try {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
329 final File repRoot = new File(tempDir, "rep");
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
330 cl.setDestDir(repRoot.getAbsolutePath());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
331 cl.execute();
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
332
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
333 UpdateCommand up = new UpdateCommand(settings);
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
334 up.setWorkDirectory(repRoot.getAbsolutePath());
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
335 up.setToId(toVer.getId());
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
336 up.execute();
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
337
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
338 buildPatchFromDirectory(builder, repRoot, new FileFilter() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
339 public boolean accept(final File file) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
340 return !(file.isDirectory() && ".hg".equals(file.getName()));
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
341 }
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
342 }, checkoutRules);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
343 } finally {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
344 FileUtil.delete(tempDir);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
345 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
346 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
347
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
348 private void buildPatchFromDirectory(final PatchBuilder builder, final File repRoot, final FileFilter filter, final CheckoutRules checkoutRules) throws IOException {
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
349 buildPatchFromDirectory(repRoot, builder, repRoot, filter, checkoutRules);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
350 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
351
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
352 private void buildPatchFromDirectory(File curDir, final PatchBuilder builder, final File repRoot, final FileFilter filter, final CheckoutRules checkoutRules) throws IOException {
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
353 File[] files = curDir.listFiles(filter);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
354 if (files != null) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
355 for (File realFile: files) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
356 String relPath = realFile.getAbsolutePath().substring(repRoot.getAbsolutePath().length());
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
357 String mappedPath = checkoutRules.map(relPath);
97
3ac056b1932e TW-10928
Pavel.Sher
parents: 95
diff changeset
358 if (mappedPath != null && mappedPath.length() > 0) {
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
359 final File virtualFile = new File(mappedPath);
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
360 if (realFile.isDirectory()) {
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
361 builder.createDirectory(virtualFile);
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
362 buildPatchFromDirectory(realFile, builder, repRoot, filter, checkoutRules);
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
363 } else {
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
364 final FileInputStream is = new FileInputStream(realFile);
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
365 try {
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
366 builder.createBinaryFile(virtualFile, null, is, realFile.length());
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
367 } finally {
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
368 is.close();
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
369 }
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
370 }
97
3ac056b1932e TW-10928
Pavel.Sher
parents: 95
diff changeset
371 } else {
3ac056b1932e TW-10928
Pavel.Sher
parents: 95
diff changeset
372 if (realFile.isDirectory()) {
3ac056b1932e TW-10928
Pavel.Sher
parents: 95
diff changeset
373 buildPatchFromDirectory(realFile, builder, repRoot, filter, checkoutRules);
3ac056b1932e TW-10928
Pavel.Sher
parents: 95
diff changeset
374 }
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
375 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
376 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
377 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
378 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
379
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
380 // updates current working copy of repository by pulling changes from the repository specified in VCS root
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
381 private void syncClonedRepository(final VcsRoot root) throws VcsException {
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
382 Settings settings = createSettings(root);
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
383 File workDir = settings.getLocalRepositoryDir();
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
384 lockWorkDir(workDir);
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
385 try {
29
798e750e4f26 first version of agent side checkout
Pavel.Sher
parents: 28
diff changeset
386 if (settings.hasCopyOfRepository()) {
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
387 // update
8
2cb2df5a0dcd fix working directory update (use pull command)
Pavel.Sher
parents: 1
diff changeset
388 PullCommand pull = new PullCommand(settings);
2cb2df5a0dcd fix working directory update (use pull command)
Pavel.Sher
parents: 1
diff changeset
389 pull.execute();
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
390 } else {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
391 // clone
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
392 CloneCommand cl = new CloneCommand(settings);
22
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
393 cl.setDestDir(workDir.getAbsolutePath());
18
d787c696225c do not update local working directory
Pavel.Sher
parents: 17
diff changeset
394 cl.setUpdateWorkingDir(false);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
395 cl.execute();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
396 }
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
397 } finally {
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
398 unlockWorkDir(workDir);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
399 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
400 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
401
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
402 @Override
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
403 public LabelingSupport getLabelingSupport() {
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
404 return this;
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
405 }
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
406
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
407 @NotNull
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
408 public VcsFileContentProvider getContentProvider() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
409 return this;
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
410 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
411
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
412 @NotNull
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
413 public CollectChangesPolicy getCollectChangesPolicy() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
414 return new CollectChangesByIncludeRules() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
415 @NotNull
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
416 public IncludeRuleChangeCollector getChangeCollector(@NotNull final VcsRoot root, @NotNull final String fromVersion, @Nullable final String currentVersion) throws VcsException {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
417 return new IncludeRuleChangeCollector() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
418 @NotNull
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
419 public List<ModificationData> collectChanges(@NotNull final IncludeRule includeRule) throws VcsException {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
420 syncClonedRepository(root);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
421
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
422 // first obtain changes between specified versions
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
423 List<ModificationData> result = new ArrayList<ModificationData>();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
424 if (currentVersion == null) return result;
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
425
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
426 Settings settings = createSettings(root);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
427 LogCommand lc = new LogCommand(settings);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
428 String fromId = new ChangeSetRevision(fromVersion).getId();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
429 lc.setFromRevId(fromId);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
430 lc.setToRevId(new ChangeSetRevision(currentVersion).getId());
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
431 List<ChangeSet> changeSets = lc.execute();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
432 if (changeSets.isEmpty()) {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
433 return result;
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
434 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
435
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
436 // invoke status command for each changeset and determine what files were modified in these changesets
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
437 StatusCommand st = new StatusCommand(settings);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
438 ChangeSet prev = new ChangeSet(fromVersion);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
439 for (ChangeSet cur : changeSets) {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
440 if (cur.getId().equals(fromId)) continue; // skip already reported changeset
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
441
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
442 String prevId = prev.getId();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
443 List<ChangeSetRevision> curParents = cur.getParents();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
444 boolean merge = curParents != null && curParents.size() > 1;
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
445 if (curParents != null && !merge) {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
446 prevId = curParents.get(0).getId();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
447 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
448
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
449 List<ModifiedFile> modifiedFiles = new ArrayList<ModifiedFile>();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
450 if (merge) {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
451 modifiedFiles.addAll(computeModifiedFilesForMergeCommit(settings, cur));
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
452 } else {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
453 st.setFromRevId(prevId);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
454 st.setToRevId(cur.getId());
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
455 modifiedFiles = st.execute();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
456 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
457
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
458 // changeset full version will be set into VcsChange structure and
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
459 // stored in database (note that getContent method will be invoked with this version)
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
460 List<VcsChange> files = toVcsChanges(modifiedFiles, prev.getFullVersion(), cur.getFullVersion(), includeRule);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
461 if (files.isEmpty() && !merge) continue;
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
462 ModificationData md = new ModificationData(cur.getTimestamp(), files, cur.getDescription(), cur.getUser(), root, cur.getFullVersion(), cur.getId());
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
463 if (merge) {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
464 md.setCanBeIgnored(false);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
465 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
466 result.add(md);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
467 prev = cur;
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
468 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
469
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
470 return result;
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
471 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
472
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
473 public void dispose() throws VcsException {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
474 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
475 };
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
476 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
477 };
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
478 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
479
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
480 @NotNull
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
481 public BuildPatchPolicy getBuildPatchPolicy() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
482 return new BuildPatchByCheckoutRules() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
483 public void buildPatch(@NotNull final VcsRoot root,
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
484 @Nullable final String fromVersion,
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
485 @NotNull final String toVersion,
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
486 @NotNull final PatchBuilder builder,
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
487 @NotNull final CheckoutRules checkoutRules) throws IOException, VcsException {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
488 syncClonedRepository(root);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
489 Settings settings = createSettings(root);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
490 if (fromVersion == null) {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
491 buildFullPatch(settings, new ChangeSet(toVersion), builder, checkoutRules);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
492 } else {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
493 buildIncrementalPatch(settings, new ChangeSet(fromVersion), new ChangeSet(toVersion), builder, checkoutRules);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
494 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
495 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
496 };
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
497 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
498
22
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
499 private void lockWorkDir(@NotNull File workDir) {
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
500 getWorkDirLock(workDir).lock();
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
501 }
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
502
22
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
503 private void unlockWorkDir(@NotNull File workDir) {
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
504 getWorkDirLock(workDir).unlock();
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
505 }
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
506
31
1c11478f515b disable cache
Pavel.Sher
parents: 30
diff changeset
507 @Override
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
508 public boolean allowSourceCaching() {
31
1c11478f515b disable cache
Pavel.Sher
parents: 30
diff changeset
509 // since a copy of repository for each VCS root is already stored on disk
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
510 // we do not need separate cache for our patches
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
511 return false;
31
1c11478f515b disable cache
Pavel.Sher
parents: 30
diff changeset
512 }
1c11478f515b disable cache
Pavel.Sher
parents: 30
diff changeset
513
22
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
514 private Lock getWorkDirLock(final File workDir) {
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
515 String path = workDir.getAbsolutePath();
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
516 Lock lock = myWorkDirLocks.get(path);
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
517 if (lock == null) {
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
518 lock = new ReentrantLock();
22
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
519 Lock curLock = myWorkDirLocks.putIfAbsent(path, lock);
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
520 if (curLock != null) {
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
521 lock = curLock;
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
522 }
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
523 }
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
524 return lock;
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
525 }
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
526
23
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
527 private void removeOldWorkFolders() {
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
528 Set<File> workDirs = getAllClonedRepos();
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
529 if (workDirs == null) return;
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
530
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
531 for (VcsRoot vcsRoot: getMercurialVcsRoots()) {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
532 try {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
533 Settings s = createSettings(vcsRoot);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
534 workDirs.remove(PathUtil.getCanonicalFile(s.getLocalRepositoryDir()));
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
535 } catch (VcsException e) {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
536 Loggers.VCS.error(e);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
537 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
538 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
539
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
540 for (File f: workDirs) {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
541 lockWorkDir(f);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
542 try {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
543 FileUtil.delete(f);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
544 } finally {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
545 unlockWorkDir(f);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
546 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
547 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
548 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
549
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
550 private Collection<VcsRoot> getMercurialVcsRoots() {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
551 List<VcsRoot> res = new ArrayList<VcsRoot>(myVcsManager.getAllRegisteredVcsRoots());
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
552 FilterUtil.filterCollection(res, new Filter<VcsRoot>() {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
553 public boolean accept(@NotNull final VcsRoot data) {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
554 return getName().equals(data.getVcsName());
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
555 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
556 });
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
557 return res;
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
558 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
559
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
560 @Nullable
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
561 private Set<File> getAllClonedRepos() {
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
562 File workFoldersParent = myDefaultWorkFolderParent;
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
563 if (!workFoldersParent.isDirectory()) return null;
23
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
564
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
565 Set<File> workDirs = new HashSet<File>();
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
566 File[] files = workFoldersParent.listFiles(new FileFilter() {
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
567 public boolean accept(final File file) {
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
568 return file.isDirectory() && file.getName().startsWith(Settings.DEFAULT_WORK_DIR_PREFIX);
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
569 }
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
570 });
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
571 if (files != null) {
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
572 for (File f: files) {
46
13d5f0f56e70 compatibility with TeamCity 3.x
Pavel.Sher
parents: 44
diff changeset
573 workDirs.add(PathUtil.getCanonicalFile(f));
23
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
574 }
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
575 }
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
576 return workDirs;
23
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
577 }
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
578
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
579 public String label(@NotNull String label, @NotNull String version, @NotNull VcsRoot root, @NotNull CheckoutRules checkoutRules) throws VcsException {
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
580 syncClonedRepository(root);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
581
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
582 Settings settings = createSettings(root);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
583
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
584 // I do not know why but hg tag does not work correctly if
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
585 // update command was not invoked for the current repo
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
586 // in such case if there were no tags before Mercurial attempts to
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
587 // create new head when tag is pushed to the parent repository
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
588 UpdateCommand uc = new UpdateCommand(settings);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
589 uc.execute();
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
590
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
591 String fixedTagname = fixTagName(label);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
592 TagCommand tc = new TagCommand(settings);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
593 tc.setRevId(new ChangeSet(version).getId());
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
594 tc.setTag(fixedTagname);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
595 tc.execute();
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
596
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
597 PushCommand pc = new PushCommand(settings);
101
95c28c14b26e TW-10075
Pavel.Sher
parents: 97
diff changeset
598 // pc.setForce(true);
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
599 pc.execute();
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
600 return fixedTagname;
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
601 }
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
602
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
603 private String fixTagName(final String label) {
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
604 // according to Mercurial documentation http://hgbook.red-bean.com/hgbookch8.html#x12-1570008
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
605 // tag name must not contain:
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
606 // Colon (ASCII 58, “:”)
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
607 // Carriage return (ASCII 13, “\r”)
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
608 // Newline (ASCII 10, “\n”)
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
609 // all these characters will be replaced with _ (underscore)
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
610 return label.replace(':', '_').replace('\r', '_').replace('\n', '_');
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
611 }
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
612
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
613 private Settings createSettings(final VcsRoot root) throws VcsException {
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
614 Settings settings = new Settings(myDefaultWorkFolderParent, root);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
615 String customClonePath = root.getProperty(Constants.SERVER_CLONE_PATH_PROP);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
616 if (!StringUtil.isEmptyOrSpaces(customClonePath) && !myDefaultWorkFolderParent.equals(new File(customClonePath).getAbsoluteFile())) {
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
617 File parentDir = new File(customClonePath);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
618 createClonedRepositoryParentDir(parentDir);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
619
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
620 // take last part of repository path
67
e6971dc6b17c always use url with credentials if username/password are specified
Pavel.Sher
parents: 62
diff changeset
621 String repPath = settings.getRepositoryUrl();
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
622 String[] splitted = repPath.split("[/\\\\]");
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
623 if (splitted.length > 0) {
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
624 repPath = splitted[splitted.length-1];
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
625 }
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
626
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
627 File customWorkingDir = new File(parentDir, repPath);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
628 settings.setWorkingDir(customWorkingDir);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
629 } else {
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
630 createClonedRepositoryParentDir(myDefaultWorkFolderParent);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
631 }
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
632 return settings;
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
633 }
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
634
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
635 private void createClonedRepositoryParentDir(final File parentDir) throws VcsException {
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
636 if (!parentDir.exists() && !parentDir.mkdirs()) {
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
637 throw new VcsException("Failed to create parent directory for cloned repository: " + parentDir.getAbsolutePath());
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
638 }
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
639 }
48
8665bfa3e03a enable agent side checkout
Pavel.Sher
parents: 47
diff changeset
640
8665bfa3e03a enable agent side checkout
Pavel.Sher
parents: 47
diff changeset
641 public boolean isAgentSideCheckoutAvailable() {
8665bfa3e03a enable agent side checkout
Pavel.Sher
parents: 47
diff changeset
642 return true;
8665bfa3e03a enable agent side checkout
Pavel.Sher
parents: 47
diff changeset
643 }
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
644 }