changeset 498:0205a5c4566b

Issue 38: preserve user formatting and comments when updating configuration files
author Artem Tikhomirov <tikhomirov.artem@gmail.com>
date Fri, 26 Oct 2012 18:17:15 +0200 (2012-10-26)
parents 02140be396d5
children 899a1b68ef03
files src/org/tmatesoft/hg/core/HgIOException.java src/org/tmatesoft/hg/core/HgMissingConfigElementException.java src/org/tmatesoft/hg/core/HgUpdateConfigCommand.java src/org/tmatesoft/hg/internal/ConfigFile.java src/org/tmatesoft/hg/internal/ConfigFileParser.java
diffstat 5 files changed, 322 insertions(+), 89 deletions(-) [+]
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/org/tmatesoft/hg/core/HgIOException.java	Fri Oct 26 18:17:15 2012 +0200
@@ -0,0 +1,51 @@
+/*
+ * Copyright (c) 2012 TMate Software Ltd
+ *  
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; version 2 of the License.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * For information on how to redistribute this software under
+ * the terms of a license other than GNU General Public License
+ * contact TMate Software at support@hg4j.com
+ */
+package org.tmatesoft.hg.core;
+
+import java.io.File;
+import java.io.IOException;
+
+/**
+ * Tailored wrap for {@link IOException} and similar I/O-related issues. Unlike {@link IOException}, 
+ * keeps track of {@link File} that caused the problem. Besides, additional information (like revision, 
+ * see {@link HgException}) may be attached.
+ * 
+ * @author Artem Tikhomirov
+ * @author TMate Software Ltd.
+ */
+@SuppressWarnings("serial")
+public class HgIOException extends HgException {
+	private final File file;
+
+	public HgIOException(String message, File troubleFile) {
+		this(message, null, troubleFile);
+	}
+
+	/**
+	 * @param message describes the issue, never <code>null</code>
+	 * @param cause root cause for the error, likely {@link IOException} or its subclass, but not necessarily, and may be omitted. 
+	 * @param troubleFile file we tried to deal with, never <code>null</code>
+	 */
+	public HgIOException(String message, Exception cause, File troubleFile) {
+		super(message, cause);
+		file = troubleFile;
+	}
+
+	public File getFile() {
+		return file;
+	}
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/org/tmatesoft/hg/core/HgMissingConfigElementException.java	Fri Oct 26 18:17:15 2012 +0200
@@ -0,0 +1,44 @@
+/*
+ * Copyright (c) 2012 TMate Software Ltd
+ *  
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; version 2 of the License.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * For information on how to redistribute this software under
+ * the terms of a license other than GNU General Public License
+ * contact TMate Software at support@hg4j.com
+ */
+package org.tmatesoft.hg.core;
+
+/**
+ * Exception 
+ * 
+ * @author Artem Tikhomirov
+ * @author TMate Software Ltd.
+ */
+@SuppressWarnings("serial")
+public class HgMissingConfigElementException extends HgException {
+
+	private final String section;
+	private final String key;
+
+	public HgMissingConfigElementException(String message, String troubleSection, String missingKey) {
+		super(message);
+		section = troubleSection;
+		key = missingKey;
+	}
+
+	public String getSection() {
+		return section;
+	}
+	
+	public String getKey() {
+		return key;
+	}
+}
--- a/src/org/tmatesoft/hg/core/HgUpdateConfigCommand.java	Thu Oct 25 19:59:08 2012 +0200
+++ b/src/org/tmatesoft/hg/core/HgUpdateConfigCommand.java	Fri Oct 26 18:17:15 2012 +0200
@@ -17,33 +17,33 @@
 package org.tmatesoft.hg.core;
 
 import java.io.File;
+import java.io.FileInputStream;
+import java.io.FileOutputStream;
 import java.io.IOException;
-import java.util.ArrayList;
-import java.util.LinkedHashMap;
+import java.util.LinkedList;
 import java.util.List;
 import java.util.Map;
 
 import org.tmatesoft.hg.internal.ConfigFile;
-import org.tmatesoft.hg.internal.Experimental;
+import org.tmatesoft.hg.internal.ConfigFileParser;
 import org.tmatesoft.hg.internal.Internals;
 import org.tmatesoft.hg.repo.HgInternals;
+import org.tmatesoft.hg.repo.HgInvalidStateException;
 import org.tmatesoft.hg.repo.HgRepository;
 
 /**
- * WORK IN PROGRESS, DO NOT USE
+ * Command to alter Mercurial configuration settings at various levels (system-wide, user-wide, repository-wide).
  *
  * @author Artem Tikhomirov
  * @author TMate Software Ltd.
  */
-@Experimental(reason="Investigating approaches to alter Hg configuration files")
 public final class HgUpdateConfigCommand extends HgAbstractCommand<HgUpdateConfigCommand> {
 	
+	private final SessionContext sessionCtx;
 	private final File configFile;
 	
-	private Map<String,List<String>> toRemove;
-	private Map<String,Map<String,String>> toSet;
-
-	private final SessionContext sessionCtx;
+	private final List<Operation> changes = new LinkedList<Operation>();
+	private boolean ignoreMissingKeys = false;
 
 	private HgUpdateConfigCommand(SessionContext sessionContext, File configurationFile) {
 		sessionCtx = sessionContext;
@@ -51,7 +51,6 @@
 	}
 	
 	public static HgUpdateConfigCommand forRepository(HgRepository hgRepo) {
-		// XXX HgRepository to implement SessionContext.Provider (with getContext())?
 		return new HgUpdateConfigCommand(hgRepo.getSessionContext(), HgInternals.getImplementationRepo(hgRepo).getFileFromRepoDir("hgrc"));
 	}
 	
@@ -62,94 +61,230 @@
 	public static HgUpdateConfigCommand forInstallation(SessionContext ctx) {
 		return new HgUpdateConfigCommand(ctx, Internals.getInstallationConfigurationFileToWrite(ctx));
 	}
-	
+
 	/**
-	 * Remove a property altogether
+	 * Remove an entry altogether. If no entry with the key found, {@link #execute()} fails.
+	 * 
+	 * @param section identifies section to alter, not <code>null</code> or otherwise ill-formed name
+	 * @param key identifies entry within section, not <code>null</code> or otherwise ill-formed name 
 	 * @return <code>this</code> for convenience
+	 * @throws IllegalArgumentException if arguments are <code>null</code> or empty
 	 */
-	public HgUpdateConfigCommand remove(String section, String key) {
-		if (toRemove == null) {
-			toRemove = new LinkedHashMap<String, List<String>>();
-		}
-		List<String> s = toRemove.get(section);
-		if (s == null) {
-			toRemove.put(section, s = new ArrayList<String>(5));
-		}
-		s.add(key);
-		if (toSet != null && toSet.containsKey(section)) {
-			toSet.get(section).remove(key);
-		}
+	public HgUpdateConfigCommand remove(String section, String key) throws IllegalArgumentException {
+		checkSection(section);
+		checkKey(key);
+		changes.add(Operation.deleteEntry(section, key));
 		return this;
 	}
 
 	/**
-	 * Delete single attribute in a multi-valued property
-	 * @return <code>this</code> for convenience
-	 */
-	public HgUpdateConfigCommand remove(String section, String key, String value) {
-		throw new UnsupportedOperationException();
-	}
-	
-	/**
-	 * Set single-valued properties or update multi-valued with a single value
+	 * Delete single attribute in a multi-valued property. If specified value not found among values of
+	 * the identified entry, {@link #execute()} fails. 
+	 * 
+	 * @param section identifies section to alter, not <code>null</code> or otherwise ill-formed name
+	 * @param key identifies entry within section, not <code>null</code> or otherwise ill-formed name 
+	 * @param value one of the values to remove, not <code>null</code> or an empty value
 	 * @return <code>this</code> for convenience
+	 * @throws IllegalArgumentException if arguments are <code>null</code> or empty
 	 */
-	public HgUpdateConfigCommand put(String section, String key, String value) {
-		if (toSet == null) {
-			toSet =  new LinkedHashMap<String, Map<String,String>>();
-		}
-		Map<String,String> s = toSet.get(section);
-		if (s == null) {
-			toSet.put(section, s = new LinkedHashMap<String, String>());
-		}
-		s.put(key, value);
+	public HgUpdateConfigCommand remove(String section, String key, String value) throws IllegalArgumentException {
+		checkSection(section);
+		checkKey(key);
+		changes.add(Operation.deleteValue(section, key, value));
 		return this;
 	}
 	
 	/**
-	 * Multi-valued properties
+	 * Set single-valued property or update multi-valued with a single value
+	 * 
+	 * @param section identifies section to alter, not <code>null</code> or otherwise ill-formed name
+	 * @param key identifies entry within section, not <code>null</code> or otherwise ill-formed name 
+	 * @param value new value, may be <code>null</code>
 	 * @return <code>this</code> for convenience
+	 * @throws IllegalArgumentException if arguments are <code>null</code> or empty
 	 */
-	public HgUpdateConfigCommand add(String section, String key, String value) {
-		throw new UnsupportedOperationException();
+	public HgUpdateConfigCommand put(String section, String key, String value) throws IllegalArgumentException {
+		checkSection(section);
+		checkKey(key);
+		changes.add(Operation.setValue(section, key, value));
+		return this;
 	}
 	
 	/**
-	 * Perform config file update
+	 * Add value to a multi-valued entry. If specified entry not found, {@link #execute()} fails.
 	 * 
+	 * @param section identifies section to alter, not <code>null</code> or otherwise ill-formed name
+	 * @param key identifies entry within section, not <code>null</code> or otherwise ill-formed name 
+	 * @param value new value to add, not <code>null</code> or an empty value
+	 * @return <code>this</code> for convenience
+	 * @throws IllegalArgumentException if arguments are <code>null</code> or empty
+	 */
+	public HgUpdateConfigCommand add(String section, String key, String value) throws IllegalArgumentException {
+		checkSection(section);
+		checkKey(key);
+		changes.add(Operation.addValue(section, key, value));
+		return this;
+	}
+	
+	/**
+	 * Tells whether {@link #execute()} shall fail with exception if keys selected for modification were not found. 
+	 * If <code>true</code>, missing keys would be silently ignored. 
+	 * When <code>false</code>(<em>default</em>), exception would be raised.
+	 *  
+	 * @param ignoreMissing pass <code>true</code> to ignore any incorrect keys
+	 * @return <code>this</code> for convenience
+	 */
+	public HgUpdateConfigCommand ignoreMissing(boolean ignoreMissing) {
+		ignoreMissingKeys = ignoreMissing;
+		return this;
+	}
+	
+	/**
+	 * Perform configuration file update.
+	 * 
+	 * @throws HgMissingConfigElementException if attempt to alter an entry failed to find one, and missing keys are not ignored
+	 * @throws HgIOException when configuration file read/write attemt has failed
 	 * @throws HgException subclass thereof to indicate specific issue with the command arguments or repository state
 	 */
-	public void execute() throws HgException {
+	public void execute() throws HgMissingConfigElementException, HgIOException, HgException {
 		try {
-			ConfigFile cfg = new ConfigFile(sessionCtx);
-			cfg.addLocation(configFile);
-			if (toRemove != null) {
-				for (Map.Entry<String,List<String>> s : toRemove.entrySet()) {
-					for (String e : s.getValue()) {
-						cfg.putString(s.getKey(), e, null);
+			ConfigFile cfgRead = new ConfigFile(sessionCtx);
+			cfgRead.addLocation(configFile);
+			ConfigFileParser cfgWrite = new ConfigFileParser();
+			FileInputStream fis = new FileInputStream(configFile);
+			cfgWrite.parse(fis);
+			fis.close();
+			for (Operation op : changes) {
+				if (!ignoreMissingKeys && !cfgRead.hasSection(op.section)) {
+					throw new HgMissingConfigElementException("Bad section name", op.section, op.key);
+				}
+				Map<String, String> sect = cfgRead.getSection(op.section);
+				if (!ignoreMissingKeys && !sect.containsKey(op.key)) {
+					throw new HgMissingConfigElementException("Bad key name", op.section, op.key);
+				}
+				String oldValue = sect.get(op.key);
+				if (oldValue == null) {
+					oldValue = "";
+				}
+				switch (op.kind) {
+				case AddValue: {
+					String separator = ", "; // XXX shall parse and find out separator kind in use
+					String newValue = oldValue + separator + op.value;
+					if (sect.containsKey(op.key)) {
+						cfgWrite.change(op.section, op.key, newValue);
+					} else {
+						cfgWrite.add(op.section, op.key, newValue);
 					}
+					break;
+				}
+				case DelValue: {
+					if (!ignoreMissingKeys && (oldValue.length() == 0 || !oldValue.contains(op.value))) {
+						throw new HgMissingConfigElementException(String.format("Bad value '%s' to delete from '%s'", op.value, oldValue), op.section, op.key);
+					}
+					int start = oldValue.indexOf(op.value);
+					if (start == -1) {
+						// nothing to change
+						break;
+					}
+					int commaPos = -1;
+					for (int i = start-1; i >=0; i--) {
+						if (oldValue.charAt(i) == ',') {
+							commaPos = i;
+							break;
+						}
+					}
+					for (int i = start + op.value.length(); commaPos == -1 && i < oldValue.length(); i++) {
+						if (oldValue.charAt(i) == ',') {
+							commaPos = i;
+							break;
+						}
+					}
+					String newValue;
+					if (commaPos >= 0) {
+						if (commaPos < start) {
+							// from preceding comma up to end of value
+							newValue = oldValue.substring(0, commaPos) + oldValue.substring(start + op.value.length());
+						} else {
+							// from value start up to and including subsequent comma
+							newValue = oldValue.substring(0, start) + oldValue.substring(commaPos+1);
+						}
+					} else {
+						// found no separator, just remove the value
+						// extra whitespaces (if space, not a comma is a separator) won't hurt
+						newValue = oldValue.substring(0, start) + oldValue.substring(start + op.value.length());
+					}
+					cfgWrite.change(op.section, op.key, newValue);
+					break;
+				}
+				case SetValue: {
+					if (sect.containsKey(op.key)) {
+						cfgWrite.change(op.section, op.key, op.value);
+					} else {
+						cfgWrite.add(op.section, op.key, op.value);
+					}
+					break;
+				}
+				case DelEntry: {
+					cfgWrite.delete(op.section, op.key);
+					break;
+				}
+				default: throw new HgInvalidStateException(String.format("Unknown change %s", op.kind));
 				}
 			}
-			if (toSet != null) {
-				for (Map.Entry<String,Map<String,String>> s : toSet.entrySet()) {
-					for (Map.Entry<String, String> e : s.getValue().entrySet()) {
-						cfg.putString(s.getKey(), e.getKey(), e.getValue());
-					}
-				}
-			}
-			cfg.writeTo(configFile);
+			FileOutputStream fos = new FileOutputStream(configFile);
+			cfgWrite.update(fos);
+			fos.close();
 		} catch (IOException ex) {
 			String m = String.format("Failed to update configuration file %s", configFile);
 			throw new HgBadArgumentException(m, ex); // TODO [post-1.0] better exception, it's not bad argument case
 		}
 	}
+	
+	private static void checkSection(String section) throws IllegalArgumentException {
+		if (section == null || section.trim().length() == 0) {
+			throw new IllegalArgumentException(String.format("Section name can't be empty: %s", section));
+		}
+	}
+	
+	private static void checkKey(String key) throws IllegalArgumentException {
+		if (key == null || key.trim().length() == 0) {
+			throw new IllegalArgumentException(String.format("Entry key can't be empty: %s", key));
+		}
+	}
 
 
-	public static void main(String[] args) throws Exception {
-		HgUpdateConfigCommand cmd = HgUpdateConfigCommand.forUser(null);
-		cmd.remove("test1", "sample1");
-		cmd.put("test2", "sample2", "value2");
-		cmd.put("ui", "user-name", "Another User <email@domain.com>");
-		cmd.execute();
+	private static class Operation {
+		private enum OpKind { AddValue, SetValue, DelValue, DelEntry };
+
+		public final OpKind kind;
+		public final String section;
+		public final String key;
+		public final String value;
+		
+		private Operation(OpKind t, String s, String k, String v) {
+			kind = t;
+			section = s;
+			key = k;
+			value = v;
+		}
+		
+		public static Operation deleteEntry(String section, String key) throws IllegalArgumentException {
+			return new Operation(OpKind.DelEntry, section, key, null);
+		}
+		public static Operation deleteValue(String section, String key, String value) throws IllegalArgumentException {
+			if (value == null || value.trim().length() == 0) {
+				throw new IllegalArgumentException(String.format("Can't remove empty value '%s'", value));
+			}
+			return new Operation(OpKind.DelValue, section, key, value);
+		}
+		public static Operation addValue(String section, String key, String value) throws IllegalArgumentException {
+			if (value == null || value.trim().length() == 0) {
+				throw new IllegalArgumentException(String.format("Can't add empty value '%s'", value));
+			}
+			return new Operation(OpKind.AddValue, section, key, value);
+		}
+		public static Operation setValue(String section, String key, String value) throws IllegalArgumentException {
+			return new Operation(OpKind.SetValue, section, key, value);
+		}
 	}
 }
--- a/src/org/tmatesoft/hg/internal/ConfigFile.java	Thu Oct 25 19:59:08 2012 +0200
+++ b/src/org/tmatesoft/hg/internal/ConfigFile.java	Fri Oct 26 18:17:15 2012 +0200
@@ -62,6 +62,11 @@
 		return sections == null ? Collections.<String>emptyList() : Collections.unmodifiableList(sections);
 	}
 
+	/**
+	 * Access map of section keys and values.
+	 * @param sectionName name of the section to retrieve
+	 * @return never <code>null</code>, empty map in case no section with specified name found 
+	 */
 	public Map<String,String> getSection(String sectionName) {
 		if (sections == null) {
 			return Collections.emptyMap();
--- a/src/org/tmatesoft/hg/internal/ConfigFileParser.java	Thu Oct 25 19:59:08 2012 +0200
+++ b/src/org/tmatesoft/hg/internal/ConfigFileParser.java	Fri Oct 26 18:17:15 2012 +0200
@@ -16,7 +16,6 @@
  */
 package org.tmatesoft.hg.internal;
 
-import java.io.ByteArrayInputStream;
 import java.io.ByteArrayOutputStream;
 import java.io.IOException;
 import java.io.InputStream;
@@ -209,7 +208,27 @@
 			// if section comes more than once, update only first one.
 			processedSections.add(section.name);
 		}
+		// push rest of the contents
 		out.write(contents, contentsOffset, contents.length - contentsOffset);
+		//
+		// add entries in new sections
+		LinkedHashSet<String> newSections = new LinkedHashSet<String>();
+		for (Iterator<String> it = additions.iterator(); it.hasNext();) {
+			String s = it.next(); it.next(); it.next();
+			if (!processedSections.contains(s)) {
+				newSections.add(s);
+			}
+		}
+		for (String newSectionName : newSections) {
+			out.write(String.format("\n[%s]", newSectionName).getBytes());
+			for (Iterator<String> it = additions.iterator(); it.hasNext();) {
+				String s = it.next(), k = it.next(), v = it.next();
+				if (newSectionName.equals(s)) {
+					out.write(String.format("\n%s = %s", k, v).getBytes());
+				}
+			}
+			out.write("\n".getBytes());
+		}
 	}
 	
 	private void processLine(int lineNumber, int offset, byte[] line) throws IOException {
@@ -359,25 +378,4 @@
 			e.toArray(entries);
 		}
 	}
-
-	public static void main(String[] args) throws Exception {
-		ConfigFileParser p = new ConfigFileParser();
-		p.parse(new ByteArrayInputStream(xx.getBytes()));
-		System.out.println(">>>");
-		System.out.println(xx);
-		System.out.println("===");
-		p.add("sect1", "key5", "x");
-		ByteArrayOutputStream out = new ByteArrayOutputStream(xx.length());
-		p.update(out);
-		System.out.println(new String(out.toByteArray()));
-		/*
-		for (Section s : p.sections) {
-			System.out.printf("[%s@%d]\n", s.name, s.start);
-			for (Entry e : s.entries) {
-				System.out.printf("%s@%d = %d..%d\n", e.name, e.start, e.valueStart, e.valueEnd);
-			}
-		}
-		*/
-	}
-	private static final String xx = "#comment1\n [sect1]\nkey = value #not a comment2\n#comment3\nkey2=   \nkey3 =  \n  value1, #cc\n  value2\nkey4 = v1,\n  v2 \n  ,v3\n\n\n[sect2]\nx = a";
 }