From 8384e02b7d6f1a1ecd6782c6b5f682c109cd6e04 Mon Sep 17 00:00:00 2001
From: James Moger <james.moger@gitblit.com>
Date: Thu, 10 Apr 2014 18:58:08 -0400
Subject: [PATCH] Disregard comment/description for add/remove in FileKeyManager

---
 src/main/java/com/gitblit/transport/ssh/FileKeyManager.java |  104 +++++++++++++++++++++++++++++++++++++++-------------
 1 files changed, 78 insertions(+), 26 deletions(-)

diff --git a/src/main/java/com/gitblit/transport/ssh/FileKeyManager.java b/src/main/java/com/gitblit/transport/ssh/FileKeyManager.java
index 04b42fc..1eb470b 100644
--- a/src/main/java/com/gitblit/transport/ssh/FileKeyManager.java
+++ b/src/main/java/com/gitblit/transport/ssh/FileKeyManager.java
@@ -28,40 +28,40 @@
 
 import com.gitblit.Keys;
 import com.gitblit.manager.IRuntimeManager;
-import com.gitblit.utils.FileUtils;
 import com.google.common.base.Charsets;
+import com.google.common.base.Joiner;
 import com.google.common.io.Files;
 
 /**
  * Manages SSH keys on the filesystem.
- * 
+ *
  * @author James Moger
  *
  */
 public class FileKeyManager implements IKeyManager {
 
 	protected final IRuntimeManager runtimeManager;
-	
+
 	public FileKeyManager(IRuntimeManager runtimeManager) {
 		this.runtimeManager = runtimeManager;
 	}
-	
+
 	@Override
 	public String toString() {
 		File dir = runtimeManager.getFileOrFolder(Keys.git.sshKeysFolder, "${baseFolder}/ssh");
 		return MessageFormat.format("{0} ({1})", getClass().getSimpleName(), dir);
 	}
-	
+
 	@Override
 	public FileKeyManager start() {
 		return this;
 	}
-	
+
 	@Override
 	public boolean isReady() {
 		return true;
 	}
-	
+
 	@Override
 	public FileKeyManager stop() {
 		return this;
@@ -75,10 +75,8 @@
 				return null;
 			}
 			if (keys.exists()) {
-				String str = Files.toString(keys, Charsets.ISO_8859_1);
-				String [] entries = str.split("\n");
 				List<PublicKey> list = new ArrayList<PublicKey>();
-				for (String entry : entries) {
+				for (String entry : Files.readLines(keys, Charsets.ISO_8859_1)) {
 					if (entry.trim().length() == 0) {
 						// skip blanks
 						continue;
@@ -91,7 +89,7 @@
 					final byte[] bin = Base64.decodeBase64(Constants.encodeASCII(parts[1]));
 					list.add(new Buffer(bin).getRawPublicKey());
 				}
-				
+
 				if (list.isEmpty()) {
 					return null;
 				}
@@ -103,40 +101,87 @@
 		return null;
 	}
 
+	/**
+	 * Adds a unique key to the keystore.  This function determines uniqueness
+	 * by disregarding the comment/description field during key comparisons.
+	 */
 	@Override
 	public boolean addKey(String username, String data) {
 		try {
-			File keys = getKeystore(username);
-			Files.append(data + '\n', keys, Charsets.ISO_8859_1);
+			String newKey = stripCommentFromKey(data);
+
+			List<String> lines = new ArrayList<String>();
+			File keystore = getKeystore(username);
+			if (keystore.exists()) {
+				for (String entry : Files.readLines(keystore, Charsets.ISO_8859_1)) {
+					String line = entry.trim();
+					if (line.length() == 0) {
+						// keep blanks
+						lines.add(entry);
+						continue;
+					}
+					if (line.charAt(0) == '#') {
+						// keep comments
+						lines.add(entry);
+						continue;
+					}
+
+					// only add keys that do not match the new key
+					String oldKey = stripCommentFromKey(line);
+					if (!newKey.equals(oldKey)) {
+						lines.add(entry);
+					}
+				}
+			}
+
+			// add new key
+			lines.add(data);
+
+			// write keystore
+			String content = Joiner.on("\n").join(lines).trim().concat("\n");
+			Files.write(content, keystore, Charsets.ISO_8859_1);
 			return true;
 		} catch (IOException e) {
 			throw new RuntimeException("Cannot add ssh key", e);
 		}
 	}
-	
+
+	/**
+	 * Removes a key from the keystore.
+	 */
 	@Override
 	public boolean removeKey(String username, String data) {
 		try {
+			String rmKey = stripCommentFromKey(data);
+
 			File keystore = getKeystore(username);
 			if (keystore.exists()) {
-				String str = Files.toString(keystore, Charsets.ISO_8859_1);
-				List<String> keep = new ArrayList<String>();
-				String [] entries = str.split("\n");
-				for (String entry : entries) {
-					if (entry.trim().length() == 0) {
+				List<String> lines = new ArrayList<String>();
+				for (String entry : Files.readLines(keystore, Charsets.ISO_8859_1)) {
+					String line = entry.trim();
+					if (line.length() == 0) {
 						// keep blanks
-						keep.add(entry);
+						lines.add(entry);
 						continue;
 					}
-					if (entry.charAt(0) == '#') {
+					if (line.charAt(0) == '#') {
 						// keep comments
-						keep.add(entry);
+						lines.add(entry);
 						continue;
 					}
-					final String[] parts = entry.split(" ");
-					if (!parts[1].equals(data)) {
-						keep.add(entry);
+
+					// only include keys that are NOT rmKey
+					String oldKey = stripCommentFromKey(line);
+					if (!rmKey.equals(oldKey)) {
+						lines.add(entry);
 					}
+				}
+				if (lines.isEmpty()) {
+					keystore.delete();
+				} else {
+					// write keystore
+					String content = Joiner.on("\n").join(lines).trim().concat("\n");
+					Files.write(content, keystore, Charsets.ISO_8859_1);
 				}
 				return true;
 			}
@@ -148,7 +193,7 @@
 
 	@Override
 	public boolean removeAllKeys(String username) {
-		return FileUtils.delete(getKeystore(username));
+		return getKeystore(username).delete();
 	}
 
 	protected File getKeystore(String username) {
@@ -157,4 +202,11 @@
 		File keys = new File(dir, username + ".keys");
 		return keys;
 	}
+
+	/* Strips the comment from the key data and eliminates whitespace diffs */
+	protected String stripCommentFromKey(String data) {
+		String [] cols = data.split(" ");
+		String key = Joiner.on(" ").join(cols[0], cols[1]);
+		return key;
+	}
 }

--
Gitblit v1.9.1