mirror of https://github.com/apache/lucene.git
SOLR-9717: Refactor '/export' to not hardcode the JSON output and to use an API
This commit is contained in:
parent
cbf8235e57
commit
ef074a61f8
|
@ -141,6 +141,8 @@ Other Changes
|
||||||
* SOLR-9720: Refactor Responsewriters to remove dependencies on TupleStream,
|
* SOLR-9720: Refactor Responsewriters to remove dependencies on TupleStream,
|
||||||
Tuple, Explanation (noble)
|
Tuple, Explanation (noble)
|
||||||
|
|
||||||
|
* SOLR-9717: Refactor '/export' to not hardcode the JSON output and to use an API (noble)
|
||||||
|
|
||||||
================== 6.3.0 ==================
|
================== 6.3.0 ==================
|
||||||
|
|
||||||
Consult the LUCENE_CHANGES.txt file for additional, low level, changes in this release.
|
Consult the LUCENE_CHANGES.txt file for additional, low level, changes in this release.
|
||||||
|
|
|
@ -110,7 +110,6 @@ import org.apache.solr.response.RubyResponseWriter;
|
||||||
import org.apache.solr.response.SchemaXmlResponseWriter;
|
import org.apache.solr.response.SchemaXmlResponseWriter;
|
||||||
import org.apache.solr.response.SmileResponseWriter;
|
import org.apache.solr.response.SmileResponseWriter;
|
||||||
import org.apache.solr.response.SolrQueryResponse;
|
import org.apache.solr.response.SolrQueryResponse;
|
||||||
import org.apache.solr.response.SortingResponseWriter;
|
|
||||||
import org.apache.solr.response.XMLResponseWriter;
|
import org.apache.solr.response.XMLResponseWriter;
|
||||||
import org.apache.solr.response.transform.TransformerFactory;
|
import org.apache.solr.response.transform.TransformerFactory;
|
||||||
import org.apache.solr.rest.ManagedResourceStorage;
|
import org.apache.solr.rest.ManagedResourceStorage;
|
||||||
|
@ -2332,7 +2331,6 @@ public final class SolrCore implements SolrInfoMBean, Closeable {
|
||||||
m.put("raw", new RawResponseWriter());
|
m.put("raw", new RawResponseWriter());
|
||||||
m.put(CommonParams.JAVABIN, new BinaryResponseWriter());
|
m.put(CommonParams.JAVABIN, new BinaryResponseWriter());
|
||||||
m.put("csv", new CSVResponseWriter());
|
m.put("csv", new CSVResponseWriter());
|
||||||
m.put("xsort", new SortingResponseWriter());
|
|
||||||
m.put("schema.xml", new SchemaXmlResponseWriter());
|
m.put("schema.xml", new SchemaXmlResponseWriter());
|
||||||
m.put("smile", new SmileResponseWriter());
|
m.put("smile", new SmileResponseWriter());
|
||||||
m.put(ReplicationHandler.FILE_STREAM, getFileStreamWriter());
|
m.put(ReplicationHandler.FILE_STREAM, getFileStreamWriter());
|
||||||
|
@ -2350,12 +2348,21 @@ public final class SolrCore implements SolrInfoMBean, Closeable {
|
||||||
@Override
|
@Override
|
||||||
public void write(OutputStream out, SolrQueryRequest req, SolrQueryResponse response) throws IOException {
|
public void write(OutputStream out, SolrQueryRequest req, SolrQueryResponse response) throws IOException {
|
||||||
RawWriter rawWriter = (RawWriter) response.getValues().get(ReplicationHandler.FILE_STREAM);
|
RawWriter rawWriter = (RawWriter) response.getValues().get(ReplicationHandler.FILE_STREAM);
|
||||||
if(rawWriter!=null) rawWriter.write(out);
|
if (rawWriter != null) {
|
||||||
|
rawWriter.write(out);
|
||||||
|
if (rawWriter instanceof Closeable) ((Closeable) rawWriter).close();
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getContentType(SolrQueryRequest request, SolrQueryResponse response) {
|
public String getContentType(SolrQueryRequest request, SolrQueryResponse response) {
|
||||||
return BinaryResponseParser.BINARY_CONTENT_TYPE;
|
RawWriter rawWriter = (RawWriter) response.getValues().get(ReplicationHandler.FILE_STREAM);
|
||||||
|
if (rawWriter != null) {
|
||||||
|
return rawWriter.getContentType();
|
||||||
|
} else {
|
||||||
|
return BinaryResponseParser.BINARY_CONTENT_TYPE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -2365,6 +2372,9 @@ public final class SolrCore implements SolrInfoMBean, Closeable {
|
||||||
}
|
}
|
||||||
|
|
||||||
public interface RawWriter {
|
public interface RawWriter {
|
||||||
|
default String getContentType() {
|
||||||
|
return BinaryResponseParser.BINARY_CONTENT_TYPE;
|
||||||
|
}
|
||||||
void write(OutputStream os) throws IOException ;
|
void write(OutputStream os) throws IOException ;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,48 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF 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.apache.solr.handler;
|
||||||
|
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.apache.solr.common.params.CommonParams;
|
||||||
|
import org.apache.solr.common.params.MapSolrParams;
|
||||||
|
import org.apache.solr.common.params.SolrParams;
|
||||||
|
import org.apache.solr.handler.component.SearchHandler;
|
||||||
|
import org.apache.solr.request.SolrQueryRequest;
|
||||||
|
import org.apache.solr.response.SolrQueryResponse;
|
||||||
|
|
||||||
|
import static org.apache.solr.common.params.CommonParams.JSON;
|
||||||
|
|
||||||
|
public class ExportHandler extends SearchHandler {
|
||||||
|
@Override
|
||||||
|
public void handleRequestBody(SolrQueryRequest req, SolrQueryResponse rsp) throws Exception {
|
||||||
|
try {
|
||||||
|
super.handleRequestBody(req, rsp);
|
||||||
|
} catch (Exception e) {
|
||||||
|
rsp.setException(e);
|
||||||
|
}
|
||||||
|
String wt = req.getParams().get(CommonParams.WT, JSON);
|
||||||
|
if("xsort".equals(wt)) wt = JSON;
|
||||||
|
Map<String, String> map = new HashMap<>(1);
|
||||||
|
map.put(CommonParams.WT, ReplicationHandler.FILE_STREAM);
|
||||||
|
req.setParams(SolrParams.wrapDefaults(new MapSolrParams(map),req.getParams()));
|
||||||
|
rsp.add(ReplicationHandler.FILE_STREAM, new ExportWriter(req, rsp, wt));
|
||||||
|
}
|
||||||
|
}
|
|
@ -24,7 +24,11 @@ import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
|
import org.apache.solr.common.IteratorWriter;
|
||||||
|
import org.apache.solr.common.MapWriter.EntryWriter;
|
||||||
|
import org.apache.solr.common.PushWriter;
|
||||||
import org.apache.solr.common.SolrDocument;
|
import org.apache.solr.common.SolrDocument;
|
||||||
|
import org.apache.solr.common.MapWriter;
|
||||||
import org.apache.solr.common.params.SolrParams;
|
import org.apache.solr.common.params.SolrParams;
|
||||||
import org.apache.solr.common.util.NamedList;
|
import org.apache.solr.common.util.NamedList;
|
||||||
import org.apache.solr.common.util.SimpleOrderedMap;
|
import org.apache.solr.common.util.SimpleOrderedMap;
|
||||||
|
@ -74,6 +78,11 @@ public class JSONResponseWriter implements QueryResponseWriter {
|
||||||
public String getContentType(SolrQueryRequest request, SolrQueryResponse response) {
|
public String getContentType(SolrQueryRequest request, SolrQueryResponse response) {
|
||||||
return contentType;
|
return contentType;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static PushWriter getPushWriter(Writer writer, SolrQueryRequest req, SolrQueryResponse rsp) {
|
||||||
|
return new JSONWriter(writer, req, rsp);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class JSONWriter extends TextResponseWriter {
|
class JSONWriter extends TextResponseWriter {
|
||||||
|
@ -507,6 +516,53 @@ class JSONWriter extends TextResponseWriter {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeIterator(IteratorWriter val) throws IOException {
|
||||||
|
writeArrayOpener(-1);
|
||||||
|
incLevel();
|
||||||
|
val.writeIter(new IteratorWriter.ItemWriter() {
|
||||||
|
boolean first = true;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public IteratorWriter.ItemWriter add(Object o) throws IOException {
|
||||||
|
if (!first) {
|
||||||
|
JSONWriter.this.indent();
|
||||||
|
JSONWriter.this.writeArraySeparator();
|
||||||
|
}
|
||||||
|
JSONWriter.this.writeVal(null, o);
|
||||||
|
first = false;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
decLevel();
|
||||||
|
writeArrayCloser();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeMap(MapWriter val)
|
||||||
|
throws IOException {
|
||||||
|
writeMapOpener(-1);
|
||||||
|
incLevel();
|
||||||
|
|
||||||
|
val.writeMap(new EntryWriter() {
|
||||||
|
boolean isFirst = true;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public EntryWriter put(String k, Object v) throws IOException {
|
||||||
|
if (isFirst) {
|
||||||
|
isFirst = false;
|
||||||
|
} else {
|
||||||
|
JSONWriter.this.writeMapSeparator();
|
||||||
|
}
|
||||||
|
if (doIndent) JSONWriter.this.indent();
|
||||||
|
JSONWriter.this.writeKey(k, true);
|
||||||
|
JSONWriter.this.writeVal(k, v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
decLevel();
|
||||||
|
writeMapCloser();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void writeMap(String name, Map val, boolean excludeOuter, boolean isFirstVal) throws IOException {
|
public void writeMap(String name, Map val, boolean excludeOuter, boolean isFirstVal) throws IOException {
|
||||||
|
|
|
@ -31,9 +31,12 @@ import org.apache.lucene.document.Document;
|
||||||
import org.apache.lucene.index.IndexableField;
|
import org.apache.lucene.index.IndexableField;
|
||||||
import org.apache.lucene.util.BytesRef;
|
import org.apache.lucene.util.BytesRef;
|
||||||
import org.apache.solr.common.EnumFieldValue;
|
import org.apache.solr.common.EnumFieldValue;
|
||||||
|
import org.apache.solr.common.IteratorWriter;
|
||||||
import org.apache.solr.common.MapSerializable;
|
import org.apache.solr.common.MapSerializable;
|
||||||
|
import org.apache.solr.common.PushWriter;
|
||||||
import org.apache.solr.common.SolrDocument;
|
import org.apache.solr.common.SolrDocument;
|
||||||
import org.apache.solr.common.SolrDocumentList;
|
import org.apache.solr.common.SolrDocumentList;
|
||||||
|
import org.apache.solr.common.MapWriter;
|
||||||
import org.apache.solr.common.params.CommonParams;
|
import org.apache.solr.common.params.CommonParams;
|
||||||
import org.apache.solr.common.util.Base64;
|
import org.apache.solr.common.util.Base64;
|
||||||
import org.apache.solr.common.util.NamedList;
|
import org.apache.solr.common.util.NamedList;
|
||||||
|
@ -48,7 +51,7 @@ import org.apache.solr.util.FastWriter;
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public abstract class TextResponseWriter {
|
public abstract class TextResponseWriter implements PushWriter {
|
||||||
|
|
||||||
// indent up to 40 spaces
|
// indent up to 40 spaces
|
||||||
static final char[] indentChars = new char[81];
|
static final char[] indentChars = new char[81];
|
||||||
|
@ -138,19 +141,19 @@ public abstract class TextResponseWriter {
|
||||||
writeStr(name, f.stringValue(), true);
|
writeStr(name, f.stringValue(), true);
|
||||||
}
|
}
|
||||||
} else if (val instanceof Number) {
|
} else if (val instanceof Number) {
|
||||||
writeNumber(name, (Number)val);
|
writeNumber(name, (Number) val);
|
||||||
} else if (val instanceof Boolean) {
|
} else if (val instanceof Boolean) {
|
||||||
writeBool(name, (Boolean)val);
|
writeBool(name, (Boolean) val);
|
||||||
} else if (val instanceof Date) {
|
} else if (val instanceof Date) {
|
||||||
writeDate(name,(Date)val);
|
writeDate(name, (Date) val);
|
||||||
} else if (val instanceof Document) {
|
} else if (val instanceof Document) {
|
||||||
SolrDocument doc = DocsStreamer.getDoc((Document) val, schema);
|
SolrDocument doc = DocsStreamer.getDoc((Document) val, schema);
|
||||||
writeSolrDocument(name, doc,returnFields, 0 );
|
writeSolrDocument(name, doc, returnFields, 0);
|
||||||
} else if (val instanceof SolrDocument) {
|
} else if (val instanceof SolrDocument) {
|
||||||
writeSolrDocument(name, (SolrDocument)val,returnFields, 0);
|
writeSolrDocument(name, (SolrDocument) val, returnFields, 0);
|
||||||
} else if (val instanceof ResultContext) {
|
} else if (val instanceof ResultContext) {
|
||||||
// requires access to IndexReader
|
// requires access to IndexReader
|
||||||
writeDocuments(name, (ResultContext)val);
|
writeDocuments(name, (ResultContext) val);
|
||||||
} else if (val instanceof DocList) {
|
} else if (val instanceof DocList) {
|
||||||
// Should not happen normally
|
// Should not happen normally
|
||||||
ResultContext ctx = new BasicResultContext((DocList)val, returnFields, null, null, req);
|
ResultContext ctx = new BasicResultContext((DocList)val, returnFields, null, null, req);
|
||||||
|
@ -168,6 +171,8 @@ public abstract class TextResponseWriter {
|
||||||
writeNamedList(name, (NamedList)val);
|
writeNamedList(name, (NamedList)val);
|
||||||
} else if (val instanceof Path) {
|
} else if (val instanceof Path) {
|
||||||
writeStr(name, ((Path) val).toAbsolutePath().toString(), true);
|
writeStr(name, ((Path) val).toAbsolutePath().toString(), true);
|
||||||
|
} else if (val instanceof IteratorWriter) {
|
||||||
|
writeIterator((IteratorWriter) val);
|
||||||
} else if (val instanceof Iterable) {
|
} else if (val instanceof Iterable) {
|
||||||
writeArray(name,((Iterable)val).iterator());
|
writeArray(name,((Iterable)val).iterator());
|
||||||
} else if (val instanceof Object[]) {
|
} else if (val instanceof Object[]) {
|
||||||
|
@ -184,6 +189,8 @@ public abstract class TextResponseWriter {
|
||||||
writeStr(name, val.toString(), true);
|
writeStr(name, val.toString(), true);
|
||||||
} else if (val instanceof WriteableValue) {
|
} else if (val instanceof WriteableValue) {
|
||||||
((WriteableValue)val).write(name, this);
|
((WriteableValue)val).write(name, this);
|
||||||
|
} else if (val instanceof MapWriter) {
|
||||||
|
writeMap((MapWriter) val);
|
||||||
} else if (val instanceof MapSerializable) {
|
} else if (val instanceof MapSerializable) {
|
||||||
//todo find a better way to reuse the map more efficiently
|
//todo find a better way to reuse the map more efficiently
|
||||||
writeMap(name, ((MapSerializable) val).toMap(new LinkedHashMap<>()), false, true);
|
writeMap(name, ((MapSerializable) val).toMap(new LinkedHashMap<>()), false, true);
|
||||||
|
@ -192,6 +199,15 @@ public abstract class TextResponseWriter {
|
||||||
writeStr(name, val.getClass().getName() + ':' + val.toString(), true);
|
writeStr(name, val.getClass().getName() + ':' + val.toString(), true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@Override
|
||||||
|
public void writeMap(MapWriter mw) throws IOException {
|
||||||
|
//todo
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeIterator(IteratorWriter iw) throws IOException {
|
||||||
|
/*todo*/
|
||||||
|
}
|
||||||
|
|
||||||
protected void writeBool(String name , Boolean val) throws IOException {
|
protected void writeBool(String name , Boolean val) throws IOException {
|
||||||
writeBool(name, val.toString());
|
writeBool(name, val.toString());
|
||||||
|
|
|
@ -92,14 +92,16 @@
|
||||||
"useParams":"_ADMIN_FILE"
|
"useParams":"_ADMIN_FILE"
|
||||||
},
|
},
|
||||||
"/export": {
|
"/export": {
|
||||||
"class": "solr.SearchHandler",
|
"class": "solr.ExportHandler",
|
||||||
"useParams":"_EXPORT",
|
"useParams":"_EXPORT",
|
||||||
"components": [
|
"components": [
|
||||||
"query"
|
"query"
|
||||||
],
|
],
|
||||||
|
"defaults": {
|
||||||
|
"wt": "json"
|
||||||
|
},
|
||||||
"invariants": {
|
"invariants": {
|
||||||
"rq": "{!xport}",
|
"rq": "{!xport}",
|
||||||
"wt": "xsort",
|
|
||||||
"distrib": false
|
"distrib": false
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
|
@ -97,7 +97,7 @@ public class SolrCoreTest extends SolrTestCaseJ4 {
|
||||||
++ihCount; assertEquals(pathToClassMap.get("/admin/system"), "solr.SystemInfoHandler");
|
++ihCount; assertEquals(pathToClassMap.get("/admin/system"), "solr.SystemInfoHandler");
|
||||||
++ihCount; assertEquals(pathToClassMap.get("/admin/threads"), "solr.ThreadDumpHandler");
|
++ihCount; assertEquals(pathToClassMap.get("/admin/threads"), "solr.ThreadDumpHandler");
|
||||||
++ihCount; assertEquals(pathToClassMap.get("/config"), "solr.SolrConfigHandler");
|
++ihCount; assertEquals(pathToClassMap.get("/config"), "solr.SolrConfigHandler");
|
||||||
++ihCount; assertEquals(pathToClassMap.get("/export"), "solr.SearchHandler");
|
++ihCount; assertEquals(pathToClassMap.get("/export"), "solr.ExportHandler");
|
||||||
++ihCount; assertEquals(pathToClassMap.get("/terms"), "solr.SearchHandler");
|
++ihCount; assertEquals(pathToClassMap.get("/terms"), "solr.SearchHandler");
|
||||||
++ihCount; assertEquals(pathToClassMap.get("/get"), "solr.RealTimeGetHandler");
|
++ihCount; assertEquals(pathToClassMap.get("/get"), "solr.RealTimeGetHandler");
|
||||||
++ihCount; assertEquals(pathToClassMap.get(ReplicationHandler.PATH), "solr.ReplicationHandler");
|
++ihCount; assertEquals(pathToClassMap.get(ReplicationHandler.PATH), "solr.ReplicationHandler");
|
||||||
|
|
|
@ -184,12 +184,15 @@ public class JSONWriterTest extends SolrTestCaseJ4 {
|
||||||
methodsExpectedNotOverriden.add("writeArrayOpener");
|
methodsExpectedNotOverriden.add("writeArrayOpener");
|
||||||
methodsExpectedNotOverriden.add("writeArraySeparator");
|
methodsExpectedNotOverriden.add("writeArraySeparator");
|
||||||
methodsExpectedNotOverriden.add("writeArrayCloser");
|
methodsExpectedNotOverriden.add("writeArrayCloser");
|
||||||
|
methodsExpectedNotOverriden.add("public void org.apache.solr.response.JSONWriter.writeMap(org.apache.solr.common.MapWriter) throws java.io.IOException");
|
||||||
|
methodsExpectedNotOverriden.add("public void org.apache.solr.response.JSONWriter.writeIterator(org.apache.solr.common.IteratorWriter) throws java.io.IOException");
|
||||||
|
|
||||||
final Class<?> subClass = ArrayOfNamedValuePairJSONWriter.class;
|
final Class<?> subClass = ArrayOfNamedValuePairJSONWriter.class;
|
||||||
final Class<?> superClass = subClass.getSuperclass();
|
final Class<?> superClass = subClass.getSuperclass();
|
||||||
|
|
||||||
for (final Method superClassMethod : superClass.getDeclaredMethods()) {
|
for (final Method superClassMethod : superClass.getDeclaredMethods()) {
|
||||||
final String methodName = superClassMethod.getName();
|
final String methodName = superClassMethod.getName();
|
||||||
|
final String methodFullName = superClassMethod.toString();
|
||||||
if (!methodName.startsWith("write")) continue;
|
if (!methodName.startsWith("write")) continue;
|
||||||
|
|
||||||
final int modifiers = superClassMethod.getModifiers();
|
final int modifiers = superClassMethod.getModifiers();
|
||||||
|
@ -197,7 +200,8 @@ public class JSONWriterTest extends SolrTestCaseJ4 {
|
||||||
if (Modifier.isStatic(modifiers)) continue;
|
if (Modifier.isStatic(modifiers)) continue;
|
||||||
if (Modifier.isPrivate(modifiers)) continue;
|
if (Modifier.isPrivate(modifiers)) continue;
|
||||||
|
|
||||||
final boolean expectOverriden = !methodsExpectedNotOverriden.contains(methodName);
|
final boolean expectOverriden = !methodsExpectedNotOverriden.contains(methodName)
|
||||||
|
&& !methodsExpectedNotOverriden.contains(methodFullName);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
final Method subClassMethod = subClass.getDeclaredMethod(
|
final Method subClassMethod = subClass.getDeclaredMethod(
|
||||||
|
@ -215,7 +219,7 @@ public class JSONWriterTest extends SolrTestCaseJ4 {
|
||||||
if (expectOverriden) {
|
if (expectOverriden) {
|
||||||
fail(subClass + " needs to override '" + superClassMethod + "'");
|
fail(subClass + " needs to override '" + superClassMethod + "'");
|
||||||
} else {
|
} else {
|
||||||
assertTrue(methodName+" not found in remaining "+methodsExpectedNotOverriden, methodsExpectedNotOverriden.remove(methodName));
|
assertTrue(methodName+" not found in remaining "+methodsExpectedNotOverriden, methodsExpectedNotOverriden.remove(methodName)|| methodsExpectedNotOverriden.remove(methodFullName));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,11 +17,12 @@
|
||||||
package org.apache.solr.response;
|
package org.apache.solr.response;
|
||||||
|
|
||||||
import org.apache.solr.SolrTestCaseJ4;
|
import org.apache.solr.SolrTestCaseJ4;
|
||||||
|
import org.apache.solr.common.util.Utils;
|
||||||
import org.junit.*;
|
import org.junit.*;
|
||||||
import org.apache.lucene.util.LuceneTestCase.SuppressCodecs;
|
import org.apache.lucene.util.LuceneTestCase.SuppressCodecs;
|
||||||
|
|
||||||
@SuppressCodecs({"Lucene3x", "Lucene40","Lucene41","Lucene42","Lucene45"})
|
@SuppressCodecs({"Lucene3x", "Lucene40","Lucene41","Lucene42","Lucene45"})
|
||||||
public class TestSortingResponseWriter extends SolrTestCaseJ4 {
|
public class TestExportWriter extends SolrTestCaseJ4 {
|
||||||
@BeforeClass
|
@BeforeClass
|
||||||
public static void beforeClass() throws Exception {
|
public static void beforeClass() throws Exception {
|
||||||
System.setProperty("export.test", "true");
|
System.setProperty("export.test", "true");
|
||||||
|
@ -109,67 +110,71 @@ public class TestSortingResponseWriter extends SolrTestCaseJ4 {
|
||||||
//Test single value DocValue output
|
//Test single value DocValue output
|
||||||
String s = h.query(req("q", "id:1", "qt", "/export", "fl", "floatdv,intdv,stringdv,longdv,doubledv", "sort", "intdv asc"));
|
String s = h.query(req("q", "id:1", "qt", "/export", "fl", "floatdv,intdv,stringdv,longdv,doubledv", "sort", "intdv asc"));
|
||||||
|
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":1, \"docs\":[{\"floatdv\":2.1,\"intdv\":1,\"stringdv\":\"hello world\",\"longdv\":323223232323,\"doubledv\":2344.345}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":1, \"docs\":[{\"floatdv\":2.1,\"intdv\":1,\"stringdv\":\"hello world\",\"longdv\":323223232323,\"doubledv\":2344.345}]}}");
|
||||||
|
|
||||||
//Test null value string:
|
//Test null value string:
|
||||||
s = h.query(req("q", "id:7", "qt", "/export", "fl", "floatdv,intdv,stringdv,longdv,doubledv", "sort", "intdv asc"));
|
s = h.query(req("q", "id:7", "qt", "/export", "fl", "floatdv,intdv,stringdv,longdv,doubledv", "sort", "intdv asc"));
|
||||||
|
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":1, \"docs\":[{\"floatdv\":2.1,\"intdv\":7,\"longdv\":323223232323,\"doubledv\":2344.345}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":1, \"docs\":[{\"floatdv\":2.1,\"intdv\":7,\"longdv\":323223232323,\"doubledv\":2344.345}]}}");
|
||||||
|
|
||||||
//Test multiValue docValues output
|
//Test multiValue docValues output
|
||||||
s = h.query(req("q", "id:1", "qt", "/export", "fl", "intdv_m,floatdv_m,doubledv_m,longdv_m,stringdv_m", "sort", "intdv asc"));
|
s = h.query(req("q", "id:1", "qt", "/export", "fl", "intdv_m,floatdv_m,doubledv_m,longdv_m,stringdv_m", "sort", "intdv asc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":1, \"docs\":[{\"intdv_m\":[100,250],\"floatdv_m\":[123.321,345.123],\"doubledv_m\":[3444.222,23232.2],\"longdv_m\":[343332,43434343434],\"stringdv_m\":[\"Everton\",\"liverpool\",\"manchester \\\"city\\\"\"]}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":1, \"docs\":[{\"intdv_m\":[100,250],\"floatdv_m\":[123.321,345.123],\"doubledv_m\":[3444.222,23232.2],\"longdv_m\":[343332,43434343434],\"stringdv_m\":[\"Everton\",\"liverpool\",\"manchester \\\"city\\\"\"]}]}}");
|
||||||
|
|
||||||
//Test multiValues docValues output with nulls
|
//Test multiValues docValues output with nulls
|
||||||
s = h.query(req("q", "id:7", "qt", "/export", "fl", "intdv_m,floatdv_m,doubledv_m,longdv_m,stringdv_m", "sort", "intdv asc"));
|
s = h.query(req("q", "id:7", "qt", "/export", "fl", "intdv_m,floatdv_m,doubledv_m,longdv_m,stringdv_m", "sort", "intdv asc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":1, \"docs\":[{\"floatdv_m\":[123.321,345.123],\"doubledv_m\":[3444.222,23232.2],\"longdv_m\":[343332,43434343434]}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":1, \"docs\":[{\"floatdv_m\":[123.321,345.123],\"doubledv_m\":[3444.222,23232.2],\"longdv_m\":[343332,43434343434]}]}}");
|
||||||
|
|
||||||
//Test single sort param is working
|
//Test single sort param is working
|
||||||
s = h.query(req("q", "id:(1 2)", "qt", "/export", "fl", "intdv", "sort", "intdv desc"));
|
s = h.query(req("q", "id:(1 2)", "qt", "/export", "fl", "intdv", "sort", "intdv desc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":2},{\"intdv\":1}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":2},{\"intdv\":1}]}}");
|
||||||
|
|
||||||
s = h.query(req("q", "id:(1 2)", "qt", "/export", "fl", "intdv", "sort", "intdv asc"));
|
s = h.query(req("q", "id:(1 2)", "qt", "/export", "fl", "intdv", "sort", "intdv asc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":1},{\"intdv\":2}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":1},{\"intdv\":2}]}}");
|
||||||
|
|
||||||
// Test sort on String will null value. Null value should sort last on desc and first on asc.
|
// Test sort on String will null value. Null value should sort last on desc and first on asc.
|
||||||
|
|
||||||
s = h.query(req("q", "id:(1 7)", "qt", "/export", "fl", "intdv", "sort", "stringdv desc"));
|
s = h.query(req("q", "id:(1 7)", "qt", "/export", "fl", "intdv", "sort", "stringdv desc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":1},{\"intdv\":7}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":1},{\"intdv\":7}]}}");
|
||||||
|
|
||||||
s = h.query(req("q", "id:(1 7)", "qt", "/export", "fl", "intdv", "sort", "stringdv asc"));
|
s = h.query(req("q", "id:(1 7)", "qt", "/export", "fl", "intdv", "sort", "stringdv asc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":7},{\"intdv\":1}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":7},{\"intdv\":1}]}}");
|
||||||
|
|
||||||
|
|
||||||
//Test multi-sort params
|
//Test multi-sort params
|
||||||
s = h.query(req("q", "id:(1 2)", "qt", "/export", "fl", "intdv", "sort", "floatdv asc,intdv desc"));
|
s = h.query(req("q", "id:(1 2)", "qt", "/export", "fl", "intdv", "sort", "floatdv asc,intdv desc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":2},{\"intdv\":1}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":2},{\"intdv\":1}]}}");
|
||||||
|
|
||||||
s = h.query(req("q", "id:(1 2)", "qt", "/export", "fl", "intdv", "sort", "floatdv desc,intdv asc"));
|
s = h.query(req("q", "id:(1 2)", "qt", "/export", "fl", "intdv", "sort", "floatdv desc,intdv asc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":1},{\"intdv\":2}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":2, \"docs\":[{\"intdv\":1},{\"intdv\":2}]}}");
|
||||||
|
|
||||||
//Test three sort fields
|
//Test three sort fields
|
||||||
s = h.query(req("q", "id:(1 2 3)", "qt", "/export", "fl", "intdv", "sort", "floatdv asc,stringdv asc,intdv desc"));
|
s = h.query(req("q", "id:(1 2 3)", "qt", "/export", "fl", "intdv", "sort", "floatdv asc,stringdv asc,intdv desc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":3, \"docs\":[{\"intdv\":3},{\"intdv\":2},{\"intdv\":1}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":3, \"docs\":[{\"intdv\":3},{\"intdv\":2},{\"intdv\":1}]}}");
|
||||||
|
|
||||||
//Test three sort fields
|
//Test three sort fields
|
||||||
s = h.query(req("q", "id:(1 2 3)", "qt", "/export", "fl", "intdv", "sort", "floatdv asc,stringdv desc,intdv asc"));
|
s = h.query(req("q", "id:(1 2 3)", "qt", "/export", "fl", "intdv", "sort", "floatdv asc,stringdv desc,intdv asc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":3, \"docs\":[{\"intdv\":1},{\"intdv\":2},{\"intdv\":3}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":3, \"docs\":[{\"intdv\":1},{\"intdv\":2},{\"intdv\":3}]}}");
|
||||||
|
|
||||||
//Test four sort fields
|
//Test four sort fields
|
||||||
s = h.query(req("q", "id:(1 2 3)", "qt", "/export", "fl", "intdv", "sort", "floatdv asc,floatdv desc,floatdv asc,intdv desc"));
|
s = h.query(req("q", "id:(1 2 3)", "qt", "/export", "fl", "intdv", "sort", "floatdv asc,floatdv desc,floatdv asc,intdv desc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":3, \"docs\":[{\"intdv\":3},{\"intdv\":2},{\"intdv\":1}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":3, \"docs\":[{\"intdv\":3},{\"intdv\":2},{\"intdv\":1}]}}");
|
||||||
|
|
||||||
s = h.query(req("q", "id:(1 2 3)", "qt", "/export", "fl", "intdv", "sort", "doubledv desc"));
|
s = h.query(req("q", "id:(1 2 3)", "qt", "/export", "fl", "intdv", "sort", "doubledv desc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":3, \"docs\":[{\"intdv\":3},{\"intdv\":1},{\"intdv\":2}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":3, \"docs\":[{\"intdv\":3},{\"intdv\":1},{\"intdv\":2}]}}");
|
||||||
|
|
||||||
s = h.query(req("q", "intdv:[2 TO 1000]", "qt", "/export", "fl", "intdv", "sort", "doubledv desc"));
|
s = h.query(req("q", "intdv:[2 TO 1000]", "qt", "/export", "fl", "intdv", "sort", "doubledv desc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":3, \"docs\":[{\"intdv\":3},{\"intdv\":7},{\"intdv\":2}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":3, \"docs\":[{\"intdv\":3},{\"intdv\":7},{\"intdv\":2}]}}");
|
||||||
|
|
||||||
s = h.query(req("q", "stringdv:blah", "qt", "/export", "fl", "intdv", "sort", "doubledv desc"));
|
s = h.query(req("q", "stringdv:blah", "qt", "/export", "fl", "intdv", "sort", "doubledv desc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":0, \"docs\":[]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":0, \"docs\":[]}}");
|
||||||
|
|
||||||
s = h.query(req("q", "id:8", "qt", "/export", "fl", "stringdv", "sort", "intdv asc"));
|
s = h.query(req("q", "id:8", "qt", "/export", "fl", "stringdv", "sort", "intdv asc"));
|
||||||
assertEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":1, \"docs\":[{\"stringdv\":\"chello \\\"world\\\"\"}]}}");
|
assertJsonEquals(s, "{\"responseHeader\": {\"status\": 0}, \"response\":{\"numFound\":1, \"docs\":[{\"stringdv\":\"chello \\\"world\\\"\"}]}}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertJsonEquals(String actual, String expected) {
|
||||||
|
assertEquals(Utils.toJSONString(Utils.fromJSONString(expected)), Utils.toJSONString(Utils.fromJSONString(actual)));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
|
@ -0,0 +1,76 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF 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.apache.solr.response;
|
||||||
|
|
||||||
|
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.OutputStreamWriter;
|
||||||
|
import java.lang.invoke.MethodHandles;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.apache.solr.SolrTestCaseJ4;
|
||||||
|
import org.apache.solr.common.IteratorWriter;
|
||||||
|
import org.apache.solr.common.MapWriter;
|
||||||
|
import org.apache.solr.common.PushWriter;
|
||||||
|
import org.apache.solr.common.params.ModifiableSolrParams;
|
||||||
|
import org.apache.solr.common.util.Utils;
|
||||||
|
import org.apache.solr.request.LocalSolrQueryRequest;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import static java.nio.charset.StandardCharsets.UTF_8;
|
||||||
|
import static java.util.Collections.singletonMap;
|
||||||
|
|
||||||
|
public class TestPushWriter extends SolrTestCaseJ4 {
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
|
||||||
|
|
||||||
|
|
||||||
|
public void testStandardResponse() throws IOException {
|
||||||
|
ByteArrayOutputStream baos = new ByteArrayOutputStream();
|
||||||
|
OutputStreamWriter osw = new OutputStreamWriter(baos, UTF_8);
|
||||||
|
PushWriter pw = new JSONWriter(osw,
|
||||||
|
new LocalSolrQueryRequest(null, new ModifiableSolrParams()), new SolrQueryResponse());
|
||||||
|
writeData(pw);
|
||||||
|
osw.flush();
|
||||||
|
log.info(new String(baos.toByteArray(), "UTF-8"));
|
||||||
|
Object m = Utils.fromJSON(baos.toByteArray());
|
||||||
|
checkValues((Map) m);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void checkValues(Map m) {
|
||||||
|
assertEquals(0, ((Number)Utils.getObjectByPath(m, true, "responseHeader/status")).intValue());
|
||||||
|
assertEquals(10, ((Number)Utils.getObjectByPath(m, true, "response/numFound")).intValue());
|
||||||
|
assertEquals(1, ((Number)Utils.getObjectByPath(m, true, "response/docs[0]/id")).intValue());
|
||||||
|
assertEquals(2, ((Number)Utils.getObjectByPath(m, true, "response/docs[1]/id")).intValue());
|
||||||
|
assertEquals(3, ((Number)Utils.getObjectByPath(m, true, "response/docs[2]/id")).intValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void writeData(PushWriter pw) throws IOException {
|
||||||
|
pw.writeMap(m -> {
|
||||||
|
m.put("responseHeader", singletonMap("status", 0))
|
||||||
|
.put("response", (MapWriter) m1 -> {
|
||||||
|
m1.put("numFound", 10)
|
||||||
|
.put("docs", (IteratorWriter) w -> {
|
||||||
|
w.add((MapWriter) m3 -> m3.put("id", 1))
|
||||||
|
.add(singletonMap("id", 2))
|
||||||
|
.add(singletonMap("id", 3));
|
||||||
|
}); }); });
|
||||||
|
pw.close();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,63 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF 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.apache.solr.common;
|
||||||
|
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface to help do push writing to an array
|
||||||
|
*/
|
||||||
|
public interface IteratorWriter {
|
||||||
|
/**
|
||||||
|
* @param w after this method returns , the EntryWriter Object is invalid
|
||||||
|
* Do not hold a reference to this object
|
||||||
|
*/
|
||||||
|
void writeIter(ItemWriter w) throws IOException;
|
||||||
|
|
||||||
|
interface ItemWriter {
|
||||||
|
ItemWriter add(Object o) throws IOException;
|
||||||
|
|
||||||
|
default ItemWriter add(int v) throws IOException {
|
||||||
|
add((Integer) v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
default ItemWriter add(long v) throws IOException {
|
||||||
|
add((Long) v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
default ItemWriter add(float v) throws IOException {
|
||||||
|
add((Float) v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
default ItemWriter add(double v) throws IOException {
|
||||||
|
add((Double) v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
default ItemWriter add(boolean v) throws IOException {
|
||||||
|
add((Boolean) v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,81 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF 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.apache.solr.common;
|
||||||
|
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Use this class if the Map size is not known
|
||||||
|
*/
|
||||||
|
public interface MapWriter extends MapSerializable {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
default Map toMap(Map<String, Object> map) {
|
||||||
|
try {
|
||||||
|
writeMap(new EntryWriter() {
|
||||||
|
@Override
|
||||||
|
public EntryWriter put(String k, Object v) throws IOException {
|
||||||
|
map.put(k, v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
void writeMap(EntryWriter ew) throws IOException;
|
||||||
|
|
||||||
|
interface EntryWriter {
|
||||||
|
/**Writes a key value into the map
|
||||||
|
* @param k The key
|
||||||
|
* @param v The value can be any supported object
|
||||||
|
*/
|
||||||
|
EntryWriter put(String k, Object v) throws IOException;
|
||||||
|
|
||||||
|
default EntryWriter put(String k, int v) throws IOException {
|
||||||
|
put(k, (Integer) v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
default EntryWriter put(String k, long v) throws IOException {
|
||||||
|
put(k, (Long) v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
default EntryWriter put(String k, float v) throws IOException {
|
||||||
|
put(k, (Float) v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
default EntryWriter put(String k, double v) throws IOException{
|
||||||
|
put(k, (Double) v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
default EntryWriter put(String k, boolean v) throws IOException{
|
||||||
|
put(k, (Boolean) v);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,42 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF 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.apache.solr.common;
|
||||||
|
|
||||||
|
|
||||||
|
import java.io.Closeable;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**This is an interface to stream data out using a push API
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public interface PushWriter extends Closeable {
|
||||||
|
|
||||||
|
/**Write a Map. The map is opened in the beginning of the method
|
||||||
|
* and closed at the end. All map entries MUST be written before this
|
||||||
|
* method returns
|
||||||
|
*/
|
||||||
|
void writeMap(MapWriter mw) throws IOException;
|
||||||
|
|
||||||
|
/**Write an array. The array is opened at the beginning of this method
|
||||||
|
* and closed at the end. All array entries must be returned before this
|
||||||
|
* method returns
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
void writeIterator(IteratorWriter iw) throws IOException;
|
||||||
|
|
||||||
|
}
|
|
@ -16,6 +16,7 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.util;
|
package org.apache.solr.util;
|
||||||
|
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.StringWriter;
|
import java.io.StringWriter;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
|
@ -30,12 +31,22 @@ import org.apache.solr.common.SolrException;
|
||||||
import org.apache.solr.common.params.CommonParams;
|
import org.apache.solr.common.params.CommonParams;
|
||||||
import org.apache.solr.common.util.NamedList;
|
import org.apache.solr.common.util.NamedList;
|
||||||
import org.apache.solr.common.util.NamedList.NamedListEntry;
|
import org.apache.solr.common.util.NamedList.NamedListEntry;
|
||||||
import org.apache.solr.core.*;
|
import org.apache.solr.core.CloudConfig;
|
||||||
|
import org.apache.solr.core.CoreContainer;
|
||||||
|
import org.apache.solr.core.CoreDescriptor;
|
||||||
|
import org.apache.solr.core.CorePropertiesLocator;
|
||||||
|
import org.apache.solr.core.CoresLocator;
|
||||||
|
import org.apache.solr.core.NodeConfig;
|
||||||
|
import org.apache.solr.core.SolrConfig;
|
||||||
|
import org.apache.solr.core.SolrCore;
|
||||||
|
import org.apache.solr.core.SolrResourceLoader;
|
||||||
|
import org.apache.solr.core.SolrXmlConfig;
|
||||||
import org.apache.solr.handler.UpdateRequestHandler;
|
import org.apache.solr.handler.UpdateRequestHandler;
|
||||||
import org.apache.solr.request.LocalSolrQueryRequest;
|
import org.apache.solr.request.LocalSolrQueryRequest;
|
||||||
import org.apache.solr.request.SolrQueryRequest;
|
import org.apache.solr.request.SolrQueryRequest;
|
||||||
import org.apache.solr.request.SolrRequestHandler;
|
import org.apache.solr.request.SolrRequestHandler;
|
||||||
import org.apache.solr.request.SolrRequestInfo;
|
import org.apache.solr.request.SolrRequestInfo;
|
||||||
|
import org.apache.solr.response.BinaryQueryResponseWriter;
|
||||||
import org.apache.solr.response.QueryResponseWriter;
|
import org.apache.solr.response.QueryResponseWriter;
|
||||||
import org.apache.solr.response.SolrQueryResponse;
|
import org.apache.solr.response.SolrQueryResponse;
|
||||||
import org.apache.solr.schema.IndexSchema;
|
import org.apache.solr.schema.IndexSchema;
|
||||||
|
@ -311,10 +322,18 @@ public class TestHarness extends BaseTestHarness {
|
||||||
if (rsp.getException() != null) {
|
if (rsp.getException() != null) {
|
||||||
throw rsp.getException();
|
throw rsp.getException();
|
||||||
}
|
}
|
||||||
StringWriter sw = new StringWriter(32000);
|
|
||||||
QueryResponseWriter responseWriter = core.getQueryResponseWriter(req);
|
QueryResponseWriter responseWriter = core.getQueryResponseWriter(req);
|
||||||
responseWriter.write(sw,req,rsp);
|
if (responseWriter instanceof BinaryQueryResponseWriter) {
|
||||||
return sw.toString();
|
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream(32000);
|
||||||
|
BinaryQueryResponseWriter writer = (BinaryQueryResponseWriter) responseWriter;
|
||||||
|
writer.write(byteArrayOutputStream, req, rsp);
|
||||||
|
return new String(byteArrayOutputStream.toByteArray(), "UTF-8");
|
||||||
|
} else {
|
||||||
|
StringWriter sw = new StringWriter(32000);
|
||||||
|
responseWriter.write(sw,req,rsp);
|
||||||
|
return sw.toString();
|
||||||
|
}
|
||||||
|
|
||||||
} finally {
|
} finally {
|
||||||
req.close();
|
req.close();
|
||||||
SolrRequestInfo.clearRequestInfo();
|
SolrRequestInfo.clearRequestInfo();
|
||||||
|
|
Loading…
Reference in New Issue