Skip to content

Commit 9481840

Browse files
committed
CAPI-571 Fix SolrJ Bind Nested Child DTO Fields
The current solrj Document DTO to SolrInputDocument binder treat all child docs as anonymous rather than prefix them with the given path. This will result in the document being indexed without the `_nest_path_` field populated which is required for the ChildDocumentTransformer (`fl=[child]`) to work. Add a new property to `@Field` interface to determine if a `Field` annoted with `child=true` should be treated as anonymous field or if it should be treated as a nested child document. Where they should be nested child docs, each nested object would be converted to its own `SolrInputDocument` and added as a regular field to the parent `SolrInputDocument`. Having it as a regular field with nested objects will result in Solr indexing these child docs correctly, and it will populate appropriate nested child docs fields during index time. We can now have more that one child documents fields listed in the DTO until they are set and `anonymizeChild=false`.
1 parent d06ab17 commit 9481840

File tree

3 files changed

+171
-6
lines changed

3 files changed

+171
-6
lines changed

solr/solrj/src/java/org/apache/solr/client/solrj/beans/DocumentObjectBinder.java

+45-5
Original file line numberDiff line numberDiff line change
@@ -91,8 +91,10 @@ public SolrInputDocument toSolrInputDocument(Object obj) {
9191
doc.setField(e.getKey(), e.getValue());
9292
}
9393
} else {
94-
if (field.child != null) {
95-
addChild(obj, field, doc);
94+
if (field.child != null && field.annotation.anonymizeChild()) {
95+
addAnonymousChild(obj, field, doc);
96+
} else if (field.child != null) {
97+
addNestedChild(obj, field, doc);
9698
} else {
9799
doc.setField(field.name, field.get(obj));
98100
}
@@ -101,7 +103,7 @@ public SolrInputDocument toSolrInputDocument(Object obj) {
101103
return doc;
102104
}
103105

104-
private void addChild(Object obj, DocField field, SolrInputDocument doc) {
106+
private void addAnonymousChild(Object obj, DocField field, SolrInputDocument doc) {
105107
Object val = field.get(obj);
106108
if (val == null) return;
107109
if (val instanceof Collection) {
@@ -119,6 +121,31 @@ private void addChild(Object obj, DocField field, SolrInputDocument doc) {
119121
}
120122
}
121123

124+
private void addNestedChild(Object obj, DocField field, SolrInputDocument doc) {
125+
Object val = field.get(obj);
126+
if (val == null) return;
127+
if (val instanceof Collection) {
128+
@SuppressWarnings({"rawtypes"})
129+
Collection collection = (Collection) val;
130+
List<SolrInputDocument> docs = new ArrayList<>(collection.size());
131+
for (final Object o : collection) {
132+
final SolrInputDocument inpDoc = toSolrInputDocument(o);
133+
docs.add(inpDoc);
134+
}
135+
val = docs;
136+
} else if (val.getClass().isArray()) {
137+
Object[] objs = (Object[]) val;
138+
SolrInputDocument[] docs = (SolrInputDocument[]) Array.newInstance(SolrInputDocument.class, objs.length);
139+
for (int i = 0; i < objs.length; i++) {
140+
docs[i] = toSolrInputDocument(objs[i]);
141+
}
142+
val = docs;
143+
} else {
144+
val = toSolrInputDocument(val);
145+
}
146+
doc.addField(field.name, val);
147+
}
148+
122149
private List<DocField> getDocFields(@SuppressWarnings({"rawtypes"})Class clazz) {
123150
List<DocField> fields = infocache.get(clazz);
124151
if (fields == null) {
@@ -146,7 +173,7 @@ private List<DocField> collectInfo(@SuppressWarnings({"rawtypes"})Class clazz) {
146173
if (member.isAnnotationPresent(Field.class)) {
147174
AccessController.doPrivileged((PrivilegedAction<Void>) () -> { member.setAccessible(true); return null; });
148175
DocField df = new DocField(member);
149-
if (df.child != null) {
176+
if (df.child != null && df.annotation.anonymizeChild()) {
150177
if (childFieldFound)
151178
throw new BindingException(clazz.getName() + " cannot have more than one Field with child=true");
152179
childFieldFound = true;
@@ -334,7 +361,20 @@ private void populateChild(Type typ) {
334361
@SuppressWarnings({"unchecked", "rawtypes"})
335362
private Object getFieldValue(SolrDocument solrDocument) {
336363
if (child != null) {
337-
List<SolrDocument> children = solrDocument.getChildDocuments();
364+
List<SolrDocument> children = null;
365+
if(solrDocument.hasChildDocuments()){
366+
children = solrDocument.getChildDocuments();
367+
} else if (!annotation.anonymizeChild()){
368+
final Object val = solrDocument.getFieldValue(name);
369+
if(val == null){
370+
return null;
371+
} else if (isList || isArray) {
372+
children = (List<SolrDocument>) val;
373+
} else {
374+
children = new ArrayList<>();
375+
children.add((SolrDocument) val);
376+
}
377+
}
338378
if (children == null || children.isEmpty()) return null;
339379
if (isList) {
340380
ArrayList list = new ArrayList(children.size());

solr/solrj/src/java/org/apache/solr/client/solrj/beans/Field.java

+11
Original file line numberDiff line numberDiff line change
@@ -34,5 +34,16 @@
3434
@Retention(RUNTIME)
3535
public @interface Field {
3636
boolean child() default false;
37+
38+
/**
39+
* When converting fields with the `@Field(child=true)` annotation, this will
40+
* determine if they should be converted as an anonymized nested child document
41+
* or as a regular (mapped) nested child document.
42+
*
43+
* If intending to use `AtomicUpdates` or `ChildDocumentTransformer`, this
44+
* should be set to `false`.
45+
*/
46+
boolean anonymizeChild() default true;
47+
3748
String value() default DEFAULT;
3849
}

solr/solrj/src/test/org/apache/solr/client/solrj/beans/TestDocumentObjectBinder.java

+115-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,9 @@
2727
import org.junit.Test;
2828

2929
import java.io.StringReader;
30+
import java.util.ArrayList;
3031
import java.util.Arrays;
32+
import java.util.Collection;
3133
import java.util.Date;
3234
import java.util.List;
3335
import java.util.Map;
@@ -151,10 +153,97 @@ public void testChild() throws Exception {
151153
assertEquals(arrIn.child[1].name, arrOut.child[1].name);
152154

153155
}
156+
public void testMappedChild() throws Exception {
157+
DocumentObjectBinder binder = new DocumentObjectBinder();
158+
159+
160+
SingleValueNestedChild nestedNullChildIn = new SingleValueNestedChild();
161+
nestedNullChildIn.id = "1-null-child";
162+
nestedNullChildIn.child = null;
163+
SolrInputDocument nestedNullSolrInputDoc = binder.toSolrInputDocument(nestedNullChildIn);
164+
SolrDocument nestedNullSolrDoc = toSolrDocument(nestedNullSolrInputDoc);
165+
assertNull(nestedNullSolrInputDoc.getChildDocuments());
166+
assertNull(nestedNullSolrDoc.getChildDocuments());
167+
SingleValueNestedChild nestedNullChildOut = binder.getBean(SingleValueNestedChild.class, nestedNullSolrDoc);
168+
assertEquals(nestedNullChildIn.id, nestedNullChildOut.id);
169+
assertNull(nestedNullChildIn.child);
170+
assertNull(nestedNullChildOut.child);
171+
172+
SingleValueNestedChild singleNestedIn = new SingleValueNestedChild();
173+
singleNestedIn.id = "1";
174+
singleNestedIn.child = new Child();
175+
singleNestedIn.child.id = "1.1";
176+
singleNestedIn.child.name = "Name One";
177+
SolrInputDocument singleNestedSolrInputDoc = binder.toSolrInputDocument(singleNestedIn);
178+
SolrDocument singleNestedSolrDoc = toSolrDocument(singleNestedSolrInputDoc);
179+
assertNull(singleNestedSolrInputDoc.getChildDocuments());
180+
assertNull(singleNestedSolrDoc.getChildDocuments());
181+
SingleValueNestedChild singleNestedOut = binder.getBean(SingleValueNestedChild.class, singleNestedSolrDoc);
182+
assertEquals(singleNestedIn.id, singleNestedOut.id);
183+
assertEquals(singleNestedIn.child.id, singleNestedOut.child.id);
184+
assertEquals(singleNestedIn.child.name, singleNestedOut.child.name);
185+
186+
ListNestedChild listNestedIn = new ListNestedChild();
187+
listNestedIn.id = "2";
188+
Child child = new Child();
189+
child.id = "1.2";
190+
child.name = "Name Two";
191+
listNestedIn.child = Arrays.asList(singleNestedIn.child, child);
192+
SolrInputDocument listNestedSolrInputDoc = binder.toSolrInputDocument(listNestedIn);
193+
SolrDocument listNestedSolrDoc = toSolrDocument(listNestedSolrInputDoc);
194+
assertNull(listNestedSolrInputDoc.getChildDocuments());
195+
assertNull(listNestedSolrDoc.getChildDocuments());
196+
ListNestedChild listNestedOut = binder.getBean(ListNestedChild.class, listNestedSolrDoc);
197+
assertEquals(listNestedIn.id, listNestedOut.id);
198+
assertEquals(listNestedIn.child.get(0).id, listNestedOut.child.get(0).id);
199+
assertEquals(listNestedIn.child.get(0).name, listNestedOut.child.get(0).name);
200+
assertEquals(listNestedIn.child.get(1).id, listNestedOut.child.get(1).id);
201+
assertEquals(listNestedIn.child.get(1).name, listNestedOut.child.get(1).name);
202+
203+
ArrayNestedChild arrayNestedIn = new ArrayNestedChild();
204+
arrayNestedIn.id = "3";
205+
arrayNestedIn.child = new Child[]{singleNestedIn.child, child};
206+
SolrInputDocument arrayNestedSolrInputDoc = binder.toSolrInputDocument(arrayNestedIn);
207+
SolrDocument arrayNestedSolrDoc = toSolrDocument(arrayNestedSolrInputDoc);
208+
assertNull(arrayNestedSolrInputDoc.getChildDocuments());
209+
assertNull(arrayNestedSolrDoc.getChildDocuments());
210+
ArrayNestedChild arrayNestedOut = binder.getBean(ArrayNestedChild.class, arrayNestedSolrDoc);
211+
assertEquals(arrayNestedIn.id, arrayNestedOut.id);
212+
assertEquals(arrayNestedIn.child[0].id, arrayNestedOut.child[0].id);
213+
assertEquals(arrayNestedIn.child[0].name, arrayNestedOut.child[0].name);
214+
assertEquals(arrayNestedIn.child[1].id, arrayNestedOut.child[1].id);
215+
assertEquals(arrayNestedIn.child[1].name, arrayNestedOut.child[1].name);
216+
}
154217

155218
private static SolrDocument toSolrDocument(SolrInputDocument d) {
156219
SolrDocument doc = new SolrDocument();
157220
for (SolrInputField field : d) {
221+
if (field.getValue() != null) {
222+
if (field.getValue() instanceof Collection) {
223+
Collection<?> values = (Collection<?>) field.getValue();
224+
if (!values.isEmpty() && values.iterator().next() instanceof SolrInputDocument) {
225+
List<SolrDocument> docs = new ArrayList<>(values.size());
226+
for (Object value : values) {
227+
docs.add(toSolrDocument((SolrInputDocument) value));
228+
}
229+
doc.setField(field.getName(), docs);
230+
continue;
231+
}
232+
} else if (field.getValue().getClass().isArray()) {
233+
Object[] values = (Object[]) field.getValue();
234+
if (values.length > 0 && values[0] instanceof SolrInputDocument) {
235+
SolrDocument[] docs = new SolrDocument[values.length];
236+
for (int i = 0; i < values.length; i++) {
237+
docs[i] = toSolrDocument((SolrInputDocument) values[i]);
238+
}
239+
doc.setField(field.getName(), docs);
240+
continue;
241+
}
242+
} else if (field.getValue() instanceof SolrInputDocument) {
243+
doc.setField(field.getName(), toSolrDocument((SolrInputDocument) field.getValue()));
244+
continue;
245+
}
246+
}
158247
doc.setField(field.getName(), field.getValue());
159248
}
160249
if (d.getChildDocuments() != null) {
@@ -245,7 +334,32 @@ public static class ArrayChild {
245334
@Field(child = true)
246335
Child[] child;
247336
}
248-
337+
338+
public static class SingleValueNestedChild {
339+
@Field
340+
String id;
341+
342+
@Field(child = true, anonymizeChild = false)
343+
Child child;
344+
}
345+
346+
public static class ListNestedChild {
347+
348+
@Field
349+
String id;
350+
351+
@Field(child = true, anonymizeChild = false)
352+
List<Child> child;
353+
}
354+
355+
public static class ArrayNestedChild {
356+
357+
@Field
358+
String id;
359+
360+
@Field(child = true, anonymizeChild = false)
361+
Child[] child;
362+
}
249363

250364
public static class NotGettableItem {
251365
@Field

0 commit comments

Comments
 (0)