2978a58a2eb60ea31e9bc9e0cd60e2a591282cba
[yangtools.git] / benchmarks / src / main / java / org / opendaylight / yangtools / yang / data / impl / tree / InMemoryDataTreeBenchmark.java
1 /*
2  * Copyright (c) 2013 Cisco Systems, Inc. and others.  All rights reserved.
3  *
4  * This program and the accompanying materials are made available under the
5  * terms of the Eclipse Public License v1.0 which accompanies this distribution,
6  * and is available at http://www.eclipse.org/legal/epl-v10.html
7  */
8 package org.opendaylight.yangtools.yang.data.impl.tree;
9
10 import com.google.common.collect.Streams;
11 import java.io.IOException;
12 import java.util.Arrays;
13 import java.util.concurrent.TimeUnit;
14 import java.util.stream.Collectors;
15 import java.util.stream.IntStream;
16 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
17 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifierWithPredicates;
18 import org.opendaylight.yangtools.yang.data.api.schema.MapEntryNode;
19 import org.opendaylight.yangtools.yang.data.api.schema.MapNode;
20 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
21 import org.opendaylight.yangtools.yang.data.api.schema.tree.CursorAwareDataTreeModification;
22 import org.opendaylight.yangtools.yang.data.api.schema.tree.DataTree;
23 import org.opendaylight.yangtools.yang.data.api.schema.tree.DataTreeConfiguration;
24 import org.opendaylight.yangtools.yang.data.api.schema.tree.DataTreeModification;
25 import org.opendaylight.yangtools.yang.data.api.schema.tree.DataValidationFailedException;
26 import org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNodes;
27 import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.CollectionNodeBuilder;
28 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableContainerNodeBuilder;
29 import org.opendaylight.yangtools.yang.data.impl.schema.tree.InMemoryDataTreeFactory;
30 import org.openjdk.jmh.annotations.Benchmark;
31 import org.openjdk.jmh.annotations.BenchmarkMode;
32 import org.openjdk.jmh.annotations.Fork;
33 import org.openjdk.jmh.annotations.Level;
34 import org.openjdk.jmh.annotations.Measurement;
35 import org.openjdk.jmh.annotations.Mode;
36 import org.openjdk.jmh.annotations.OutputTimeUnit;
37 import org.openjdk.jmh.annotations.Scope;
38 import org.openjdk.jmh.annotations.Setup;
39 import org.openjdk.jmh.annotations.State;
40 import org.openjdk.jmh.annotations.TearDown;
41 import org.openjdk.jmh.annotations.Warmup;
42 import org.openjdk.jmh.runner.Runner;
43 import org.openjdk.jmh.runner.RunnerException;
44 import org.openjdk.jmh.runner.options.Options;
45 import org.openjdk.jmh.runner.options.OptionsBuilder;
46
47 /**
48  * Benchmarking of InMemoryDataTree performance.
49  *
50  * JMH is used for microbenchmarking.
51  *
52  * @author Lukas Sedlak <lsedlak@cisco.com>
53  *
54  * @see <a href="http://openjdk.java.net/projects/code-tools/jmh/">JMH</a>
55  */
56 @State(Scope.Thread)
57 @BenchmarkMode(Mode.AverageTime)
58 @OutputTimeUnit(TimeUnit.MILLISECONDS)
59 @Fork(1)
60 public class InMemoryDataTreeBenchmark {
61
62     private static final int WARMUP_ITERATIONS = 10;
63     private static final int MEASUREMENT_ITERATIONS = 10;
64
65     private static final int OUTER_LIST_100K = 100000;
66     private static final int OUTER_LIST_50K = 50000;
67     private static final int OUTER_LIST_10K = 10000;
68
69     private static final NodeIdentifierWithPredicates[] OUTER_LIST_IDS = Streams.mapWithIndex(
70         IntStream.range(0, OUTER_LIST_100K),
71         (i, index) -> new NodeIdentifierWithPredicates(BenchmarkModel.OUTER_LIST_QNAME, BenchmarkModel.ID_QNAME, i))
72             .collect(Collectors.toList()).toArray(new NodeIdentifierWithPredicates[0]);
73
74     private static final YangInstanceIdentifier[] OUTER_LIST_PATHS = Arrays.stream(OUTER_LIST_IDS)
75             .map(id -> BenchmarkModel.OUTER_LIST_PATH.node(id).toOptimized())
76             .collect(Collectors.toList()).toArray(new YangInstanceIdentifier[0]);
77
78     private static final MapNode EMPTY_OUTER_LIST = ImmutableNodes.mapNodeBuilder(BenchmarkModel.OUTER_LIST).build();
79     private static final MapNode ONE_ITEM_INNER_LIST = initInnerListItems(1);
80     private static final MapNode TWO_ITEM_INNER_LIST = initInnerListItems(2);
81     private static final MapNode TEN_ITEM_INNER_LIST = initInnerListItems(10);
82
83     private static MapNode initInnerListItems(final int count) {
84         final CollectionNodeBuilder<MapEntryNode, MapNode> mapEntryBuilder = ImmutableNodes
85             .mapNodeBuilder(BenchmarkModel.INNER_LIST);
86
87         for (int i = 0; i < count; ++i) {
88             mapEntryBuilder
89                 .withChild(ImmutableNodes.mapEntry(BenchmarkModel.INNER_LIST_QNAME, BenchmarkModel.NAME_QNAME, i));
90         }
91
92         return mapEntryBuilder.build();
93     }
94
95     private static final NormalizedNode<?, ?>[] OUTER_LIST_ONE_ITEM_INNER_LIST = initOuterListItems(OUTER_LIST_100K,
96         ONE_ITEM_INNER_LIST);
97     private static final NormalizedNode<?, ?>[] OUTER_LIST_TWO_ITEM_INNER_LIST = initOuterListItems(OUTER_LIST_50K,
98         TWO_ITEM_INNER_LIST);
99     private static final NormalizedNode<?, ?>[] OUTER_LIST_TEN_ITEM_INNER_LIST = initOuterListItems(OUTER_LIST_10K,
100         TEN_ITEM_INNER_LIST);
101
102     private static NormalizedNode<?,?>[] initOuterListItems(final int outerListItemsCount, final MapNode innerList) {
103         return Arrays.stream(OUTER_LIST_IDS).limit(outerListItemsCount)
104                 .map(id -> ImmutableNodes.mapEntryBuilder().withNodeIdentifier(id).withChild(innerList).build())
105                 .collect(Collectors.toList()).toArray(new NormalizedNode[0]);
106     }
107
108     private DataTree datastore;
109
110     public static void main(final String... args) throws IOException, RunnerException {
111         Options opt = new OptionsBuilder()
112             .include(".*" + InMemoryDataTreeBenchmark.class.getSimpleName() + ".*")
113             .forks(1)
114             .build();
115
116         new Runner(opt).run();
117     }
118
119     @Setup(Level.Trial)
120     public void setup() throws DataValidationFailedException {
121         datastore = new InMemoryDataTreeFactory().create(DataTreeConfiguration.DEFAULT_CONFIGURATION,
122             BenchmarkModel.createTestContext());
123
124         final DataTreeModification modification = begin();
125         modification.write(BenchmarkModel.TEST_PATH, ImmutableContainerNodeBuilder.create()
126             .withNodeIdentifier(BenchmarkModel.TEST).withChild(EMPTY_OUTER_LIST).build());
127         commit(modification);
128     }
129
130     @TearDown
131     public void tearDown() {
132         datastore = null;
133     }
134
135     @Benchmark
136     @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
137     @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
138     public void write100KSingleNodeWithOneInnerItemInOneCommitBenchmark() throws DataValidationFailedException {
139         final DataTreeModification modification = begin();
140         for (int outerListKey = 0; outerListKey < OUTER_LIST_100K; ++outerListKey) {
141             modification.write(OUTER_LIST_PATHS[outerListKey], OUTER_LIST_ONE_ITEM_INNER_LIST[outerListKey]);
142         }
143         commit(modification);
144     }
145
146     @Benchmark
147     @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
148     @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
149     public void write100KSingleNodeWithOneInnerItemInCommitPerWriteBenchmark() throws DataValidationFailedException {
150         for (int outerListKey = 0; outerListKey < OUTER_LIST_100K; ++outerListKey) {
151             final DataTreeModification modification = begin();
152             modification.write(OUTER_LIST_PATHS[outerListKey], OUTER_LIST_ONE_ITEM_INNER_LIST[outerListKey]);
153             commit(modification);
154         }
155     }
156
157     @Benchmark
158     @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
159     @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
160     public void write50KSingleNodeWithTwoInnerItemsInOneCommitBenchmark() throws DataValidationFailedException {
161         final DataTreeModification modification = begin();
162         for (int outerListKey = 0; outerListKey < OUTER_LIST_50K; ++outerListKey) {
163             modification.write(OUTER_LIST_PATHS[outerListKey], OUTER_LIST_TWO_ITEM_INNER_LIST[outerListKey]);
164         }
165         commit(modification);
166     }
167
168     @Benchmark
169     @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
170     @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
171     public void write50KSingleNodeWithTwoInnerItemsInCommitPerWriteBenchmark() throws DataValidationFailedException {
172         for (int outerListKey = 0; outerListKey < OUTER_LIST_50K; ++outerListKey) {
173             final DataTreeModification modification = begin();
174             modification.write(OUTER_LIST_PATHS[outerListKey], OUTER_LIST_TWO_ITEM_INNER_LIST[outerListKey]);
175             commit(modification);
176         }
177     }
178
179     @Benchmark
180     @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
181     @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
182     public void write10KSingleNodeWithTenInnerItemsInOneCommitBenchmark() throws DataValidationFailedException {
183         final DataTreeModification modification = begin();
184         for (int outerListKey = 0; outerListKey < OUTER_LIST_10K; ++outerListKey) {
185             modification.write(OUTER_LIST_PATHS[outerListKey], OUTER_LIST_TEN_ITEM_INNER_LIST[outerListKey]);
186         }
187         commit(modification);
188     }
189
190     @Benchmark
191     @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
192     @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
193     public void write10KSingleNodeWithTenInnerItemsInCommitPerWriteBenchmark() throws DataValidationFailedException {
194         for (int outerListKey = 0; outerListKey < OUTER_LIST_10K; ++outerListKey) {
195             final DataTreeModification modification = begin();
196             modification.write(OUTER_LIST_PATHS[outerListKey], OUTER_LIST_TEN_ITEM_INNER_LIST[outerListKey]);
197             commit(modification);
198         }
199     }
200
201     private CursorAwareDataTreeModification begin() {
202         return (CursorAwareDataTreeModification) datastore.takeSnapshot().newModification();
203     }
204
205     private void commit(final DataTreeModification modification) throws DataValidationFailedException {
206         modification.ready();
207         datastore.validate(modification);
208         datastore.commit(datastore.prepare(modification));
209     }
210 }