View Javadoc
1   /*
2    * Copyright (C) 2009, Constantine Plotnikov <constantine.plotnikov@gmail.com>
3    * Copyright (C) 2007, Dave Watson <dwatson@mimvista.com>
4    * Copyright (C) 2009, Google Inc.
5    * Copyright (C) 2009, JetBrains s.r.o.
6    * Copyright (C) 2008-2009, Robin Rosenberg <robin.rosenberg@dewire.com>
7    * Copyright (C) 2008, Shawn O. Pearce <spearce@spearce.org>
8    * Copyright (C) 2008, Thad Hughes <thadh@thad.corp.google.com>
9    * and other copyright owners as documented in the project's IP log.
10   *
11   * This program and the accompanying materials are made available
12   * under the terms of the Eclipse Distribution License v1.0 which
13   * accompanies this distribution, is reproduced below, and is
14   * available at http://www.eclipse.org/org/documents/edl-v10.php
15   *
16   * All rights reserved.
17   *
18   * Redistribution and use in source and binary forms, with or
19   * without modification, are permitted provided that the following
20   * conditions are met:
21   *
22   * - Redistributions of source code must retain the above copyright
23   *   notice, this list of conditions and the following disclaimer.
24   *
25   * - Redistributions in binary form must reproduce the above
26   *   copyright notice, this list of conditions and the following
27   *   disclaimer in the documentation and/or other materials provided
28   *   with the distribution.
29   *
30   * - Neither the name of the Eclipse Foundation, Inc. nor the
31   *   names of its contributors may be used to endorse or promote
32   *   products derived from this software without specific prior
33   *   written permission.
34   *
35   * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
36   * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
37   * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
38   * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
39   * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
40   * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
41   * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
42   * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
43   * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
44   * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
45   * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
46   * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
47   * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
48   */
49  
50  package org.eclipse.jgit.storage.file;
51  
52  import static java.nio.charset.StandardCharsets.UTF_8;
53  
54  import java.io.ByteArrayOutputStream;
55  import java.io.File;
56  import java.io.FileNotFoundException;
57  import java.io.IOException;
58  import java.text.MessageFormat;
59  
60  import org.eclipse.jgit.errors.ConfigInvalidException;
61  import org.eclipse.jgit.errors.LockFailedException;
62  import org.eclipse.jgit.internal.JGitText;
63  import org.eclipse.jgit.internal.storage.file.FileSnapshot;
64  import org.eclipse.jgit.internal.storage.file.LockFile;
65  import org.eclipse.jgit.lib.Config;
66  import org.eclipse.jgit.lib.Constants;
67  import org.eclipse.jgit.lib.ObjectId;
68  import org.eclipse.jgit.lib.StoredConfig;
69  import org.eclipse.jgit.util.FS;
70  import org.eclipse.jgit.util.FileUtils;
71  import org.eclipse.jgit.util.IO;
72  import org.eclipse.jgit.util.RawParseUtils;
73  import org.slf4j.Logger;
74  import org.slf4j.LoggerFactory;
75  
76  /**
77   * The configuration file that is stored in the file of the file system.
78   */
79  public class FileBasedConfig extends StoredConfig {
80  	private final static Logger LOG = LoggerFactory
81  			.getLogger(FileBasedConfig.class);
82  
83  	private final File configFile;
84  
85  	private final FS fs;
86  
87  	private boolean utf8Bom;
88  
89  	private volatile FileSnapshot snapshot;
90  
91  	private volatile ObjectId hash;
92  
93  	/**
94  	 * Create a configuration with no default fallback.
95  	 *
96  	 * @param cfgLocation
97  	 *            the location of the configuration file on the file system
98  	 * @param fs
99  	 *            the file system abstraction which will be necessary to perform
100 	 *            certain file system operations.
101 	 */
102 	public FileBasedConfig(File cfgLocation, FS fs) {
103 		this(null, cfgLocation, fs);
104 	}
105 
106 	/**
107 	 * The constructor
108 	 *
109 	 * @param base
110 	 *            the base configuration file
111 	 * @param cfgLocation
112 	 *            the location of the configuration file on the file system
113 	 * @param fs
114 	 *            the file system abstraction which will be necessary to perform
115 	 *            certain file system operations.
116 	 */
117 	public FileBasedConfig(Config base, File cfgLocation, FS fs) {
118 		super(base);
119 		configFile = cfgLocation;
120 		this.fs = fs;
121 		this.snapshot = FileSnapshot.DIRTY;
122 		this.hash = ObjectId.zeroId();
123 	}
124 
125 	/** {@inheritDoc} */
126 	@Override
127 	protected boolean notifyUponTransientChanges() {
128 		// we will notify listeners upon save()
129 		return false;
130 	}
131 
132 	/**
133 	 * Get location of the configuration file on disk
134 	 *
135 	 * @return location of the configuration file on disk
136 	 */
137 	public final File getFile() {
138 		return configFile;
139 	}
140 
141 	/**
142 	 * {@inheritDoc}
143 	 * <p>
144 	 * Load the configuration as a Git text style configuration file.
145 	 * <p>
146 	 * If the file does not exist, this configuration is cleared, and thus
147 	 * behaves the same as though the file exists, but is empty.
148 	 */
149 	@Override
150 	public void load() throws IOException, ConfigInvalidException {
151 		final int maxRetries = 5;
152 		int retryDelayMillis = 20;
153 		int retries = 0;
154 		while (true) {
155 			final FileSnapshot oldSnapshot = snapshot;
156 			final FileSnapshot newSnapshot;
157 			// don't use config in this snapshot to avoid endless recursion
158 			newSnapshot = FileSnapshot.saveNoConfig(getFile());
159 			try {
160 				final byte[] in = IO.readFully(getFile());
161 				final ObjectId newHash = hash(in);
162 				if (hash.equals(newHash)) {
163 					if (oldSnapshot.equals(newSnapshot)) {
164 						oldSnapshot.setClean(newSnapshot);
165 					} else {
166 						snapshot = newSnapshot;
167 					}
168 				} else {
169 					final String decoded;
170 					if (isUtf8(in)) {
171 						decoded = RawParseUtils.decode(UTF_8,
172 								in, 3, in.length);
173 						utf8Bom = true;
174 					} else {
175 						decoded = RawParseUtils.decode(in);
176 					}
177 					fromText(decoded);
178 					snapshot = newSnapshot;
179 					hash = newHash;
180 				}
181 				return;
182 			} catch (FileNotFoundException noFile) {
183 				// might be locked by another process (see exception Javadoc)
184 				if (retries < maxRetries && configFile.exists()) {
185 					if (LOG.isDebugEnabled()) {
186 						LOG.debug(MessageFormat.format(
187 								JGitText.get().configHandleMayBeLocked,
188 								Integer.valueOf(retries)), noFile);
189 					}
190 					try {
191 						Thread.sleep(retryDelayMillis);
192 					} catch (InterruptedException e) {
193 						Thread.currentThread().interrupt();
194 					}
195 					retries++;
196 					retryDelayMillis *= 2; // max wait 1260 ms
197 					continue;
198 				}
199 				if (configFile.exists()) {
200 					throw noFile;
201 				}
202 				clear();
203 				snapshot = newSnapshot;
204 				return;
205 			} catch (IOException e) {
206 				if (FileUtils.isStaleFileHandle(e)
207 						&& retries < maxRetries) {
208 					if (LOG.isDebugEnabled()) {
209 						LOG.debug(MessageFormat.format(
210 								JGitText.get().configHandleIsStale,
211 								Integer.valueOf(retries)), e);
212 					}
213 					retries++;
214 					continue;
215 				}
216 				throw new IOException(MessageFormat
217 						.format(JGitText.get().cannotReadFile, getFile()), e);
218 			} catch (ConfigInvalidException e) {
219 				throw new ConfigInvalidException(MessageFormat
220 						.format(JGitText.get().cannotReadFile, getFile()), e);
221 			}
222 		}
223 	}
224 
225 	/**
226 	 * {@inheritDoc}
227 	 * <p>
228 	 * Save the configuration as a Git text style configuration file.
229 	 * <p>
230 	 * <b>Warning:</b> Although this method uses the traditional Git file
231 	 * locking approach to protect against concurrent writes of the
232 	 * configuration file, it does not ensure that the file has not been
233 	 * modified since the last read, which means updates performed by other
234 	 * objects accessing the same backing file may be lost.
235 	 */
236 	@Override
237 	public void save() throws IOException {
238 		final byte[] out;
239 		final String text = toText();
240 		if (utf8Bom) {
241 			final ByteArrayOutputStream bos = new ByteArrayOutputStream();
242 			bos.write(0xEF);
243 			bos.write(0xBB);
244 			bos.write(0xBF);
245 			bos.write(text.getBytes(UTF_8));
246 			out = bos.toByteArray();
247 		} else {
248 			out = Constants.encode(text);
249 		}
250 
251 		final LockFileal/storage/file/LockFile.html#LockFile">LockFile lf = new LockFile(getFile());
252 		if (!lf.lock())
253 			throw new LockFailedException(getFile());
254 		try {
255 			lf.setNeedSnapshot(true);
256 			lf.write(out);
257 			if (!lf.commit())
258 				throw new IOException(MessageFormat.format(JGitText.get().cannotCommitWriteTo, getFile()));
259 		} finally {
260 			lf.unlock();
261 		}
262 		snapshot = lf.getCommitSnapshot();
263 		hash = hash(out);
264 		// notify the listeners
265 		fireConfigChangedEvent();
266 	}
267 
268 	/** {@inheritDoc} */
269 	@Override
270 	public void clear() {
271 		hash = hash(new byte[0]);
272 		super.clear();
273 	}
274 
275 	private static ObjectId hash(byte[] rawText) {
276 		return ObjectId.fromRaw(Constants.newMessageDigest().digest(rawText));
277 	}
278 
279 	/** {@inheritDoc} */
280 	@SuppressWarnings("nls")
281 	@Override
282 	public String toString() {
283 		return getClass().getSimpleName() + "[" + getFile().getPath() + "]";
284 	}
285 
286 	/**
287 	 * Whether the currently loaded configuration file is outdated
288 	 *
289 	 * @return returns true if the currently loaded configuration file is older
290 	 *         than the file on disk
291 	 */
292 	public boolean isOutdated() {
293 		return snapshot.isModified(getFile());
294 	}
295 
296 	/**
297 	 * {@inheritDoc}
298 	 *
299 	 * @since 4.10
300 	 */
301 	@Override
302 	protected byte[] readIncludedConfig(String relPath)
303 			throws ConfigInvalidException {
304 		final File file;
305 		if (relPath.startsWith("~/")) { //$NON-NLS-1$
306 			file = fs.resolve(fs.userHome(), relPath.substring(2));
307 		} else {
308 			file = fs.resolve(configFile.getParentFile(), relPath);
309 		}
310 
311 		if (!file.exists()) {
312 			return null;
313 		}
314 
315 		try {
316 			return IO.readFully(file);
317 		} catch (IOException ioe) {
318 			throw new ConfigInvalidException(MessageFormat
319 					.format(JGitText.get().cannotReadFile, relPath), ioe);
320 		}
321 	}
322 }