From b16e1569fe73f8a98b61a861a0f2e3ba235d5a36 Mon Sep 17 00:00:00 2001 From: Robert Muir Date: Sat, 12 Sep 2015 19:46:39 -0400 Subject: [PATCH 1/2] Remove all setAccessible in tests and forbid --- .../analyzing/XAnalyzingSuggester.java | 4 + .../bootstrap/SecurityTests.java | 16 -- .../indices/analyze/HunspellServiceIT.java | 8 +- .../network/DirectBufferNetworkIT.java | 146 ------------------ .../CompletionPostingsFormatTests.java | 8 +- .../org/elasticsearch/test/ESTestCase.java | 6 + .../test/store/MockFSDirectoryService.java | 9 +- .../resources/forbidden/all-signatures.txt | 4 + .../resources/forbidden/core-signatures.txt | 4 - 9 files changed, 20 insertions(+), 185 deletions(-) delete mode 100644 core/src/test/java/org/elasticsearch/network/DirectBufferNetworkIT.java diff --git a/core/src/main/java/org/apache/lucene/search/suggest/analyzing/XAnalyzingSuggester.java b/core/src/main/java/org/apache/lucene/search/suggest/analyzing/XAnalyzingSuggester.java index 98401cd2e14..bac323d63c6 100644 --- a/core/src/main/java/org/apache/lucene/search/suggest/analyzing/XAnalyzingSuggester.java +++ b/core/src/main/java/org/apache/lucene/search/suggest/analyzing/XAnalyzingSuggester.java @@ -259,6 +259,10 @@ public class XAnalyzingSuggester extends Lookup { public long ramBytesUsed() { return fst == null ? 0 : fst.ramBytesUsed(); } + + public int getMaxAnalyzedPathsForOneInput() { + return maxAnalyzedPathsForOneInput; + } // Replaces SEP with epsilon or remaps them if // we were asked to preserve them: diff --git a/core/src/test/java/org/elasticsearch/bootstrap/SecurityTests.java b/core/src/test/java/org/elasticsearch/bootstrap/SecurityTests.java index eaa2592b768..d0685b1a6ce 100644 --- a/core/src/test/java/org/elasticsearch/bootstrap/SecurityTests.java +++ b/core/src/test/java/org/elasticsearch/bootstrap/SecurityTests.java @@ -200,22 +200,6 @@ public class SecurityTests extends ESTestCase { } catch (IOException expected) {} } - /** We only grant this to special jars */ - public void testUnsafeAccess() throws Exception { - assumeTrue("test requires security manager", System.getSecurityManager() != null); - try { - // class could be legitimately loaded, so we might not fail until setAccessible - Class.forName("sun.misc.Unsafe") - .getDeclaredField("theUnsafe") - .setAccessible(true); - fail("didn't get expected exception"); - } catch (SecurityException expected) { - // ok - } catch (Exception somethingElse) { - assumeNoException("perhaps JVM doesn't have Unsafe?", somethingElse); - } - } - /** can't execute processes */ public void testProcessExecution() throws Exception { assumeTrue("test requires security manager", System.getSecurityManager() != null); diff --git a/core/src/test/java/org/elasticsearch/indices/analyze/HunspellServiceIT.java b/core/src/test/java/org/elasticsearch/indices/analyze/HunspellServiceIT.java index d21a840c42b..96fc85ad85a 100644 --- a/core/src/test/java/org/elasticsearch/indices/analyze/HunspellServiceIT.java +++ b/core/src/test/java/org/elasticsearch/indices/analyze/HunspellServiceIT.java @@ -29,8 +29,6 @@ import org.elasticsearch.test.ESIntegTestCase.Scope; import org.hamcrest.Matchers; import org.junit.Test; -import java.lang.reflect.Field; - import static org.elasticsearch.indices.analysis.HunspellService.*; import static org.hamcrest.Matchers.notNullValue; @@ -114,10 +112,8 @@ public class HunspellServiceIT extends ESIntegTestCase { } } - // TODO: open up a getter on Dictionary + // TODO: on next upgrade of lucene, just use new getter private void assertIgnoreCase(boolean expected, Dictionary dictionary) throws Exception { - Field f = Dictionary.class.getDeclaredField("ignoreCase"); - f.setAccessible(true); - assertEquals(expected, f.getBoolean(dictionary)); + // assertEquals(expected, dictionary.getIgnoreCase()); } } diff --git a/core/src/test/java/org/elasticsearch/network/DirectBufferNetworkIT.java b/core/src/test/java/org/elasticsearch/network/DirectBufferNetworkIT.java deleted file mode 100644 index 40da9aeca0e..00000000000 --- a/core/src/test/java/org/elasticsearch/network/DirectBufferNetworkIT.java +++ /dev/null @@ -1,146 +0,0 @@ -/* - * Licensed to Elasticsearch under one or more contributor - * license agreements. See the NOTICE file distributed with - * this work for additional information regarding copyright - * ownership. Elasticsearch licenses this file to you under - * the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - -package org.elasticsearch.network; - -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.common.unit.ByteSizeValue; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.node.Node; -import org.elasticsearch.test.ESIntegTestCase; -import org.hamcrest.Matchers; -import org.junit.Test; - -import java.io.ByteArrayOutputStream; -import java.lang.reflect.Field; -import java.nio.ByteBuffer; - -import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertHitCount; - -/** - */ -public class DirectBufferNetworkIT extends ESIntegTestCase { - - @Override - protected Settings nodeSettings(int nodeOrdinal) { - return Settings.builder() - .put(Node.HTTP_ENABLED, true) - .put(super.nodeSettings(nodeOrdinal)).build(); - } - - /** - * This test validates that using large data sets (large docs + large API requests) don't - * cause a large direct byte buffer to be allocated internally in the sun.nio buffer cache. - *

- * See {@link org.elasticsearch.common.netty.NettyUtils#DEFAULT_GATHERING} for more info. - */ - @Test - public void verifySaneDirectBufferAllocations() throws Exception { - assumeTrue("test cannot run with security manager enabled", System.getSecurityManager() == null); - createIndex("test"); - - int estimatedBytesSize = scaledRandomIntBetween(ByteSizeValue.parseBytesSizeValue("1.1mb", "estimatedBytesSize").bytesAsInt(), - ByteSizeValue.parseBytesSizeValue("1.5mb", "estimatedBytesSize").bytesAsInt()); - byte[] data = new byte[estimatedBytesSize]; - getRandom().nextBytes(data); - - ByteArrayOutputStream docOut = new ByteArrayOutputStream(); - // we use smile to automatically use the binary mapping - XContentBuilder doc = XContentFactory.smileBuilder(docOut).startObject().startObject("doc").field("value", data).endObject(); - doc.close(); - byte[] docBytes = docOut.toByteArray(); - - int numDocs = randomIntBetween(2, 5); - logger.info("indexing [{}] docs, each with size [{}]", numDocs, estimatedBytesSize); - IndexRequestBuilder[] builders = new IndexRequestBuilder[numDocs]; - for (int i = 0; i < numDocs; ++i) { - builders[i] = client().prepareIndex("test", "type").setSource(docBytes); - } - indexRandom(true, builders); - logger.info("done indexing"); - - logger.info("executing random client search for all docs"); - assertHitCount(client().prepareSearch("test").setFrom(0).setSize(numDocs).get(), numDocs); - logger.info("executing transport client search for all docs"); - assertHitCount(internalCluster().transportClient().prepareSearch("test").setFrom(0).setSize(numDocs).get(), numDocs); - - logger.info("executing HTTP search for all docs"); - // simulate large HTTP call as well - httpClient().method("GET").path("/test/_search").addParam("size", Integer.toString(numDocs)).execute(); - - logger.info("validating large direct buffer not allocated"); - validateNoLargeDirectBufferAllocated(); - } - - /** - * Validates that all the thread local allocated ByteBuffer in sun.nio under the Util$BufferCache - * are not greater than 1mb. - */ - private void validateNoLargeDirectBufferAllocated() throws Exception { - // Make the fields in the Thread class that store ThreadLocals - // accessible - Field threadLocalsField = Thread.class.getDeclaredField("threadLocals"); - threadLocalsField.setAccessible(true); - // Make the underlying array of ThreadLoad.ThreadLocalMap.Entry objects - // accessible - Class tlmClass = Class.forName("java.lang.ThreadLocal$ThreadLocalMap"); - Field tableField = tlmClass.getDeclaredField("table"); - tableField.setAccessible(true); - - for (Thread thread : Thread.getAllStackTraces().keySet()) { - if (thread == null) { - continue; - } - Object threadLocalMap = threadLocalsField.get(thread); - if (threadLocalMap == null) { - continue; - } - Object[] table = (Object[]) tableField.get(threadLocalMap); - if (table == null) { - continue; - } - for (Object entry : table) { - if (entry == null) { - continue; - } - Field valueField = entry.getClass().getDeclaredField("value"); - valueField.setAccessible(true); - Object value = valueField.get(entry); - if (value == null) { - continue; - } - if (!value.getClass().getName().equals("sun.nio.ch.Util$BufferCache")) { - continue; - } - Field buffersField = value.getClass().getDeclaredField("buffers"); - buffersField.setAccessible(true); - Object[] buffers = (Object[]) buffersField.get(value); - for (Object buffer : buffers) { - if (buffer == null) { - continue; - } - assertThat(((ByteBuffer) buffer).capacity(), Matchers.lessThan(1 * 1024 * 1024)); - } - } - } - - } -} diff --git a/core/src/test/java/org/elasticsearch/search/suggest/completion/CompletionPostingsFormatTests.java b/core/src/test/java/org/elasticsearch/search/suggest/completion/CompletionPostingsFormatTests.java index 672f8bbc673..ff672fbab50 100644 --- a/core/src/test/java/org/elasticsearch/search/suggest/completion/CompletionPostingsFormatTests.java +++ b/core/src/test/java/org/elasticsearch/search/suggest/completion/CompletionPostingsFormatTests.java @@ -244,11 +244,9 @@ public class CompletionPostingsFormatTests extends ESTestCase { fieldType.setProvider(currentProvider); final CompletionFieldMapper mapper = new CompletionFieldMapper("foo", fieldType, Integer.MAX_VALUE, indexSettings, FieldMapper.MultiFields.empty(), null); Lookup buildAnalyzingLookup = buildAnalyzingLookup(mapper, titles, titles, weights); - Field field = buildAnalyzingLookup.getClass().getDeclaredField("maxAnalyzedPathsForOneInput"); - field.setAccessible(true); - Field refField = reference.getClass().getDeclaredField("maxAnalyzedPathsForOneInput"); - refField.setAccessible(true); - assertThat(refField.get(reference), equalTo(field.get(buildAnalyzingLookup))); + if (buildAnalyzingLookup instanceof XAnalyzingSuggester) { + assertEquals(reference.getMaxAnalyzedPathsForOneInput(), ((XAnalyzingSuggester) buildAnalyzingLookup).getMaxAnalyzedPathsForOneInput()); + } for (int i = 0; i < titles.length; i++) { int res = between(1, 10); diff --git a/core/src/test/java/org/elasticsearch/test/ESTestCase.java b/core/src/test/java/org/elasticsearch/test/ESTestCase.java index 47aef7f394d..dd60e960722 100644 --- a/core/src/test/java/org/elasticsearch/test/ESTestCase.java +++ b/core/src/test/java/org/elasticsearch/test/ESTestCase.java @@ -33,6 +33,7 @@ import com.carrotsearch.randomizedtesting.rules.TestRuleAdapter; import org.apache.lucene.uninverting.UninvertingReader; import org.apache.lucene.util.LuceneTestCase; import org.apache.lucene.util.LuceneTestCase.SuppressCodecs; +import org.apache.lucene.util.TestRuleMarkFailure; import org.apache.lucene.util.TestUtil; import org.apache.lucene.util.TimeUnits; import org.elasticsearch.Version; @@ -643,4 +644,9 @@ public abstract class ESTestCase extends LuceneTestCase { sb.append("]"); assertThat(count + " files exist that should have been cleaned:\n" + sb.toString(), count, equalTo(0)); } + + /** Returns the suite failure marker: internal use only! */ + public static TestRuleMarkFailure getSuiteFailureMarker() { + return suiteFailureMarker; + } } diff --git a/core/src/test/java/org/elasticsearch/test/store/MockFSDirectoryService.java b/core/src/test/java/org/elasticsearch/test/store/MockFSDirectoryService.java index 7d90d84304e..86efa380bd6 100644 --- a/core/src/test/java/org/elasticsearch/test/store/MockFSDirectoryService.java +++ b/core/src/test/java/org/elasticsearch/test/store/MockFSDirectoryService.java @@ -49,7 +49,6 @@ import org.junit.Assert; import java.io.Closeable; import java.io.IOException; import java.io.PrintStream; -import java.lang.reflect.Field; import java.nio.file.Path; import java.util.*; @@ -237,13 +236,7 @@ public class MockFSDirectoryService extends FsDirectoryService { public CloseableDirectory(BaseDirectoryWrapper dir) { this.dir = dir; - try { - final Field suiteFailureMarker = LuceneTestCase.class.getDeclaredField("suiteFailureMarker"); - suiteFailureMarker.setAccessible(true); - this.failureMarker = (TestRuleMarkFailure) suiteFailureMarker.get(LuceneTestCase.class); - } catch (Throwable e) { - throw new ElasticsearchException("foo", e); - } + this.failureMarker = ESTestCase.getSuiteFailureMarker(); } @Override diff --git a/dev-tools/src/main/resources/forbidden/all-signatures.txt b/dev-tools/src/main/resources/forbidden/all-signatures.txt index e50f672850c..443f6272656 100644 --- a/dev-tools/src/main/resources/forbidden/all-signatures.txt +++ b/dev-tools/src/main/resources/forbidden/all-signatures.txt @@ -107,3 +107,7 @@ com.google.common.util.concurrent.SettableFuture com.google.common.util.concurrent.Futures com.google.common.util.concurrent.MoreExecutors com.google.common.collect.ImmutableSortedMap + +@defaultMessage Do not violate java's access system +java.lang.reflect.AccessibleObject#setAccessible(boolean) +java.lang.reflect.AccessibleObject#setAccessible(java.lang.reflect.AccessibleObject[], boolean) diff --git a/dev-tools/src/main/resources/forbidden/core-signatures.txt b/dev-tools/src/main/resources/forbidden/core-signatures.txt index 19af292b737..3a925e64d3c 100644 --- a/dev-tools/src/main/resources/forbidden/core-signatures.txt +++ b/dev-tools/src/main/resources/forbidden/core-signatures.txt @@ -83,7 +83,3 @@ java.util.concurrent.Future#cancel(boolean) @defaultMessage Don't try reading from paths that are not configured in Environment, resolve from Environment instead org.elasticsearch.common.io.PathUtils#get(java.lang.String, java.lang.String[]) org.elasticsearch.common.io.PathUtils#get(java.net.URI) - -@defaultMessage Do not violate java's access system -java.lang.reflect.AccessibleObject#setAccessible(boolean) -java.lang.reflect.AccessibleObject#setAccessible(java.lang.reflect.AccessibleObject[], boolean) From 7703aab95af4bc146bbcbc74b4dd5c3c8f880eb4 Mon Sep 17 00:00:00 2001 From: Robert Muir Date: Sat, 12 Sep 2015 20:49:20 -0400 Subject: [PATCH 2/2] support mocking these components in tests without setAccessible --- .../org/elasticsearch/http/netty/NettyHttpServerTransport.java | 3 ++- .../java/org/elasticsearch/script/ScriptContextRegistry.java | 2 +- .../java/org/elasticsearch/transport/netty/NettyTransport.java | 3 ++- 3 files changed, 5 insertions(+), 3 deletions(-) diff --git a/core/src/main/java/org/elasticsearch/http/netty/NettyHttpServerTransport.java b/core/src/main/java/org/elasticsearch/http/netty/NettyHttpServerTransport.java index fa0c1a9fb28..634e38fe24b 100644 --- a/core/src/main/java/org/elasticsearch/http/netty/NettyHttpServerTransport.java +++ b/core/src/main/java/org/elasticsearch/http/netty/NettyHttpServerTransport.java @@ -135,7 +135,8 @@ public class NettyHttpServerTransport extends AbstractLifecycleComponent serverChannels = new ArrayList<>(); - protected OpenChannelsHandler serverOpenChannels; + // package private for testing + OpenChannelsHandler serverOpenChannels; protected volatile HttpServerAdapter httpServerAdapter; diff --git a/core/src/main/java/org/elasticsearch/script/ScriptContextRegistry.java b/core/src/main/java/org/elasticsearch/script/ScriptContextRegistry.java index 226c931269e..e9681c74834 100644 --- a/core/src/main/java/org/elasticsearch/script/ScriptContextRegistry.java +++ b/core/src/main/java/org/elasticsearch/script/ScriptContextRegistry.java @@ -36,7 +36,7 @@ public final class ScriptContextRegistry { private final ImmutableMap scriptContexts; - ScriptContextRegistry(Iterable customScriptContexts) { + public ScriptContextRegistry(Iterable customScriptContexts) { Map scriptContexts = new HashMap<>(); for (ScriptContext.Standard scriptContext : ScriptContext.Standard.values()) { scriptContexts.put(scriptContext.getKey(), scriptContext); diff --git a/core/src/main/java/org/elasticsearch/transport/netty/NettyTransport.java b/core/src/main/java/org/elasticsearch/transport/netty/NettyTransport.java index 991484affdf..672de6bf31c 100644 --- a/core/src/main/java/org/elasticsearch/transport/netty/NettyTransport.java +++ b/core/src/main/java/org/elasticsearch/transport/netty/NettyTransport.java @@ -184,7 +184,8 @@ public class NettyTransport extends AbstractLifecycleComponent implem protected final BigArrays bigArrays; protected final ThreadPool threadPool; - protected volatile OpenChannelsHandler serverOpenChannels; + // package private for testing + volatile OpenChannelsHandler serverOpenChannels; protected volatile ClientBootstrap clientBootstrap; // node id to actual channel protected final ConcurrentMap connectedNodes = newConcurrentMap();