ensure the XContentBuilder is always closed in RestBuilderListener
There may be cases where the XContentBuilder is not used and therefore it never gets closed, which can cause a leak of bytes. This change moves the creation of the builder into a try with resources block and adds an assertion to verify that we always consume the bytes in our code; the try-with resources provides protections against memory leaks caused by plugins, which do not test this.
This commit is contained in:
parent
ef192ff2cf
commit
6e7e89159b
|
@ -94,4 +94,9 @@ public interface XContentGenerator extends Closeable, Flushable {
|
||||||
|
|
||||||
void copyCurrentStructure(XContentParser parser) throws IOException;
|
void copyCurrentStructure(XContentParser parser) throws IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns {@code true} if this XContentGenerator has been closed. A closed generator can not do any more output.
|
||||||
|
*/
|
||||||
|
boolean isClosed();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -419,4 +419,8 @@ public class JsonXContentGenerator implements XContentGenerator {
|
||||||
generator.close();
|
generator.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isClosed() {
|
||||||
|
return generator.isClosed();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -34,11 +34,22 @@ public abstract class RestBuilderListener<Response> extends RestResponseListener
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final RestResponse buildResponse(Response response) throws Exception {
|
public final RestResponse buildResponse(Response response) throws Exception {
|
||||||
return buildResponse(response, channel.newBuilder());
|
try (XContentBuilder builder = channel.newBuilder()) {
|
||||||
|
final RestResponse restResponse = buildResponse(response, builder);
|
||||||
|
assert assertBuilderClosed(builder);
|
||||||
|
return restResponse;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Builds a response to send back over the channel.
|
* Builds a response to send back over the channel. Implementors should ensure that they close the provided {@link XContentBuilder}
|
||||||
|
* using the {@link XContentBuilder#close()} method.
|
||||||
*/
|
*/
|
||||||
public abstract RestResponse buildResponse(Response response, XContentBuilder builder) throws Exception;
|
public abstract RestResponse buildResponse(Response response, XContentBuilder builder) throws Exception;
|
||||||
|
|
||||||
|
// pkg private method that we can override for testing
|
||||||
|
boolean assertBuilderClosed(XContentBuilder xContentBuilder) {
|
||||||
|
assert xContentBuilder.generator().isClosed() : "callers should ensure the XContentBuilder is closed themselves";
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,90 @@
|
||||||
|
/*
|
||||||
|
* 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.rest.action;
|
||||||
|
|
||||||
|
import org.elasticsearch.common.bytes.BytesArray;
|
||||||
|
import org.elasticsearch.common.xcontent.XContentBuilder;
|
||||||
|
import org.elasticsearch.rest.BytesRestResponse;
|
||||||
|
import org.elasticsearch.rest.RestResponse;
|
||||||
|
import org.elasticsearch.rest.RestStatus;
|
||||||
|
import org.elasticsearch.test.ESTestCase;
|
||||||
|
import org.elasticsearch.test.rest.FakeRestChannel;
|
||||||
|
import org.elasticsearch.test.rest.FakeRestRequest;
|
||||||
|
import org.elasticsearch.transport.TransportResponse;
|
||||||
|
import org.elasticsearch.transport.TransportResponse.Empty;
|
||||||
|
|
||||||
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
|
public class RestBuilderListenerTests extends ESTestCase {
|
||||||
|
|
||||||
|
public void testXContentBuilderClosedInBuildResponse() throws Exception {
|
||||||
|
AtomicReference<XContentBuilder> builderAtomicReference = new AtomicReference<>();
|
||||||
|
RestBuilderListener<TransportResponse.Empty> builderListener =
|
||||||
|
new RestBuilderListener<Empty>(new FakeRestChannel(new FakeRestRequest(), randomBoolean(), 1)) {
|
||||||
|
@Override
|
||||||
|
public RestResponse buildResponse(Empty empty, XContentBuilder builder) throws Exception {
|
||||||
|
builderAtomicReference.set(builder);
|
||||||
|
builder.close();
|
||||||
|
return new BytesRestResponse(RestStatus.OK, BytesRestResponse.TEXT_CONTENT_TYPE, BytesArray.EMPTY);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
builderListener.buildResponse(Empty.INSTANCE);
|
||||||
|
assertNotNull(builderAtomicReference.get());
|
||||||
|
assertTrue(builderAtomicReference.get().generator().isClosed());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testXContentBuilderNotClosedInBuildResponseAssertionsDisabled() throws Exception {
|
||||||
|
AtomicReference<XContentBuilder> builderAtomicReference = new AtomicReference<>();
|
||||||
|
RestBuilderListener<TransportResponse.Empty> builderListener =
|
||||||
|
new RestBuilderListener<Empty>(new FakeRestChannel(new FakeRestRequest(), randomBoolean(), 1)) {
|
||||||
|
@Override
|
||||||
|
public RestResponse buildResponse(Empty empty, XContentBuilder builder) throws Exception {
|
||||||
|
builderAtomicReference.set(builder);
|
||||||
|
return new BytesRestResponse(RestStatus.OK, BytesRestResponse.TEXT_CONTENT_TYPE, BytesArray.EMPTY);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
boolean assertBuilderClosed(XContentBuilder xContentBuilder) {
|
||||||
|
// don't check the actual builder being closed so we can test auto close
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
builderListener.buildResponse(Empty.INSTANCE);
|
||||||
|
assertNotNull(builderAtomicReference.get());
|
||||||
|
assertTrue(builderAtomicReference.get().generator().isClosed());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testXContentBuilderNotClosedInBuildResponseAssertionsEnabled() throws Exception {
|
||||||
|
assumeTrue("tests are not being run with assertions", RestBuilderListener.class.desiredAssertionStatus());
|
||||||
|
|
||||||
|
RestBuilderListener<TransportResponse.Empty> builderListener =
|
||||||
|
new RestBuilderListener<Empty>(new FakeRestChannel(new FakeRestRequest(), randomBoolean(), 1)) {
|
||||||
|
@Override
|
||||||
|
public RestResponse buildResponse(Empty empty, XContentBuilder builder) throws Exception {
|
||||||
|
return new BytesRestResponse(RestStatus.OK, BytesRestResponse.TEXT_CONTENT_TYPE, BytesArray.EMPTY);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
AssertionError error = expectThrows(AssertionError.class, () -> builderListener.buildResponse(Empty.INSTANCE));
|
||||||
|
assertEquals("callers should ensure the XContentBuilder is closed themselves", error.getMessage());
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue