├── .gitignore
├── LICENSE
├── README.md
├── pom.xml
└── src
├── assembly.xml
└── main
└── java
└── io
└── imply
└── druid
└── query
├── DruidClient.java
└── ExampleMain.java
/.gitignore:
--------------------------------------------------------------------------------
1 | dist
2 | target
3 | *.iml
4 | *.ipr
5 | *.iws
6 | *.tar.gz
7 | *.swp
8 | *.swo
9 | .classpath
10 | .idea
11 | .project
12 | .settings/
13 | *.log
14 | *.DS_Store
15 | _site
16 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 |
2 | Apache License
3 | Version 2.0, January 2004
4 | http://www.apache.org/licenses/
5 |
6 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
7 |
8 | 1. Definitions.
9 |
10 | "License" shall mean the terms and conditions for use, reproduction,
11 | and distribution as defined by Sections 1 through 9 of this document.
12 |
13 | "Licensor" shall mean the copyright owner or entity authorized by
14 | the copyright owner that is granting the License.
15 |
16 | "Legal Entity" shall mean the union of the acting entity and all
17 | other entities that control, are controlled by, or are under common
18 | control with that entity. For the purposes of this definition,
19 | "control" means (i) the power, direct or indirect, to cause the
20 | direction or management of such entity, whether by contract or
21 | otherwise, or (ii) ownership of fifty percent (50%) or more of the
22 | outstanding shares, or (iii) beneficial ownership of such entity.
23 |
24 | "You" (or "Your") shall mean an individual or Legal Entity
25 | exercising permissions granted by this License.
26 |
27 | "Source" form shall mean the preferred form for making modifications,
28 | including but not limited to software source code, documentation
29 | source, and configuration files.
30 |
31 | "Object" form shall mean any form resulting from mechanical
32 | transformation or translation of a Source form, including but
33 | not limited to compiled object code, generated documentation,
34 | and conversions to other media types.
35 |
36 | "Work" shall mean the work of authorship, whether in Source or
37 | Object form, made available under the License, as indicated by a
38 | copyright notice that is included in or attached to the work
39 | (an example is provided in the Appendix below).
40 |
41 | "Derivative Works" shall mean any work, whether in Source or Object
42 | form, that is based on (or derived from) the Work and for which the
43 | editorial revisions, annotations, elaborations, or other modifications
44 | represent, as a whole, an original work of authorship. For the purposes
45 | of this License, Derivative Works shall not include works that remain
46 | separable from, or merely link (or bind by name) to the interfaces of,
47 | the Work and Derivative Works thereof.
48 |
49 | "Contribution" shall mean any work of authorship, including
50 | the original version of the Work and any modifications or additions
51 | to that Work or Derivative Works thereof, that is intentionally
52 | submitted to Licensor for inclusion in the Work by the copyright owner
53 | or by an individual or Legal Entity authorized to submit on behalf of
54 | the copyright owner. For the purposes of this definition, "submitted"
55 | means any form of electronic, verbal, or written communication sent
56 | to the Licensor or its representatives, including but not limited to
57 | communication on electronic mailing lists, source code control systems,
58 | and issue tracking systems that are managed by, or on behalf of, the
59 | Licensor for the purpose of discussing and improving the Work, but
60 | excluding communication that is conspicuously marked or otherwise
61 | designated in writing by the copyright owner as "Not a Contribution."
62 |
63 | "Contributor" shall mean Licensor and any individual or Legal Entity
64 | on behalf of whom a Contribution has been received by Licensor and
65 | subsequently incorporated within the Work.
66 |
67 | 2. Grant of Copyright License. Subject to the terms and conditions of
68 | this License, each Contributor hereby grants to You a perpetual,
69 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable
70 | copyright license to reproduce, prepare Derivative Works of,
71 | publicly display, publicly perform, sublicense, and distribute the
72 | Work and such Derivative Works in Source or Object form.
73 |
74 | 3. Grant of Patent License. Subject to the terms and conditions of
75 | this License, each Contributor hereby grants to You a perpetual,
76 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable
77 | (except as stated in this section) patent license to make, have made,
78 | use, offer to sell, sell, import, and otherwise transfer the Work,
79 | where such license applies only to those patent claims licensable
80 | by such Contributor that are necessarily infringed by their
81 | Contribution(s) alone or by combination of their Contribution(s)
82 | with the Work to which such Contribution(s) was submitted. If You
83 | institute patent litigation against any entity (including a
84 | cross-claim or counterclaim in a lawsuit) alleging that the Work
85 | or a Contribution incorporated within the Work constitutes direct
86 | or contributory patent infringement, then any patent licenses
87 | granted to You under this License for that Work shall terminate
88 | as of the date such litigation is filed.
89 |
90 | 4. Redistribution. You may reproduce and distribute copies of the
91 | Work or Derivative Works thereof in any medium, with or without
92 | modifications, and in Source or Object form, provided that You
93 | meet the following conditions:
94 |
95 | (a) You must give any other recipients of the Work or
96 | Derivative Works a copy of this License; and
97 |
98 | (b) You must cause any modified files to carry prominent notices
99 | stating that You changed the files; and
100 |
101 | (c) You must retain, in the Source form of any Derivative Works
102 | that You distribute, all copyright, patent, trademark, and
103 | attribution notices from the Source form of the Work,
104 | excluding those notices that do not pertain to any part of
105 | the Derivative Works; and
106 |
107 | (d) If the Work includes a "NOTICE" text file as part of its
108 | distribution, then any Derivative Works that You distribute must
109 | include a readable copy of the attribution notices contained
110 | within such NOTICE file, excluding those notices that do not
111 | pertain to any part of the Derivative Works, in at least one
112 | of the following places: within a NOTICE text file distributed
113 | as part of the Derivative Works; within the Source form or
114 | documentation, if provided along with the Derivative Works; or,
115 | within a display generated by the Derivative Works, if and
116 | wherever such third-party notices normally appear. The contents
117 | of the NOTICE file are for informational purposes only and
118 | do not modify the License. You may add Your own attribution
119 | notices within Derivative Works that You distribute, alongside
120 | or as an addendum to the NOTICE text from the Work, provided
121 | that such additional attribution notices cannot be construed
122 | as modifying the License.
123 |
124 | You may add Your own copyright statement to Your modifications and
125 | may provide additional or different license terms and conditions
126 | for use, reproduction, or distribution of Your modifications, or
127 | for any such Derivative Works as a whole, provided Your use,
128 | reproduction, and distribution of the Work otherwise complies with
129 | the conditions stated in this License.
130 |
131 | 5. Submission of Contributions. Unless You explicitly state otherwise,
132 | any Contribution intentionally submitted for inclusion in the Work
133 | by You to the Licensor shall be under the terms and conditions of
134 | this License, without any additional terms or conditions.
135 | Notwithstanding the above, nothing herein shall supersede or modify
136 | the terms of any separate license agreement you may have executed
137 | with Licensor regarding such Contributions.
138 |
139 | 6. Trademarks. This License does not grant permission to use the trade
140 | names, trademarks, service marks, or product names of the Licensor,
141 | except as required for reasonable and customary use in describing the
142 | origin of the Work and reproducing the content of the NOTICE file.
143 |
144 | 7. Disclaimer of Warranty. Unless required by applicable law or
145 | agreed to in writing, Licensor provides the Work (and each
146 | Contributor provides its Contributions) on an "AS IS" BASIS,
147 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
148 | implied, including, without limitation, any warranties or conditions
149 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
150 | PARTICULAR PURPOSE. You are solely responsible for determining the
151 | appropriateness of using or redistributing the Work and assume any
152 | risks associated with Your exercise of permissions under this License.
153 |
154 | 8. Limitation of Liability. In no event and under no legal theory,
155 | whether in tort (including negligence), contract, or otherwise,
156 | unless required by applicable law (such as deliberate and grossly
157 | negligent acts) or agreed to in writing, shall any Contributor be
158 | liable to You for damages, including any direct, indirect, special,
159 | incidental, or consequential damages of any character arising as a
160 | result of this License or out of the use or inability to use the
161 | Work (including but not limited to damages for loss of goodwill,
162 | work stoppage, computer failure or malfunction, or any and all
163 | other commercial damages or losses), even if such Contributor
164 | has been advised of the possibility of such damages.
165 |
166 | 9. Accepting Warranty or Additional Liability. While redistributing
167 | the Work or Derivative Works thereof, You may choose to offer,
168 | and charge a fee for, acceptance of support, warranty, indemnity,
169 | or other liability obligations and/or rights consistent with this
170 | License. However, in accepting such obligations, You may act only
171 | on Your own behalf and on Your sole responsibility, not on behalf
172 | of any other Contributor, and only if You agree to indemnify,
173 | defend, and hold each Contributor harmless for any liability
174 | incurred by, or claims asserted against, such Contributor by reason
175 | of your accepting any such warranty or additional liability.
176 |
177 | END OF TERMS AND CONDITIONS
178 |
179 | APPENDIX: How to apply the Apache License to your work.
180 |
181 | To apply the Apache License to your work, attach the following
182 | boilerplate notice, with the fields enclosed by brackets "[]"
183 | replaced with your own identifying information. (Don't include
184 | the brackets!) The text should be enclosed in the appropriate
185 | comment syntax for the file format. We also recommend that a
186 | file or class name and description of purpose be included on the
187 | same "printed page" as the copyright notice for easier
188 | identification within third-party archives.
189 |
190 | Copyright [yyyy] [name of copyright owner]
191 |
192 | Licensed under the Apache License, Version 2.0 (the "License");
193 | you may not use this file except in compliance with the License.
194 | You may obtain a copy of the License at
195 |
196 | http://www.apache.org/licenses/LICENSE-2.0
197 |
198 | Unless required by applicable law or agreed to in writing, software
199 | distributed under the License is distributed on an "AS IS" BASIS,
200 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
201 | See the License for the specific language governing permissions and
202 | limitations under the License.
203 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | ## Druid Client
2 |
3 | This is a client that can be used to make Druid queries from the JVM. Note that unlike the
4 | [JSON query API](http://druid.io/docs/latest/querying/querying.html), this depends on non-stable Druid
5 | APIs. You may need to adjust code even for minor Druid updates.
6 |
7 | ### Installation
8 |
9 | To install this library, run `mvn install`. You can then include it in projects with Maven by using
10 | the dependency:
11 |
12 | ```xml
13 |
14 | io.imply
15 | druid-client
16 | 0.1-SNAPSHOT
17 |
18 | ```
19 |
20 | ### Example
21 |
22 | Check out [ExampleMain.java](https://github.com/implydata/druid-client/blob/master/src/main/java/io/imply/druid/query/ExampleMain.java) for example code.
23 |
--------------------------------------------------------------------------------
/pom.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
18 |
19 |
21 | 4.0.0
22 |
23 | io.imply
24 | druid-client
25 | 0.11.0-1-SNAPSHOT
26 |
27 |
28 | scm:git:https://github.com/implydata/druid-client.git
29 | scm:git:ssh://git@github.com/implydata/druid-client.git
30 | HEAD
31 |
32 |
33 |
34 | UTF-8
35 | 0.11.0
36 |
37 |
38 |
39 |
40 | io.druid
41 | druid-server
42 | ${druid.version}
43 |
44 |
45 |
46 |
47 | junit
48 | junit
49 | 4.12
50 | test
51 |
52 |
53 |
54 |
55 |
56 |
57 | org.apache.maven.plugins
58 | maven-compiler-plugin
59 | 3.1
60 |
61 | -Xlint:unchecked
62 | 1.7
63 | 1.7
64 |
65 |
66 |
67 | org.apache.maven.plugins
68 | maven-assembly-plugin
69 | 2.5.5
70 |
71 |
72 | distro-assembly
73 | package
74 |
75 | single
76 |
77 |
78 | ${project.artifactId}-${project.version}
79 | false
80 | posix
81 |
82 | src/assembly/assembly.xml
83 |
84 |
85 |
86 |
87 |
88 |
89 | org.apache.maven.plugins
90 | maven-surefire-plugin
91 | 2.19.1
92 |
93 | -Duser.timezone=UTC
94 | true
95 |
96 |
97 |
98 | maven-release-plugin
99 | 2.5.3
100 |
101 |
102 | org.apache.maven.scm
103 | maven-scm-provider-gitexe
104 | 1.9.5
105 |
106 |
107 |
108 |
109 | org.apache.maven.plugins
110 | maven-jar-plugin
111 | 3.0.2
112 |
113 |
114 | false
115 |
116 |
117 |
118 |
119 |
120 |
121 |
--------------------------------------------------------------------------------
/src/assembly.xml:
--------------------------------------------------------------------------------
1 |
2 |
17 |
20 | bin
21 |
22 | tar.gz
23 |
24 |
25 | ${project.name}
26 |
27 |
28 |
29 | false
30 | true
31 | .
32 | false
33 |
34 |
35 |
36 |
37 |
38 | ${project.build.directory}
39 | .
40 |
41 | *.jar
42 |
43 |
44 |
45 |
46 |
--------------------------------------------------------------------------------
/src/main/java/io/imply/druid/query/DruidClient.java:
--------------------------------------------------------------------------------
1 | /*
2 | * Copyright 2016 Imply Data, Inc.
3 | *
4 | * Licensed under the Apache License, Version 2.0 (the "License");
5 | * you may not use this file except in compliance with the License.
6 | * You may obtain a copy of the License at
7 | *
8 | * http://www.apache.org/licenses/LICENSE-2.0
9 | *
10 | * Unless required by applicable law or agreed to in writing, software
11 | * distributed under the License is distributed on an "AS IS" BASIS,
12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 | * See the License for the specific language governing permissions and
14 | * limitations under the License.
15 | */
16 |
17 | package io.imply.druid.query;
18 |
19 | import com.fasterxml.jackson.databind.ObjectMapper;
20 | import com.google.common.base.Throwables;
21 | import com.google.common.collect.ImmutableList;
22 | import com.google.common.collect.Maps;
23 | import com.google.common.util.concurrent.ListenableFuture;
24 | import com.google.inject.Binder;
25 | import com.google.inject.Injector;
26 | import com.google.inject.Key;
27 | import com.google.inject.Module;
28 | import com.metamx.common.guava.Sequence;
29 | import com.metamx.common.lifecycle.Lifecycle;
30 | import com.metamx.common.logger.Logger;
31 | import com.metamx.emitter.core.NoopEmitter;
32 | import com.metamx.emitter.service.ServiceEmitter;
33 | import com.metamx.http.client.HttpClient;
34 | import com.metamx.http.client.HttpClientConfig;
35 | import com.metamx.http.client.HttpClientInit;
36 | import io.druid.client.DirectDruidClient;
37 | import io.druid.guice.GuiceInjectors;
38 | import io.druid.guice.JsonConfigProvider;
39 | import io.druid.guice.annotations.Json;
40 | import io.druid.guice.annotations.Self;
41 | import io.druid.guice.annotations.Smile;
42 | import io.druid.guice.http.DruidHttpClientConfig;
43 | import io.druid.initialization.Initialization;
44 | import io.druid.query.DruidProcessingConfig;
45 | import io.druid.query.MapQueryToolChestWarehouse;
46 | import io.druid.query.Query;
47 | import io.druid.query.QueryToolChestWarehouse;
48 | import io.druid.query.QueryWatcher;
49 | import io.druid.server.DruidNode;
50 |
51 | import java.io.Closeable;
52 | import java.io.IOException;
53 | import java.util.Map;
54 |
55 | public class DruidClient implements Closeable
56 | {
57 | private static final Logger log = new Logger(DruidClient.class);
58 |
59 | private static final Injector INJECTOR;
60 | private static final QueryToolChestWarehouse WAREHOUSE;
61 | private static final QueryWatcher WATCHER;
62 | private static final ObjectMapper JSON_MAPPER;
63 | private static final ObjectMapper SMILE_MAPPER;
64 | private static final DruidHttpClientConfig HTTP_CLIENT_CONFIG;
65 | private static final ServiceEmitter SERVICE_EMITTER;
66 |
67 | static {
68 | INJECTOR = Initialization.makeInjectorWithModules(
69 | GuiceInjectors.makeStartupInjector(),
70 | ImmutableList.of(
71 | new Module()
72 | {
73 | @Override
74 | public void configure(Binder binder)
75 | {
76 | JsonConfigProvider.bindInstance(
77 | binder,
78 | Key.get(DruidNode.class, Self.class),
79 | new DruidNode("druid-client", null, null)
80 | );
81 | }
82 | },
83 | new Module()
84 | {
85 | @Override
86 | public void configure(Binder binder)
87 | {
88 | binder.bind(QueryToolChestWarehouse.class).to(MapQueryToolChestWarehouse.class);
89 | JsonConfigProvider.bind(binder, "druid.client.http", DruidHttpClientConfig.class);
90 |
91 | // Set up dummy DruidProcessingConfig to avoid large offheap buffer generation.
92 | final DruidProcessingConfig dummyConfig = new DruidProcessingConfig()
93 | {
94 | @Override
95 | public int intermediateComputeSizeBytes()
96 | {
97 | return 1;
98 | }
99 |
100 | @Override
101 | public String getFormatString()
102 | {
103 | return "dummy";
104 | }
105 | };
106 | binder.bind(DruidProcessingConfig.class).toInstance(dummyConfig);
107 | }
108 | }
109 | )
110 | );
111 | WAREHOUSE = INJECTOR.getInstance(QueryToolChestWarehouse.class);
112 | WATCHER = new QueryWatcher()
113 | {
114 | @Override
115 | public void registerQuery(Query query, ListenableFuture future)
116 | {
117 | }
118 | };
119 | JSON_MAPPER = INJECTOR.getInstance(Key.get(ObjectMapper.class, Json.class));
120 | SMILE_MAPPER = INJECTOR.getInstance(Key.get(ObjectMapper.class, Smile.class));
121 | HTTP_CLIENT_CONFIG = INJECTOR.getInstance(DruidHttpClientConfig.class);
122 | SERVICE_EMITTER = new ServiceEmitter("druid-client", "localhost", new NoopEmitter());
123 | }
124 |
125 | private final DirectDruidClient directDruidClient;
126 | private final Closeable closeable;
127 |
128 | private DruidClient(DirectDruidClient directDruidClient, Closeable closeable)
129 | {
130 | this.directDruidClient = directDruidClient;
131 | this.closeable = closeable;
132 | }
133 |
134 | /**
135 | * Creates a DruidClient that owns its own HttpClient.
136 | *
137 | * @param host broker host and port
138 | *
139 | * @return druid client
140 | */
141 | public static DruidClient create(final String host)
142 | {
143 | final HttpClientConfig.Builder builder = HttpClientConfig
144 | .builder()
145 | .withNumConnections(HTTP_CLIENT_CONFIG.getNumConnections())
146 | .withReadTimeout(HTTP_CLIENT_CONFIG.getReadTimeout());
147 |
148 | final Lifecycle lifecycle = new Lifecycle();
149 | final HttpClient httpClient = HttpClientInit.createClient(builder.build(), lifecycle);
150 | final DirectDruidClient directDruidClient = new DirectDruidClient(
151 | WAREHOUSE,
152 | WATCHER,
153 | SMILE_MAPPER,
154 | httpClient,
155 | host,
156 | SERVICE_EMITTER
157 | );
158 | return new DruidClient(
159 | directDruidClient,
160 | new Closeable()
161 | {
162 | @Override
163 | public void close() throws IOException
164 | {
165 | lifecycle.stop();
166 | }
167 | }
168 | );
169 | }
170 |
171 | /**
172 | * Creates a DruidClient using a shared HttpClient. The shared HttpClient will not be closed when this DruidClient
173 | * is closed.
174 | *
175 | * @param host broker host and port
176 | * @param httpClient shared HttpClient
177 | *
178 | * @return druid client
179 | */
180 | public static DruidClient create(final String host, final HttpClient httpClient)
181 | {
182 | final DirectDruidClient directDruidClient = new DirectDruidClient(
183 | WAREHOUSE,
184 | WATCHER,
185 | SMILE_MAPPER,
186 | httpClient,
187 | host,
188 | SERVICE_EMITTER
189 | );
190 | return new DruidClient(directDruidClient, null);
191 | }
192 |
193 | public ObjectMapper getJsonMapper()
194 | {
195 | return JSON_MAPPER;
196 | }
197 |
198 | public Sequence execute(final Query query)
199 | {
200 | final Map context = Maps.newHashMap();
201 | try {
202 | log.debug("Issuing query: %s", getJsonMapper().writeValueAsString(query));
203 | }
204 | catch (Exception e) {
205 | throw Throwables.propagate(e);
206 | }
207 | return directDruidClient.run(query, context);
208 | }
209 |
210 | public Sequence execute(final Map queryMap, final Class extends Query> queryClass)
211 | {
212 | return execute(JSON_MAPPER.convertValue(queryMap, queryClass));
213 | }
214 |
215 | @Override
216 | public void close() throws IOException
217 | {
218 | if (closeable != null) {
219 | closeable.close();
220 | }
221 | }
222 | }
223 |
--------------------------------------------------------------------------------
/src/main/java/io/imply/druid/query/ExampleMain.java:
--------------------------------------------------------------------------------
1 | /*
2 | * Copyright 2016 Imply Data, Inc.
3 | *
4 | * Licensed under the Apache License, Version 2.0 (the "License");
5 | * you may not use this file except in compliance with the License.
6 | * You may obtain a copy of the License at
7 | *
8 | * http://www.apache.org/licenses/LICENSE-2.0
9 | *
10 | * Unless required by applicable law or agreed to in writing, software
11 | * distributed under the License is distributed on an "AS IS" BASIS,
12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 | * See the License for the specific language governing permissions and
14 | * limitations under the License.
15 | */
16 |
17 | package io.imply.druid.query;
18 |
19 | import com.google.common.collect.ImmutableList;
20 | import com.google.common.collect.Lists;
21 | import com.metamx.common.guava.Sequence;
22 | import com.metamx.common.guava.Sequences;
23 | import io.druid.query.Druids;
24 | import io.druid.query.Result;
25 | import io.druid.query.filter.AndDimFilter;
26 | import io.druid.query.filter.DimFilter;
27 | import io.druid.query.filter.SelectorDimFilter;
28 | import io.druid.query.select.EventHolder;
29 | import io.druid.query.select.PagingSpec;
30 | import io.druid.query.select.SelectQuery;
31 | import io.druid.query.select.SelectResultValue;
32 | import org.joda.time.Interval;
33 |
34 | import java.util.List;
35 |
36 | public class ExampleMain
37 | {
38 | public static void main(String[] args) throws Exception
39 | {
40 | final String host = args.length == 0 ? "localhost:8082" : args[0];
41 | try (final DruidClient druidClient = DruidClient.create(host)) {
42 | // Create a simple select query using the Druids query builder.
43 | final int threshold = 50;
44 | final SelectQuery selectQuery = Druids
45 | .newSelectQueryBuilder()
46 | .dataSource("wikiticker")
47 | .intervals(ImmutableList.of(new Interval("1000/3000")))
48 | .filters(
49 | new AndDimFilter(
50 | ImmutableList.of(
51 | new SelectorDimFilter("countryName", "United States", null),
52 | new SelectorDimFilter("cityName", "San Francisco", null)
53 | )
54 | )
55 | )
56 | .dimensions(ImmutableList.of("page", "user"))
57 | .pagingSpec(new PagingSpec(null, threshold))
58 | .build();
59 |
60 | // Fetch the results.
61 | final long startTime = System.currentTimeMillis();
62 | final Sequence> resultSequence = druidClient.execute(selectQuery);
63 | final List> resultList = Sequences.toList(
64 | resultSequence,
65 | Lists.>newArrayList()
66 | );
67 | final long fetchTime = System.currentTimeMillis() - startTime;
68 |
69 | // Print the results.
70 | int resultCount = 0;
71 | for (final Result result : resultList) {
72 | for (EventHolder eventHolder : result.getValue().getEvents()) {
73 | System.out.println(eventHolder.getEvent());
74 | resultCount++;
75 | }
76 | }
77 |
78 | // Print statistics.
79 | System.out.println(
80 | String.format(
81 | "Fetched %,d rows in %,dms.",
82 | resultCount,
83 | fetchTime
84 | )
85 | );
86 | }
87 | }
88 | }
89 |
--------------------------------------------------------------------------------