annotate mercurial-server/src/jetbrains/buildServer/buildTriggers/vcs/mercurial/MercurialVcsSupport.java @ 144:2e90ef872b68

Collect changes using CheckoutRules. We built patches with CheckoutRules anyway. Also BranchSupport interface currently allows collecting changes between 2 roots only using CheckoutRules. And this will have better performance.
author Dmitry Neverov <dmitry.neverov@jetbrains.com>
date Tue, 11 Jan 2011 13:05:31 +0300
parents 2ab0bce4bf64
children 3a8af53dea6b 403d16397f5c
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 ChangedFilesCommand cfc = new ChangedFilesCommand(settings);
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
100 cfc.setRevId(cur.getId());
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
101 return cfc.execute();
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
102 }
5d9c34cb543a improving merge commits reporting
Pavel.Sher
parents: 86
diff changeset
103
144
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
104 private List<VcsChange> toVcsChanges(final List<ModifiedFile> modifiedFiles, String prevVer, String curVer) {
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
105 List<VcsChange> files = new ArrayList<VcsChange>();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
106 for (ModifiedFile mf: modifiedFiles) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
107 VcsChangeInfo.Type changeType = getChangeType(mf.getStatus());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
108 if (changeType == null) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
109 Loggers.VCS.warn("Unable to convert status: " + mf.getStatus() + " to VCS change type");
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
110 changeType = VcsChangeInfo.Type.NOT_CHANGED;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
111 }
144
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
112 files.add(new VcsChange(changeType, mf.getStatus().getName(), mf.getPath(), mf.getPath(), prevVer, curVer));
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
113 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
114 return files;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
115 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
116
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
117 private VcsChangeInfo.Type getChangeType(final ModifiedFile.Status status) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
118 switch (status) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
119 case ADDED:return VcsChangeInfo.Type.ADDED;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
120 case MODIFIED:return VcsChangeInfo.Type.CHANGED;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
121 case REMOVED:return VcsChangeInfo.Type.REMOVED;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
122 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
123 return null;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
124 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
125
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
126 @NotNull
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
127 public byte[] getContent(@NotNull final VcsModification vcsModification,
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
128 @NotNull final VcsChangeInfo change,
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
129 @NotNull final VcsChangeInfo.ContentType contentType,
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
130 @NotNull final VcsRoot vcsRoot) throws VcsException {
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
131 syncClonedRepository(vcsRoot);
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
132 String version = contentType == VcsChangeInfo.ContentType.AFTER_CHANGE ? change.getAfterChangeRevisionNumber() : change.getBeforeChangeRevisionNumber();
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
133 return getContent(change.getRelativeFileName(), vcsRoot, version);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
134 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
135
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
136 @NotNull
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
137 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
138 syncClonedRepository(vcsRoot);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
139 Settings settings = createSettings(vcsRoot);
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
140 CatCommand cc = new CatCommand(settings);
59
30cc10fe9479 accept version in two formats: with rev number and without
Pavel.Sher
parents: 58
diff changeset
141 cc.setRevId(new ChangeSet(version).getId());
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
142 File parentDir = cc.execute(Collections.singletonList(filePath));
30
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
143 try {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
144 File file = new File(parentDir, filePath);
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
145 if (file.isFile()) {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
146 try {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
147 return FileUtil.loadFileBytes(file);
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
148 } catch (IOException e) {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
149 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
150 }
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
151 } else {
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
152 Loggers.VCS.warn("Unable to obtain content of the file: " + filePath);
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
153 }
30
007c63ae45b0 delete temp dir after getContent call
Pavel.Sher
parents: 29
diff changeset
154 } finally {
118
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
155 deleteTmpDir(parentDir);
12
6989ac0f8cac getContent supported
Pavel.Sher
parents: 11
diff changeset
156 }
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
157 return new byte[0];
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
158 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
159
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
160 @NotNull
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
161 public String getName() {
28
a7cab5083ada libraries moved on top level, dummy implementation of agent side checkout interface
Pavel.Sher
parents: 27
diff changeset
162 return Constants.VCS_NAME;
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
163 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
164
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
165 @NotNull
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
166 @Used("jsp")
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
167 public String getDisplayName() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
168 return "Mercurial";
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
169 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
170
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
171 @Nullable
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
172 public PropertiesProcessor getVcsPropertiesProcessor() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
173 return new AbstractVcsPropertiesProcessor() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
174 public Collection<InvalidProperty> process(final Map<String, String> properties) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
175 List<InvalidProperty> result = new ArrayList<InvalidProperty>();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
176 if (isEmpty(properties.get(Constants.HG_COMMAND_PATH_PROP))) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
177 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
178 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
179 if (isEmpty(properties.get(Constants.REPOSITORY_PROP))) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
180 result.add(new InvalidProperty(Constants.REPOSITORY_PROP, "Repository must be specified"));
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
181 }
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
182 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
183 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
184 }
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
185 return result;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
186 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
187 };
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
188 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
189
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
190 @NotNull
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
191 public String getVcsSettingsJspFilePath() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
192 return "mercurialSettings.jsp";
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 @NotNull
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
196 public String getCurrentVersion(@NotNull final VcsRoot root) throws VcsException {
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
197 // we will return full version of the most recent change as current version
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
198 syncClonedRepository(root);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
199 Settings settings = createSettings(root);
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
200 BranchesCommand branches = new BranchesCommand(settings);
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
201 Map<String, ChangeSet> result = branches.execute();
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
202 if (!result.containsKey(settings.getBranchName())) {
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
203 throw new VcsException("Unable to find current version for the branch: " + settings.getBranchName());
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
204 }
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
205
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
206 return result.get(settings.getBranchName()).getFullVersion();
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
207 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
208
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
209 public boolean sourcesUpdatePossibleIfChangesNotFound(@NotNull final VcsRoot root) {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
210 return false;
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
211 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
212
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
213 @NotNull
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
214 public String describeVcsRoot(final VcsRoot vcsRoot) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
215 return "mercurial: " + vcsRoot.getProperty(Constants.REPOSITORY_PROP);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
216 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
217
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
218 @Override
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
219 public TestConnectionSupport getTestConnectionSupport() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
220 return new TestConnectionSupport() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
221 public String testConnection(@NotNull final VcsRoot vcsRoot) throws VcsException {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
222 Settings settings = createSettings(vcsRoot);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
223 IdentifyCommand id = new IdentifyCommand(settings);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
224 StringBuilder res = new StringBuilder();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
225 res.append(quoteIfNeeded(settings.getHgCommandPath()));
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
226 res.append(" identify ");
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
227 final String obfuscatedUrl = CommandUtil.removePrivateData(settings.getRepositoryUrl(), Collections.singleton(settings.getPassword()));
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
228 res.append(quoteIfNeeded(obfuscatedUrl));
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
229 res.append('\n').append(id.execute());
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
230 return res.toString();
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
231 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
232 };
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
233 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
234
17
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
235 private String quoteIfNeeded(@NotNull String str) {
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
236 if (str.indexOf(' ') != -1) {
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
237 return "\"" + str + "\"";
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
238 }
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
239
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
240 return str;
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
241 }
21b5b1c5dd74 test connection minor fix
Pavel.Sher
parents: 16
diff changeset
242
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
243 @Nullable
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
244 public Map<String, String> getDefaultVcsProperties() {
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
245 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
246 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
247 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
248 return defaults;
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
249 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
250
86
948a35b430e1 report parent changesets
Pavel.Sher
parents: 79
diff changeset
251 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
252 return new ChangeSet(version).getId();
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
253 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
254
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
255 @NotNull
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
256 public Comparator<String> getVersionComparator() {
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
257 // 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
258 // 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
259 // 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
260 return new Comparator<String>() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
261 public int compare(final String o1, final String o2) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
262 try {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
263 return new ChangeSet(o1).getRevNumber() - new ChangeSet(o2).getRevNumber();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
264 } catch (Exception e) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
265 return 1;
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
266 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
267 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
268 };
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
269 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
270
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
271 // builds patch from version to version
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
272 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
273 throws VcsException, IOException {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
274 StatusCommand st = new StatusCommand(settings);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
275 st.setFromRevId(fromVer.getId());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
276 st.setToRevId(toVer.getId());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
277 List<ModifiedFile> modifiedFiles = st.execute();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
278 List<String> notDeletedFiles = new ArrayList<String>();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
279 for (ModifiedFile f: modifiedFiles) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
280 if (f.getStatus() != ModifiedFile.Status.REMOVED) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
281 notDeletedFiles.add(f.getPath());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
282 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
283 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
284
47
c785bc4c5f39 minor fix: do not call cat command if there are no files to export
Pavel.Sher
parents: 46
diff changeset
285 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
286
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
287 CatCommand cc = new CatCommand(settings);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
288 cc.setRevId(toVer.getId());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
289 File parentDir = cc.execute(notDeletedFiles);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
290
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
291 try {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
292 for (ModifiedFile f: modifiedFiles) {
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
293 String mappedPath = checkoutRules.map(f.getPath());
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
294 if (mappedPath == null) continue; // skip
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
295 final File virtualFile = new File(mappedPath);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
296 if (f.getStatus() == ModifiedFile.Status.REMOVED) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
297 builder.deleteFile(virtualFile, true);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
298 } else {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
299 File realFile = new File(parentDir, f.getPath());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
300 FileInputStream is = new FileInputStream(realFile);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
301 try {
11
568707240741 fix patch
Pavel.Sher
parents: 9
diff changeset
302 builder.changeOrCreateBinaryFile(virtualFile, null, is, realFile.length());
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
303 } finally {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
304 is.close();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
305 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
306 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
307 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
308 } finally {
118
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
309 deleteTmpDir(parentDir);
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
310 }
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
311 }
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
312
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
313 private void deleteTmpDir(File parentDir) {
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
314 boolean dirDeleted = FileUtil.delete(parentDir);
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
315 if (!dirDeleted) {
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
316 Loggers.VCS.warn("Can not delete directory \"" + parentDir.getAbsolutePath() + "\"");
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
317 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
318 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
319
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
320 // builds patch by exporting files using specified version
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
321 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
322 throws IOException, VcsException {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
323 CloneCommand cl = new CloneCommand(settings);
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
324 // clone from the local repository
67
e6971dc6b17c always use url with credentials if username/password are specified
Pavel.Sher
parents: 62
diff changeset
325 cl.setRepository(settings.getLocalRepositoryDir().getAbsolutePath());
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
326 cl.setToId(toVer.getId());
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
327 cl.setUpdateWorkingDir(false);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
328 File tempDir = FileUtil.createTempDirectory("mercurial", toVer.getId());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
329 try {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
330 final File repRoot = new File(tempDir, "rep");
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
331 cl.setDestDir(repRoot.getAbsolutePath());
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
332 cl.execute();
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
333
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
334 UpdateCommand up = new UpdateCommand(settings);
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
335 up.setWorkDirectory(repRoot.getAbsolutePath());
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
336 up.setToId(toVer.getId());
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
337 up.execute();
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
338
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
339 buildPatchFromDirectory(builder, repRoot, new FileFilter() {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
340 public boolean accept(final File file) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
341 return !(file.isDirectory() && ".hg".equals(file.getName()));
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
342 }
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
343 }, checkoutRules);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
344 } finally {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
345 FileUtil.delete(tempDir);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
346 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
347 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
348
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
349 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
350 buildPatchFromDirectory(repRoot, builder, repRoot, filter, checkoutRules);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
351 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
352
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
353 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
354 File[] files = curDir.listFiles(filter);
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
355 if (files != null) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
356 for (File realFile: files) {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
357 String relPath = realFile.getAbsolutePath().substring(repRoot.getAbsolutePath().length());
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
358 String mappedPath = checkoutRules.map(relPath);
97
3ac056b1932e TW-10928
Pavel.Sher
parents: 95
diff changeset
359 if (mappedPath != null && mappedPath.length() > 0) {
79
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
360 final File virtualFile = new File(mappedPath);
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
361 if (realFile.isDirectory()) {
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
362 builder.createDirectory(virtualFile);
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
363 buildPatchFromDirectory(realFile, builder, repRoot, filter, checkoutRules);
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
364 } else {
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
365 final FileInputStream is = new FileInputStream(realFile);
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
366 try {
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
367 builder.createBinaryFile(virtualFile, null, is, realFile.length());
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
368 } finally {
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
369 is.close();
2e57d4e3f7b7 support checkout rules for server side patch
Pavel.Sher
parents: 71
diff changeset
370 }
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
371 }
97
3ac056b1932e TW-10928
Pavel.Sher
parents: 95
diff changeset
372 } else {
3ac056b1932e TW-10928
Pavel.Sher
parents: 95
diff changeset
373 if (realFile.isDirectory()) {
3ac056b1932e TW-10928
Pavel.Sher
parents: 95
diff changeset
374 buildPatchFromDirectory(realFile, builder, repRoot, filter, checkoutRules);
3ac056b1932e TW-10928
Pavel.Sher
parents: 95
diff changeset
375 }
0
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 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
380
19
40b2cf04cd4b some comments added
Pavel.Sher
parents: 18
diff changeset
381 // 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
382 private void syncClonedRepository(final VcsRoot root) throws VcsException {
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
383 Settings settings = createSettings(root);
57
99e757f2527b branches support
Pavel.Sher
parents: 56
diff changeset
384 File workDir = settings.getLocalRepositoryDir();
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
385 lockWorkDir(workDir);
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
386 try {
29
798e750e4f26 first version of agent side checkout
Pavel.Sher
parents: 28
diff changeset
387 if (settings.hasCopyOfRepository()) {
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
388 // update
8
2cb2df5a0dcd fix working directory update (use pull command)
Pavel.Sher
parents: 1
diff changeset
389 PullCommand pull = new PullCommand(settings);
2cb2df5a0dcd fix working directory update (use pull command)
Pavel.Sher
parents: 1
diff changeset
390 pull.execute();
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
391 } else {
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
392 // clone
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
393 CloneCommand cl = new CloneCommand(settings);
22
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
394 cl.setDestDir(workDir.getAbsolutePath());
18
d787c696225c do not update local working directory
Pavel.Sher
parents: 17
diff changeset
395 cl.setUpdateWorkingDir(false);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
396 cl.execute();
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
397 }
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
398 } finally {
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
399 unlockWorkDir(workDir);
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
400 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
401 }
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
402
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
403 @Override
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
404 public LabelingSupport getLabelingSupport() {
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
405 return this;
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
406 }
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
407
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
408 @NotNull
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
409 public VcsFileContentProvider getContentProvider() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
410 return this;
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
411 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
412
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
413 @NotNull
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
414 public CollectChangesPolicy getCollectChangesPolicy() {
144
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
415 return new CollectChangesByCheckoutRules() {
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
416 public List<ModificationData> collectChanges(@NotNull VcsRoot root, @NotNull String fromVersion, @Nullable String currentVersion, @NotNull CheckoutRules checkoutRules) throws VcsException {
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
417 syncClonedRepository(root);
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
418
144
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
419 // first obtain changes between specified versions
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
420 List<ModificationData> result = new ArrayList<ModificationData>();
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
421 if (currentVersion == null) return result;
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
422
144
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
423 Settings settings = createSettings(root);
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
424 LogCommand lc = new LogCommand(settings);
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
425 String fromId = new ChangeSetRevision(fromVersion).getId();
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
426 lc.setFromRevId(fromId);
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
427 lc.setToRevId(new ChangeSetRevision(currentVersion).getId());
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
428 List<ChangeSet> changeSets = lc.execute();
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
429 if (changeSets.isEmpty()) {
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
430 return result;
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
431 }
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
432
144
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
433 // invoke status command for each changeset and determine what files were modified in these changesets
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
434 StatusCommand st = new StatusCommand(settings);
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
435 ChangeSet prev = new ChangeSet(fromVersion);
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
436 for (ChangeSet cur : changeSets) {
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
437 if (cur.getId().equals(fromId)) continue; // skip already reported changeset
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
438
144
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
439 String prevId = prev.getId();
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
440 List<ChangeSetRevision> curParents = cur.getParents();
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
441 boolean merge = curParents != null && curParents.size() > 1;
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
442 if (curParents != null && !merge) {
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
443 prevId = curParents.get(0).getId();
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
444 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
445
144
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
446 List<ModifiedFile> modifiedFiles = new ArrayList<ModifiedFile>();
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
447 if (merge) {
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
448 modifiedFiles.addAll(computeModifiedFilesForMergeCommit(settings, cur));
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
449 } else {
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
450 st.setFromRevId(prevId);
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
451 st.setToRevId(cur.getId());
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
452 modifiedFiles = st.execute();
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
453 }
144
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
454
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
455 // changeset full version will be set into VcsChange structure and
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
456 // stored in database (note that getContent method will be invoked with this version)
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
457 List<VcsChange> files = toVcsChanges(modifiedFiles, prev.getFullVersion(), cur.getFullVersion());
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
458 if (files.isEmpty() && !merge) continue;
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
459 ModificationData md = new ModificationData(cur.getTimestamp(), files, cur.getDescription(), cur.getUser(), root, cur.getFullVersion(), cur.getId());
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
460 if (merge) {
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
461 md.setCanBeIgnored(false);
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
462 }
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
463 result.add(md);
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
464 prev = cur;
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
465 }
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
466
2e90ef872b68 Collect changes using CheckoutRules.
Dmitry Neverov <dmitry.neverov@jetbrains.com>
parents: 142
diff changeset
467 return result;
106
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 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
471
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
472 @NotNull
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
473 public BuildPatchPolicy getBuildPatchPolicy() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
474 return new BuildPatchByCheckoutRules() {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
475 public void buildPatch(@NotNull final VcsRoot root,
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
476 @Nullable final String fromVersion,
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
477 @NotNull final String toVersion,
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
478 @NotNull final PatchBuilder builder,
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
479 @NotNull final CheckoutRules checkoutRules) throws IOException, VcsException {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
480 syncClonedRepository(root);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
481 Settings settings = createSettings(root);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
482 if (fromVersion == null) {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
483 buildFullPatch(settings, new ChangeSet(toVersion), builder, checkoutRules);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
484 } else {
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
485 buildIncrementalPatch(settings, new ChangeSet(fromVersion), new ChangeSet(toVersion), builder, checkoutRules);
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
486 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
487 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
488 };
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
489 }
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
490
22
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
491 private void lockWorkDir(@NotNull File workDir) {
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
492 getWorkDirLock(workDir).lock();
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
493 }
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
494
22
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
495 private void unlockWorkDir(@NotNull File workDir) {
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
496 getWorkDirLock(workDir).unlock();
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
497 }
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
498
31
1c11478f515b disable cache
Pavel.Sher
parents: 30
diff changeset
499 @Override
106
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
500 public boolean allowSourceCaching() {
31
1c11478f515b disable cache
Pavel.Sher
parents: 30
diff changeset
501 // 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
502 // we do not need separate cache for our patches
8587a9c22d55 switch to new API
Pavel.Sher
parents: 103
diff changeset
503 return false;
31
1c11478f515b disable cache
Pavel.Sher
parents: 30
diff changeset
504 }
1c11478f515b disable cache
Pavel.Sher
parents: 30
diff changeset
505
22
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
506 private Lock getWorkDirLock(final File workDir) {
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
507 String path = workDir.getAbsolutePath();
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
508 Lock lock = myWorkDirLocks.get(path);
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
509 if (lock == null) {
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
510 lock = new ReentrantLock();
22
0d6f27953b30 synchronization fixes
Pavel.Sher
parents: 21
diff changeset
511 Lock curLock = myWorkDirLocks.putIfAbsent(path, lock);
21
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
512 if (curLock != null) {
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
513 lock = curLock;
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
514 }
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
515 }
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
516 return lock;
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
517 }
c76d6a2b27f6 proper synchronization for working directories
Pavel.Sher
parents: 20
diff changeset
518
23
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
519 private void removeOldWorkFolders() {
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
520 Set<File> workDirs = getAllClonedRepos();
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
521 if (workDirs == null) return;
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
522
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
523 for (VcsRoot vcsRoot: getMercurialVcsRoots()) {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
524 try {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
525 Settings s = createSettings(vcsRoot);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
526 workDirs.remove(PathUtil.getCanonicalFile(s.getLocalRepositoryDir()));
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
527 } catch (VcsException e) {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
528 Loggers.VCS.error(e);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
529 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
530 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
531
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
532 for (File f: workDirs) {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
533 lockWorkDir(f);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
534 try {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
535 FileUtil.delete(f);
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
536 } finally {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
537 unlockWorkDir(f);
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 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
541
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
542 private Collection<VcsRoot> getMercurialVcsRoots() {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
543 List<VcsRoot> res = new ArrayList<VcsRoot>(myVcsManager.getAllRegisteredVcsRoots());
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
544 FilterUtil.filterCollection(res, new Filter<VcsRoot>() {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
545 public boolean accept(@NotNull final VcsRoot data) {
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
546 return getName().equals(data.getVcsName());
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 return res;
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
550 }
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
551
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
552 @Nullable
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
553 private Set<File> getAllClonedRepos() {
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
554 File workFoldersParent = myDefaultWorkFolderParent;
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
555 if (!workFoldersParent.isDirectory()) return null;
23
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
556
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
557 Set<File> workDirs = new HashSet<File>();
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
558 File[] files = workFoldersParent.listFiles(new FileFilter() {
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
559 public boolean accept(final File file) {
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
560 return file.isDirectory() && file.getName().startsWith(Settings.DEFAULT_WORK_DIR_PREFIX);
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
561 }
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
562 });
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
563 if (files != null) {
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
564 for (File f: files) {
46
13d5f0f56e70 compatibility with TeamCity 3.x
Pavel.Sher
parents: 44
diff changeset
565 workDirs.add(PathUtil.getCanonicalFile(f));
23
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
566 }
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
567 }
103
1b80570ddadf TW-10150
Pavel.Sher
parents: 101
diff changeset
568 return workDirs;
23
3ddf410c2fd6 minor change
Pavel.Sher
parents: 22
diff changeset
569 }
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
570
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
571 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
572 syncClonedRepository(root);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
573
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
574 Settings settings = createSettings(root);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
575
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
576 // I do not know why but hg tag does not work correctly if
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
577 // update command was not invoked for the current repo
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
578 // in such case if there were no tags before Mercurial attempts to
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
579 // create new head when tag is pushed to the parent repository
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
580 UpdateCommand uc = new UpdateCommand(settings);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
581 uc.execute();
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
582
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
583 String fixedTagname = fixTagName(label);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
584 TagCommand tc = new TagCommand(settings);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
585 tc.setRevId(new ChangeSet(version).getId());
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
586 tc.setTag(fixedTagname);
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
587 tc.execute();
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
588
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
589 PushCommand pc = new PushCommand(settings);
101
95c28c14b26e TW-10075
Pavel.Sher
parents: 97
diff changeset
590 // pc.setForce(true);
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
591 pc.execute();
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
592 return fixedTagname;
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
593 }
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
594
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
595 private String fixTagName(final String label) {
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
596 // according to Mercurial documentation http://hgbook.red-bean.com/hgbookch8.html#x12-1570008
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
597 // tag name must not contain:
118
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
598 // Colon (ASCII 58, �:�)
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
599 // Carriage return (ASCII 13, �\r�)
80eb7fdc4de0 Add logging for deleting tmp directories
nd@localhost.localdomain
parents: 106
diff changeset
600 // Newline (ASCII 10, �\n�)
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
601 // all these characters will be replaced with _ (underscore)
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
602 return label.replace(':', '_').replace('\r', '_').replace('\n', '_');
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
603 }
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
604
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
605 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
606 Settings settings = new Settings(myDefaultWorkFolderParent, root);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
607 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
608 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
609 File parentDir = new File(customClonePath);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
610 createClonedRepositoryParentDir(parentDir);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
611
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
612 // take last part of repository path
67
e6971dc6b17c always use url with credentials if username/password are specified
Pavel.Sher
parents: 62
diff changeset
613 String repPath = settings.getRepositoryUrl();
62
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
614 String[] splitted = repPath.split("[/\\\\]");
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
615 if (splitted.length > 0) {
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
616 repPath = splitted[splitted.length-1];
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
617 }
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
618
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
619 File customWorkingDir = new File(parentDir, repPath);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
620 settings.setWorkingDir(customWorkingDir);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
621 } else {
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
622 createClonedRepositoryParentDir(myDefaultWorkFolderParent);
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
623 }
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
624 return settings;
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 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
628 if (!parentDir.exists() && !parentDir.mkdirs()) {
b328c6b6526d TW-5636: Mercurial plugin can easilly hit Windows MAX_PATH limitations
Pavel.Sher
parents: 59
diff changeset
629 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
630 }
44
1490e2981799 labeling/tagging support
Pavel.Sher
parents: 31
diff changeset
631 }
48
8665bfa3e03a enable agent side checkout
Pavel.Sher
parents: 47
diff changeset
632
8665bfa3e03a enable agent side checkout
Pavel.Sher
parents: 47
diff changeset
633 public boolean isAgentSideCheckoutAvailable() {
8665bfa3e03a enable agent side checkout
Pavel.Sher
parents: 47
diff changeset
634 return true;
8665bfa3e03a enable agent side checkout
Pavel.Sher
parents: 47
diff changeset
635 }
0
a530ea876f55 mercurial support sources added
Pavel.Sher
parents:
diff changeset
636 }