From 9c14cbf445c1711aa2110612ad12f16a2f93fe85 Mon Sep 17 00:00:00 2001 From: Brett Johnson Date: Fri, 23 Jan 2015 18:55:34 -0800 Subject: [PATCH] Fix AccountBySid Cache key. JNA was apparently reusing the same PSID structure, so I could not use it as the key to the cache. This uses the SID bytes as the key instead. Code Review: http://codereview.appspot.com/196220043 --- .../fs/WindowsAclFileAttributeViews.java | 48 ++++++++++++------- 1 file changed, 32 insertions(+), 16 deletions(-) diff --git a/src/com/google/enterprise/adaptor/fs/WindowsAclFileAttributeViews.java b/src/com/google/enterprise/adaptor/fs/WindowsAclFileAttributeViews.java index 766b303..714c151 100644 --- a/src/com/google/enterprise/adaptor/fs/WindowsAclFileAttributeViews.java +++ b/src/com/google/enterprise/adaptor/fs/WindowsAclFileAttributeViews.java @@ -59,8 +59,6 @@ import java.util.List; import java.util.Map; import java.util.Set; -import java.util.concurrent.Callable; -import java.util.concurrent.ExecutionException; import java.util.concurrent.TimeUnit; import java.util.logging.Level; import java.util.logging.Logger; @@ -151,7 +149,7 @@ class WindowsAclFileAttributeViews { private final Shlwapi shlwapi; /** Cache of AccountsBySid should max out at about 10-12 MB. */ - private Cache accountCache = CacheBuilder + private Cache accountCache = CacheBuilder .newBuilder().initialCapacity(10000).maximumSize(100000) .expireAfterWrite(24, TimeUnit.HOURS).build(); @@ -395,22 +393,40 @@ public AclEntry newAclEntry(WinNT.ACCESS_ACEStructure ace) { .build(); } - @VisibleForTesting - Account getAccountBySid(final WinNT.PSID sid) throws Win32Exception { - try { - return accountCache.get(sid, new Callable() { - @Override - public Account call() throws IOException { - return Advapi32Util.getAccountBySid(sid); - } - }); - } catch (ExecutionException e) { - if (e.getCause() instanceof Win32Exception) { - throw (Win32Exception)(e.getCause()); + private static class SidKey { + private byte[] sidBytes; + + SidKey(byte[] sidBytes) { + this.sidBytes = sidBytes; + } + + @Override + public boolean equals(Object other) { + if (other instanceof SidKey) { + return Arrays.equals(sidBytes, ((SidKey) other).sidBytes); } else { - throw new Win32Exception(kernel32.GetLastError()); + return false; } } + + @Override + public int hashCode() { + return Arrays.hashCode(sidBytes); + } + } + + @VisibleForTesting + Account getAccountBySid(final WinNT.PSID sid) throws Win32Exception { + // PSID made a poor cache key, but the raw bytes work much better. + Account account = accountCache.getIfPresent(new SidKey(sid.getBytes())); + if (account == null) { + account = Advapi32Util.getAccountBySid(sid); + SidKey key = new SidKey(Arrays.copyOf(account.sid, account.sid.length)); + account.sid = null; // Reduce cache memory usage by dropping the + account.sidString = null; // unused sid bytes and sidString. + accountCache.put(key, account); + } + return account; } // One-to-one corresponance to WinNT.SID_NAME_USE "enumeration".