2 * Copyright (c) 2013 Cisco Systems, Inc. and others. All rights reserved.
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
8 package org.opendaylight.yangtools.yang.data.impl.tree;
10 import org.opendaylight.yangtools.yang.parser.spi.meta.ReactorException;
11 import org.opendaylight.yangtools.yang.parser.spi.source.SourceException;
13 import java.io.IOException;
14 import java.util.concurrent.TimeUnit;
15 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
16 import org.opendaylight.yangtools.yang.data.api.schema.DataContainerChild;
17 import org.opendaylight.yangtools.yang.data.api.schema.MapEntryNode;
18 import org.opendaylight.yangtools.yang.data.api.schema.MapNode;
19 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
20 import org.opendaylight.yangtools.yang.data.api.schema.tree.*;
21 import org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNodes;
22 import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.CollectionNodeBuilder;
23 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableContainerNodeBuilder;
24 import org.opendaylight.yangtools.yang.data.impl.schema.tree.InMemoryDataTreeFactory;
25 import org.opendaylight.yangtools.yang.model.api.SchemaContext;
26 import org.openjdk.jmh.annotations.*;
27 import org.openjdk.jmh.runner.Runner;
28 import org.openjdk.jmh.runner.RunnerException;
29 import org.openjdk.jmh.runner.options.Options;
30 import org.openjdk.jmh.runner.options.OptionsBuilder;
33 * Benchmarking of InMemoryDataTree performance.
35 * JMH is used for microbenchmarking.
37 * @author Lukas Sedlak <lsedlak@cisco.com>
39 * @see <a href="http://openjdk.java.net/projects/code-tools/jmh/">JMH</a>
42 @BenchmarkMode(Mode.AverageTime)
43 @OutputTimeUnit(TimeUnit.MILLISECONDS)
45 public class InMemoryDataTreeBenchmark {
47 private static final int WARMUP_ITERATIONS = 20;
48 private static final int MEASUREMENT_ITERATIONS = 20;
50 private static final int OUTER_LIST_100K = 100000;
51 private static final int OUTER_LIST_50K = 50000;
52 private static final int OUTER_LIST_10K = 10000;
54 private static final YangInstanceIdentifier[] OUTER_LIST_100K_PATHS = initOuterListPaths(OUTER_LIST_100K);
55 private static final YangInstanceIdentifier[] OUTER_LIST_50K_PATHS = initOuterListPaths(OUTER_LIST_50K);
56 private static final YangInstanceIdentifier[] OUTER_LIST_10K_PATHS = initOuterListPaths(OUTER_LIST_10K);
58 private static YangInstanceIdentifier[] initOuterListPaths(final int outerListPathsCount) {
59 final YangInstanceIdentifier[] paths = new YangInstanceIdentifier[outerListPathsCount];
61 for (int outerListKey = 0; outerListKey < outerListPathsCount; ++outerListKey) {
62 paths[outerListKey] = YangInstanceIdentifier.builder(BenchmarkModel.OUTER_LIST_PATH)
63 .nodeWithKey(BenchmarkModel.OUTER_LIST_QNAME, BenchmarkModel.ID_QNAME, outerListKey)
69 private static final MapNode ONE_ITEM_INNER_LIST = initInnerListItems(1);
70 private static final MapNode TWO_ITEM_INNER_LIST = initInnerListItems(2);
71 private static final MapNode TEN_ITEM_INNER_LIST = initInnerListItems(10);
73 private static MapNode initInnerListItems(final int count) {
74 final CollectionNodeBuilder<MapEntryNode, MapNode> mapEntryBuilder = ImmutableNodes
75 .mapNodeBuilder(BenchmarkModel.INNER_LIST_QNAME);
77 for (int i = 1; i <= count; ++i) {
79 .withChild(ImmutableNodes.mapEntry(BenchmarkModel.INNER_LIST_QNAME, BenchmarkModel.NAME_QNAME, i));
82 return mapEntryBuilder.build();
85 private static final NormalizedNode<?, ?>[] OUTER_LIST_ONE_ITEM_INNER_LIST = initOuterListItems(OUTER_LIST_100K, ONE_ITEM_INNER_LIST);
86 private static final NormalizedNode<?, ?>[] OUTER_LIST_TWO_ITEM_INNER_LIST = initOuterListItems(OUTER_LIST_50K, TWO_ITEM_INNER_LIST);
87 private static final NormalizedNode<?, ?>[] OUTER_LIST_TEN_ITEM_INNER_LIST = initOuterListItems(OUTER_LIST_10K, TEN_ITEM_INNER_LIST);
89 private static NormalizedNode<?,?>[] initOuterListItems(int outerListItemsCount, MapNode innerList) {
90 final NormalizedNode<?,?>[] outerListItems = new NormalizedNode[outerListItemsCount];
92 for (int i = 0; i < outerListItemsCount; ++i) {
94 outerListItems[i] = ImmutableNodes.mapEntryBuilder(BenchmarkModel.OUTER_LIST_QNAME, BenchmarkModel.ID_QNAME, outerListKey)
95 .withChild(innerList).build();
97 return outerListItems;
100 private SchemaContext schemaContext;
101 private DataTree datastore;
103 public static void main(String... args) throws IOException, RunnerException {
104 Options opt = new OptionsBuilder()
105 .include(".*" + InMemoryDataTreeBenchmark.class.getSimpleName() + ".*")
109 new Runner(opt).run();
113 public void setup() throws DataValidationFailedException, SourceException, ReactorException {
114 schemaContext = BenchmarkModel.createTestContext();
115 final InMemoryDataTreeFactory factory = InMemoryDataTreeFactory.getInstance();
116 datastore = factory.create();
117 datastore.setSchemaContext(schemaContext);
118 final DataTreeSnapshot snapshot = datastore.takeSnapshot();
119 initTestNode(snapshot);
123 public void tearDown() {
124 schemaContext = null;
128 private void initTestNode(final DataTreeSnapshot snapshot) throws DataValidationFailedException {
129 final DataTreeModification modification = snapshot.newModification();
130 final YangInstanceIdentifier testPath = YangInstanceIdentifier.builder(BenchmarkModel.TEST_PATH)
133 modification.write(testPath, provideOuterListNode());
134 datastore.validate(modification);
135 final DataTreeCandidate candidate = datastore.prepare(modification);
136 datastore.commit(candidate);
139 private DataContainerChild<?, ?> provideOuterListNode() {
140 return ImmutableContainerNodeBuilder
142 .withNodeIdentifier(new YangInstanceIdentifier.NodeIdentifier(BenchmarkModel.TEST_QNAME))
144 ImmutableNodes.mapNodeBuilder(BenchmarkModel.OUTER_LIST_QNAME)
149 @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
150 @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
151 public void write100KSingleNodeWithOneInnerItemInOneCommitBenchmark() throws Exception {
152 final DataTreeSnapshot snapshot = datastore.takeSnapshot();
153 final DataTreeModification modification = snapshot.newModification();
154 for (int outerListKey = 0; outerListKey < OUTER_LIST_100K; ++outerListKey) {
155 modification.write(OUTER_LIST_100K_PATHS[outerListKey], OUTER_LIST_ONE_ITEM_INNER_LIST[outerListKey]);
157 datastore.validate(modification);
158 final DataTreeCandidate candidate = datastore.prepare(modification);
159 datastore.commit(candidate);
163 @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
164 @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
165 public void write100KSingleNodeWithOneInnerItemInCommitPerWriteBenchmark() throws Exception {
166 final DataTreeSnapshot snapshot = datastore.takeSnapshot();
167 for (int outerListKey = 0; outerListKey < OUTER_LIST_100K; ++outerListKey) {
168 final DataTreeModification modification = snapshot.newModification();
169 modification.write(OUTER_LIST_100K_PATHS[outerListKey], OUTER_LIST_ONE_ITEM_INNER_LIST[outerListKey]);
170 datastore.validate(modification);
171 final DataTreeCandidate candidate = datastore.prepare(modification);
172 datastore.commit(candidate);
177 @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
178 @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
179 public void write50KSingleNodeWithTwoInnerItemsInOneCommitBenchmark() throws Exception {
180 final DataTreeSnapshot snapshot = datastore.takeSnapshot();
181 final DataTreeModification modification = snapshot.newModification();
182 for (int outerListKey = 0; outerListKey < OUTER_LIST_50K; ++outerListKey) {
183 modification.write(OUTER_LIST_50K_PATHS[outerListKey], OUTER_LIST_TWO_ITEM_INNER_LIST[outerListKey]);
185 datastore.validate(modification);
186 final DataTreeCandidate candidate = datastore.prepare(modification);
187 datastore.commit(candidate);
191 @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
192 @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
193 public void write50KSingleNodeWithTwoInnerItemsInCommitPerWriteBenchmark() throws Exception {
194 final DataTreeSnapshot snapshot = datastore.takeSnapshot();
195 for (int outerListKey = 0; outerListKey < OUTER_LIST_50K; ++outerListKey) {
196 final DataTreeModification modification = snapshot.newModification();
197 modification.write(OUTER_LIST_50K_PATHS[outerListKey], OUTER_LIST_TWO_ITEM_INNER_LIST[outerListKey]);
198 datastore.validate(modification);
199 final DataTreeCandidate candidate = datastore.prepare(modification);
200 datastore.commit(candidate);
205 @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
206 @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
207 public void write10KSingleNodeWithTenInnerItemsInOneCommitBenchmark() throws Exception {
208 final DataTreeSnapshot snapshot = datastore.takeSnapshot();
209 final DataTreeModification modification = snapshot.newModification();
210 for (int outerListKey = 0; outerListKey < OUTER_LIST_10K; ++outerListKey) {
211 modification.write(OUTER_LIST_10K_PATHS[outerListKey], OUTER_LIST_TEN_ITEM_INNER_LIST[outerListKey]);
213 datastore.validate(modification);
214 final DataTreeCandidate candidate = datastore.prepare(modification);
215 datastore.commit(candidate);
219 @Warmup(iterations = WARMUP_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
220 @Measurement(iterations = MEASUREMENT_ITERATIONS, timeUnit = TimeUnit.MILLISECONDS)
221 public void write10KSingleNodeWithTenInnerItemsInCommitPerWriteBenchmark() throws Exception {
222 final DataTreeSnapshot snapshot = datastore.takeSnapshot();
223 for (int outerListKey = 0; outerListKey < OUTER_LIST_10K; ++outerListKey) {
224 final DataTreeModification modification = snapshot.newModification();
225 modification.write(OUTER_LIST_10K_PATHS[outerListKey], OUTER_LIST_TEN_ITEM_INNER_LIST[outerListKey]);
226 datastore.validate(modification);
227 final DataTreeCandidate candidate = datastore.prepare(modification);
228 datastore.commit(candidate);