Merge "BUG-467: reconnect-strategy configuration moved into controller/commons/protoc...
[controller.git] / opendaylight / md-sal / sal-common-impl / src / main / java / org / opendaylight / controller / md / sal / common / impl / util / compat / DataNormalizationOperation.java
1 package org.opendaylight.controller.md.sal.common.impl.util.compat;
2
3 import static com.google.common.base.Preconditions.checkArgument;
4 import static com.google.common.base.Preconditions.checkNotNull;
5
6 import java.util.Collections;
7 import java.util.HashSet;
8 import java.util.List;
9 import java.util.Map;
10 import java.util.Map.Entry;
11 import java.util.Set;
12 import java.util.concurrent.ConcurrentHashMap;
13
14 import org.opendaylight.yangtools.concepts.Identifiable;
15 import org.opendaylight.yangtools.yang.common.QName;
16 import org.opendaylight.yangtools.yang.data.api.CompositeNode;
17 import org.opendaylight.yangtools.yang.data.api.InstanceIdentifier.AugmentationIdentifier;
18 import org.opendaylight.yangtools.yang.data.api.InstanceIdentifier.NodeIdentifier;
19 import org.opendaylight.yangtools.yang.data.api.InstanceIdentifier.NodeIdentifierWithPredicates;
20 import org.opendaylight.yangtools.yang.data.api.InstanceIdentifier.NodeWithValue;
21 import org.opendaylight.yangtools.yang.data.api.InstanceIdentifier.PathArgument;
22 import org.opendaylight.yangtools.yang.data.api.Node;
23 import org.opendaylight.yangtools.yang.data.api.SimpleNode;
24 import org.opendaylight.yangtools.yang.data.api.schema.MapEntryNode;
25 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
26 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNodeContainer;
27 import org.opendaylight.yangtools.yang.data.impl.schema.Builders;
28 import org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNodes;
29 import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.DataContainerNodeAttrBuilder;
30 import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.NormalizedNodeContainerBuilder;
31 import org.opendaylight.yangtools.yang.model.api.AugmentationSchema;
32 import org.opendaylight.yangtools.yang.model.api.AugmentationTarget;
33 import org.opendaylight.yangtools.yang.model.api.ChoiceCaseNode;
34 import org.opendaylight.yangtools.yang.model.api.ContainerSchemaNode;
35 import org.opendaylight.yangtools.yang.model.api.DataNodeContainer;
36 import org.opendaylight.yangtools.yang.model.api.DataSchemaNode;
37 import org.opendaylight.yangtools.yang.model.api.LeafListSchemaNode;
38 import org.opendaylight.yangtools.yang.model.api.LeafSchemaNode;
39 import org.opendaylight.yangtools.yang.model.api.ListSchemaNode;
40 import org.opendaylight.yangtools.yang.model.api.SchemaContext;
41
42 import com.google.common.collect.FluentIterable;
43 import com.google.common.collect.ImmutableMap;
44 import com.google.common.collect.ImmutableSet;
45
46 public abstract class DataNormalizationOperation<T extends PathArgument> implements Identifiable<T> {
47
48     private final T identifier;
49
50     @Override
51     public T getIdentifier() {
52         return identifier;
53     };
54
55     protected DataNormalizationOperation(final T identifier) {
56         super();
57         this.identifier = identifier;
58     }
59
60     public boolean isMixin() {
61         return false;
62     }
63
64     protected Set<QName> getQNameIdentifiers() {
65         return Collections.singleton(identifier.getNodeType());
66     }
67
68     public abstract DataNormalizationOperation<?> getChild(final PathArgument child);
69
70     public abstract DataNormalizationOperation<?> getChild(QName child);
71
72     public abstract NormalizedNode<?, ?> normalize(Node<?> legacyData);
73
74     private static abstract class SimpleTypeNormalization<T extends PathArgument> extends DataNormalizationOperation<T> {
75
76         protected SimpleTypeNormalization(final T identifier) {
77             super(identifier);
78         }
79
80         @Override
81         public NormalizedNode<?, ?> normalize(final Node<?> legacyData) {
82             checkArgument(legacyData != null);
83             checkArgument(legacyData instanceof SimpleNode<?>);
84             return normalizeImpl((SimpleNode<?>) legacyData);
85         }
86
87         protected abstract NormalizedNode<?, ?> normalizeImpl(SimpleNode<?> node);
88
89         @Override
90         public DataNormalizationOperation<?> getChild(final PathArgument child) {
91             return null;
92         }
93
94         @Override
95         public DataNormalizationOperation<?> getChild(final QName child) {
96             return null;
97         }
98
99         @Override
100         public NormalizedNode<?, ?> createDefault(final PathArgument currentArg) {
101             // TODO Auto-generated method stub
102             return null;
103         }
104
105     }
106
107     private static final class LeafNormalization extends SimpleTypeNormalization<NodeIdentifier> {
108
109         protected LeafNormalization(final NodeIdentifier identifier) {
110             super(identifier);
111         }
112
113         @Override
114         protected NormalizedNode<?, ?> normalizeImpl(final SimpleNode<?> node) {
115             return ImmutableNodes.leafNode(node.getNodeType(), node.getValue());
116         }
117
118     }
119
120     private static final class LeafListEntryNormalization extends SimpleTypeNormalization<NodeWithValue> {
121
122         public LeafListEntryNormalization(final LeafListSchemaNode potential) {
123             super(new NodeWithValue(potential.getQName(), null));
124         }
125
126         @Override
127         protected NormalizedNode<?, ?> normalizeImpl(final SimpleNode<?> node) {
128             NodeWithValue nodeId = new NodeWithValue(node.getNodeType(), node.getValue());
129             return Builders.leafSetEntryBuilder().withNodeIdentifier(nodeId).withValue(node.getValue()).build();
130         }
131
132     }
133
134     private static abstract class CompositeNodeNormalizationOpertation<T extends PathArgument> extends
135             DataNormalizationOperation<T> {
136
137         protected CompositeNodeNormalizationOpertation(final T identifier) {
138             super(identifier);
139         }
140
141         @SuppressWarnings({ "rawtypes", "unchecked" })
142         @Override
143         public final NormalizedNodeContainer<?, ?, ?> normalize(final Node<?> legacyData) {
144             checkArgument(legacyData != null);
145             if (!isMixin() && getIdentifier().getNodeType() != null) {
146                 checkArgument(getIdentifier().getNodeType().equals(legacyData.getNodeType()),
147                         "Node QName must be %s was %s", getIdentifier().getNodeType(), legacyData.getNodeType());
148             }
149             checkArgument(legacyData instanceof CompositeNode, "Node %s should be composite", legacyData);
150             CompositeNode compositeNode = (CompositeNode) legacyData;
151             NormalizedNodeContainerBuilder builder = createBuilder(compositeNode);
152
153             Set<DataNormalizationOperation<?>> usedMixins = new HashSet<>();
154             for (Node<?> childLegacy : compositeNode.getValue()) {
155                 DataNormalizationOperation childOp = getChild(childLegacy.getNodeType());
156
157                 // We skip unknown nodes if this node is mixin since
158                 // it's nodes and parent nodes are interleaved
159                 if (childOp == null && isMixin()) {
160                     continue;
161                 }
162
163                 checkArgument(childOp != null, "Node %s is not allowed inside %s", childLegacy.getNodeType(),
164                         getIdentifier());
165                 if (childOp.isMixin()) {
166                     if (usedMixins.contains(childOp)) {
167                         // We already run / processed that mixin, so to avoid
168                         // dupliciry we are
169                         // skiping next nodes.
170                         continue;
171                     }
172                     builder.addChild(childOp.normalize(compositeNode));
173                     usedMixins.add(childOp);
174                 } else {
175                     builder.addChild(childOp.normalize(childLegacy));
176                 }
177             }
178             return (NormalizedNodeContainer<?, ?, ?>) builder.build();
179         }
180
181         @SuppressWarnings("rawtypes")
182         protected abstract NormalizedNodeContainerBuilder createBuilder(final CompositeNode compositeNode);
183
184     }
185
186     private static abstract class DataContainerNormalizationOperation<T extends PathArgument> extends
187             CompositeNodeNormalizationOpertation<T> {
188
189         private final DataNodeContainer schema;
190         private final Map<QName, DataNormalizationOperation<?>> byQName;
191         private final Map<PathArgument, DataNormalizationOperation<?>> byArg;
192
193         protected DataContainerNormalizationOperation(final T identifier, final DataNodeContainer schema) {
194             super(identifier);
195             this.schema = schema;
196             this.byArg = new ConcurrentHashMap<>();
197             this.byQName = new ConcurrentHashMap<>();
198         }
199
200         @Override
201         public DataNormalizationOperation<?> getChild(final PathArgument child) {
202             DataNormalizationOperation<?> potential = byArg.get(child);
203             if (potential != null) {
204                 return potential;
205             }
206             potential = fromSchema(schema, child);
207             return register(potential);
208         }
209
210         @Override
211         public DataNormalizationOperation<?> getChild(final QName child) {
212             DataNormalizationOperation<?> potential = byQName.get(child);
213             if (potential != null) {
214                 return potential;
215             }
216             potential = fromSchemaAndPathArgument(schema, child);
217             return register(potential);
218         }
219
220         private DataNormalizationOperation<?> register(final DataNormalizationOperation<?> potential) {
221             if (potential != null) {
222                 byArg.put(potential.getIdentifier(), potential);
223                 for (QName qName : potential.getQNameIdentifiers()) {
224                     byQName.put(qName, potential);
225                 }
226             }
227             return potential;
228         }
229
230     }
231
232     private static final class ListItemNormalization extends
233             DataContainerNormalizationOperation<NodeIdentifierWithPredicates> {
234
235         private final List<QName> keyDefinition;
236
237         protected ListItemNormalization(final NodeIdentifierWithPredicates identifier, final ListSchemaNode schema) {
238             super(identifier, schema);
239             keyDefinition = schema.getKeyDefinition();
240         }
241
242         @Override
243         protected NormalizedNodeContainerBuilder createBuilder(final CompositeNode compositeNode) {
244             ImmutableMap.Builder<QName, Object> keys = ImmutableMap.builder();
245             for (QName key : keyDefinition) {
246                 SimpleNode<?> valueNode = checkNotNull(compositeNode.getFirstSimpleByName(key),"List node %s MUST contain leaf %s with value.",getIdentifier().getNodeType(),key);
247                 keys.put(key, valueNode.getValue());
248             }
249
250             return Builders.mapEntryBuilder().withNodeIdentifier(
251                     new NodeIdentifierWithPredicates(getIdentifier().getNodeType(), keys.build()));
252         }
253
254         @Override
255         public NormalizedNode<?, ?> createDefault(final PathArgument currentArg) {
256             DataContainerNodeAttrBuilder<NodeIdentifierWithPredicates, MapEntryNode> builder = Builders
257                     .mapEntryBuilder().withNodeIdentifier((NodeIdentifierWithPredicates) currentArg);
258             for (Entry<QName, Object> keyValue : ((NodeIdentifierWithPredicates) currentArg).getKeyValues().entrySet()) {
259                 builder.addChild(Builders.leafBuilder()
260                         //
261                         .withNodeIdentifier(new NodeIdentifier(keyValue.getKey())).withValue(keyValue.getValue())
262                         .build());
263             }
264             return builder.build();
265         }
266     }
267
268     private static final class ContainerNormalization extends DataContainerNormalizationOperation<NodeIdentifier> {
269
270         protected ContainerNormalization(final ContainerSchemaNode schema) {
271             super(new NodeIdentifier(schema.getQName()), schema);
272         }
273
274         @Override
275         protected NormalizedNodeContainerBuilder createBuilder(final CompositeNode compositeNode) {
276             return Builders.containerBuilder().withNodeIdentifier(getIdentifier());
277         }
278
279         @Override
280         public NormalizedNode<?, ?> createDefault(final PathArgument currentArg) {
281             return Builders.containerBuilder().withNodeIdentifier((NodeIdentifier) currentArg).build();
282         }
283
284     }
285
286     private static abstract class MixinNormalizationOp<T extends PathArgument> extends
287             CompositeNodeNormalizationOpertation<T> {
288
289         protected MixinNormalizationOp(final T identifier) {
290             super(identifier);
291         }
292
293         @Override
294         public final boolean isMixin() {
295             return true;
296         }
297
298     }
299
300     private static final class LeafListMixinNormalization extends MixinNormalizationOp<NodeIdentifier> {
301
302         private final DataNormalizationOperation<?> innerOp;
303
304         public LeafListMixinNormalization(final LeafListSchemaNode potential) {
305             super(new NodeIdentifier(potential.getQName()));
306             innerOp = new LeafListEntryNormalization(potential);
307         }
308
309         @Override
310         protected NormalizedNodeContainerBuilder createBuilder(final CompositeNode compositeNode) {
311             return Builders.leafSetBuilder().withNodeIdentifier(getIdentifier());
312         }
313
314         @Override
315         public NormalizedNode<?, ?> createDefault(final PathArgument currentArg) {
316             return Builders.leafSetBuilder().withNodeIdentifier(getIdentifier()).build();
317         }
318
319         @Override
320         public DataNormalizationOperation<?> getChild(final PathArgument child) {
321             if (child instanceof NodeWithValue) {
322                 return innerOp;
323             }
324             return null;
325         }
326
327         @Override
328         public DataNormalizationOperation<?> getChild(final QName child) {
329             if (getIdentifier().getNodeType().equals(child)) {
330                 return innerOp;
331             }
332             return null;
333         }
334     }
335
336     private static final class AugmentationNormalization extends MixinNormalizationOp<AugmentationIdentifier> {
337
338         private final Map<QName, DataNormalizationOperation<?>> byQName;
339         private final Map<PathArgument, DataNormalizationOperation<?>> byArg;
340
341         public AugmentationNormalization(final AugmentationSchema augmentation, final DataNodeContainer schema) {
342             super(augmentationIdentifierFrom(augmentation));
343
344             ImmutableMap.Builder<QName, DataNormalizationOperation<?>> byQNameBuilder = ImmutableMap.builder();
345             ImmutableMap.Builder<PathArgument, DataNormalizationOperation<?>> byArgBuilder = ImmutableMap.builder();
346
347             for (DataSchemaNode augNode : augmentation.getChildNodes()) {
348                 DataSchemaNode resolvedNode = schema.getDataChildByName(augNode.getQName());
349                 DataNormalizationOperation<?> resolvedOp = fromDataSchemaNode(resolvedNode);
350                 byArgBuilder.put(resolvedOp.getIdentifier(), resolvedOp);
351                 for (QName resQName : resolvedOp.getQNameIdentifiers()) {
352                     byQNameBuilder.put(resQName, resolvedOp);
353                 }
354             }
355             byQName = byQNameBuilder.build();
356             byArg = byArgBuilder.build();
357
358         }
359
360         @Override
361         public DataNormalizationOperation<?> getChild(final PathArgument child) {
362             return byArg.get(child);
363         }
364
365         @Override
366         public DataNormalizationOperation<?> getChild(final QName child) {
367             return byQName.get(child);
368         }
369
370         @Override
371         protected Set<QName> getQNameIdentifiers() {
372             return getIdentifier().getPossibleChildNames();
373         }
374
375         @SuppressWarnings("rawtypes")
376         @Override
377         protected NormalizedNodeContainerBuilder createBuilder(final CompositeNode compositeNode) {
378             return Builders.augmentationBuilder().withNodeIdentifier(getIdentifier());
379         }
380
381         @Override
382         public NormalizedNode<?, ?> createDefault(final PathArgument currentArg) {
383             return Builders.augmentationBuilder().withNodeIdentifier(getIdentifier()).build();
384         }
385
386     }
387
388     private static final class ListMixinNormalization extends MixinNormalizationOp<NodeIdentifier> {
389
390         private final ListItemNormalization innerNode;
391
392         public ListMixinNormalization(final ListSchemaNode list) {
393             super(new NodeIdentifier(list.getQName()));
394             this.innerNode = new ListItemNormalization(new NodeIdentifierWithPredicates(list.getQName(),
395                     Collections.<QName, Object> emptyMap()), list);
396         }
397
398         @SuppressWarnings("rawtypes")
399         @Override
400         protected NormalizedNodeContainerBuilder createBuilder(final CompositeNode compositeNode) {
401             return Builders.mapBuilder().withNodeIdentifier(getIdentifier());
402         }
403
404         @Override
405         public NormalizedNode<?, ?> createDefault(final PathArgument currentArg) {
406             return Builders.mapBuilder().withNodeIdentifier(getIdentifier()).build();
407         }
408
409         @Override
410         public DataNormalizationOperation<?> getChild(final PathArgument child) {
411             if (child.getNodeType().equals(getIdentifier().getNodeType())) {
412                 return innerNode;
413             }
414             return null;
415         }
416
417         @Override
418         public DataNormalizationOperation<?> getChild(final QName child) {
419             if (getIdentifier().getNodeType().equals(child)) {
420                 return innerNode;
421             }
422             return null;
423         }
424
425     }
426
427     private static class ChoiceNodeNormalization extends MixinNormalizationOp<NodeIdentifier> {
428
429         private final ImmutableMap<QName, DataNormalizationOperation<?>> byQName;
430         private final ImmutableMap<PathArgument, DataNormalizationOperation<?>> byArg;
431
432         protected ChoiceNodeNormalization(final org.opendaylight.yangtools.yang.model.api.ChoiceNode schema) {
433             super(new NodeIdentifier(schema.getQName()));
434             ImmutableMap.Builder<QName, DataNormalizationOperation<?>> byQNameBuilder = ImmutableMap.builder();
435             ImmutableMap.Builder<PathArgument, DataNormalizationOperation<?>> byArgBuilder = ImmutableMap.builder();
436
437             for (ChoiceCaseNode caze : schema.getCases()) {
438                 for (DataSchemaNode cazeChild : caze.getChildNodes()) {
439                     DataNormalizationOperation<?> childOp = fromDataSchemaNode(cazeChild);
440                     byArgBuilder.put(childOp.getIdentifier(), childOp);
441                     for (QName qname : childOp.getQNameIdentifiers()) {
442                         byQNameBuilder.put(qname, childOp);
443                     }
444                 }
445             }
446             byQName = byQNameBuilder.build();
447             byArg = byArgBuilder.build();
448         }
449
450         @Override
451         public DataNormalizationOperation<?> getChild(final PathArgument child) {
452             return byArg.get(child);
453         }
454
455         @Override
456         public DataNormalizationOperation<?> getChild(final QName child) {
457             return byQName.get(child);
458         }
459
460         @Override
461         protected NormalizedNodeContainerBuilder createBuilder(final CompositeNode compositeNode) {
462             return Builders.choiceBuilder().withNodeIdentifier(getIdentifier());
463         }
464
465         @Override
466         public NormalizedNode<?, ?> createDefault(final PathArgument currentArg) {
467             return Builders.choiceBuilder().withNodeIdentifier(getIdentifier()).build();
468         }
469     }
470
471     public static DataNormalizationOperation<?> fromSchemaAndPathArgument(final DataNodeContainer schema,
472             final QName child) {
473         DataSchemaNode potential = schema.getDataChildByName(child);
474         if (potential == null) {
475             Iterable<org.opendaylight.yangtools.yang.model.api.ChoiceNode> choices = FluentIterable.from(
476                     schema.getChildNodes()).filter(org.opendaylight.yangtools.yang.model.api.ChoiceNode.class);
477             potential = findChoice(choices, child);
478         }
479         checkArgument(potential != null, "Supplied QName %s is not valid according to schema %s", child, schema);
480         if ((schema instanceof DataSchemaNode) && !((DataSchemaNode) schema).isAugmenting() && potential.isAugmenting()) {
481             return fromAugmentation(schema, (AugmentationTarget) schema, potential);
482         }
483         return fromDataSchemaNode(potential);
484     }
485
486     private static org.opendaylight.yangtools.yang.model.api.ChoiceNode findChoice(
487             final Iterable<org.opendaylight.yangtools.yang.model.api.ChoiceNode> choices, final QName child) {
488         org.opendaylight.yangtools.yang.model.api.ChoiceNode foundChoice = null;
489         choiceLoop: for (org.opendaylight.yangtools.yang.model.api.ChoiceNode choice : choices) {
490             for (ChoiceCaseNode caze : choice.getCases()) {
491                 if (caze.getDataChildByName(child) != null) {
492                     foundChoice = choice;
493                     break choiceLoop;
494                 }
495             }
496         }
497         return foundChoice;
498     }
499
500     public static AugmentationIdentifier augmentationIdentifierFrom(final AugmentationSchema augmentation) {
501         ImmutableSet.Builder<QName> potentialChildren = ImmutableSet.builder();
502         for (DataSchemaNode child : augmentation.getChildNodes()) {
503             potentialChildren.add(child.getQName());
504         }
505         return new AugmentationIdentifier(null, potentialChildren.build());
506     }
507
508     private static AugmentationNormalization fromAugmentation(final DataNodeContainer schema,
509             final AugmentationTarget augments, final DataSchemaNode potential) {
510         AugmentationSchema augmentation = null;
511         for (AugmentationSchema aug : augments.getAvailableAugmentations()) {
512             DataSchemaNode child = aug.getDataChildByName(potential.getQName());
513             if (child != null) {
514                 augmentation = aug;
515                 break;
516             }
517
518         }
519         if (augmentation != null) {
520             return new AugmentationNormalization(augmentation, schema);
521         } else {
522             return null;
523         }
524     }
525
526     private static DataNormalizationOperation<?> fromSchema(final DataNodeContainer schema, final PathArgument child) {
527         if (child instanceof AugmentationIdentifier) {
528             return fromSchemaAndPathArgument(schema, ((AugmentationIdentifier) child).getPossibleChildNames()
529                     .iterator().next());
530         }
531         return fromSchemaAndPathArgument(schema, child.getNodeType());
532     }
533
534     public static DataNormalizationOperation<?> fromDataSchemaNode(final DataSchemaNode potential) {
535         if (potential instanceof ContainerSchemaNode) {
536             return new ContainerNormalization((ContainerSchemaNode) potential);
537         } else if (potential instanceof ListSchemaNode) {
538             return new ListMixinNormalization((ListSchemaNode) potential);
539         } else if (potential instanceof LeafSchemaNode) {
540             return new LeafNormalization(new NodeIdentifier(potential.getQName()));
541         } else if (potential instanceof org.opendaylight.yangtools.yang.model.api.ChoiceNode) {
542             return new ChoiceNodeNormalization((org.opendaylight.yangtools.yang.model.api.ChoiceNode) potential);
543         } else if (potential instanceof LeafListSchemaNode) {
544             return new LeafListMixinNormalization((LeafListSchemaNode) potential);
545         }
546         return null;
547     }
548
549     public static DataNormalizationOperation<?> from(final SchemaContext ctx) {
550         return new ContainerNormalization(ctx);
551     }
552
553     public abstract NormalizedNode<?, ?> createDefault(PathArgument currentArg);
554 }