From b5a9359ee0b043675f351f4cba312a5d802c5633 Mon Sep 17 00:00:00 2001 From: Sergey Chernov Date: Wed, 22 May 2024 22:20:54 -0700 Subject: [PATCH] added initial version of client v2 example --- examples/client-v2/README.md | 26 ++++ examples/client-v2/pom.xml | 147 ++++++++++++++++++ .../clickhouse/examples/client_v2/Main.java | 43 +++++ .../examples/client_v2/SimpleReader.java | 68 ++++++++ .../examples/client_v2/SimpleWriter.java | 74 +++++++++ .../resources/sample_hacker_news_posts.json | 10 ++ .../src/main/resources/simple_writer_init.sql | 17 ++ 7 files changed, 385 insertions(+) create mode 100644 examples/client-v2/README.md create mode 100644 examples/client-v2/pom.xml create mode 100644 examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/Main.java create mode 100644 examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/SimpleReader.java create mode 100644 examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/SimpleWriter.java create mode 100644 examples/client-v2/src/main/resources/sample_hacker_news_posts.json create mode 100644 examples/client-v2/src/main/resources/simple_writer_init.sql diff --git a/examples/client-v2/README.md b/examples/client-v2/README.md new file mode 100644 index 000000000..9b930ee2c --- /dev/null +++ b/examples/client-v2/README.md @@ -0,0 +1,26 @@ +# Client V2 Example + +## Overview + +This example demonstrates how to use the client V2 to interact with the + + +## How to Run + +Apache Maven or IDE with Maven support is required to run this example. + +First we need to compile the example : +```shell +mvn clean compile +``` + +To run: +```shell + mvn exec:java -Dexec.mainClass="com.clickhouse.examples.client_v2.Main" +``` + +Addition options can be passed to the application: +- `-DchEndpoint` - Endpoint to connect in the format of URL (default: http://localhost:8123/) +- `-DchUser` - ClickHouse user name (default: default) +- `-DchPassword` - ClickHouse user password (default: empty) +- `-DchDatabase` - ClickHouse database name (default: default) \ No newline at end of file diff --git a/examples/client-v2/pom.xml b/examples/client-v2/pom.xml new file mode 100644 index 000000000..f796940c5 --- /dev/null +++ b/examples/client-v2/pom.xml @@ -0,0 +1,147 @@ + + + 4.0.0 + + com.clickhouse + java-client-examples + 1.0.0 + jar + + java-client-examples + Java Client Examples + https://github.com/ClickHouse/clickhouse-java + 2022 + + + ClickHouse, Inc. + https://clickhouse.com/ + + + + + The Apache Software License, Version 2.0 + http://www.apache.org/licenses/LICENSE-2.0 + repo + + + + + https://github.com/ClickHouse/clickhouse-java + scm:git@github.com:ClickHouse/clickhouse-java.git + scm:git@github.com:ClickHouse/clickhouse-java.git + HEAD + + + + Github + https://github.com/ClickHouse/clickhouse-java/issues + + + + Github + https://github.com/ClickHouse/clickhouse-java/actions + + + + + central + Central Repository + https://repo.maven.apache.org/maven2 + + + + + + + + + + + + 2023 + UTF-8 + UTF-8 + + 0.6.0-SNAPSHOT + 5.2.1 + + 3.8.1 + + 1.8 + + + + + com.clickhouse + client-v2 + ${clickhouse-java.version} + + + com.clickhouse + clickhouse-http-client + ${clickhouse-java.version} + + + + + org.apache.httpcomponents.client5 + httpclient5 + ${apache-httpclient.version} + + + + + + + org.projectlombok + lombok + 1.18.32 + provided + + + + + + org.slf4j + slf4j-api + 2.0.13 + + + + org.slf4j + slf4j-simple + 2.0.13 + runtime + + + + + + + + org.apache.maven.plugins + maven-compiler-plugin + ${compiler-plugin.version} + + ${minJdk} + ${minJdk} + true + + -Xlint:all + + + + + org.projectlombok + lombok + 1.18.32 + + + + + + + + \ No newline at end of file diff --git a/examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/Main.java b/examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/Main.java new file mode 100644 index 000000000..c71f7798d --- /dev/null +++ b/examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/Main.java @@ -0,0 +1,43 @@ +package com.clickhouse.examples.client_v2; + +import lombok.extern.slf4j.Slf4j; + +import java.io.InputStream; + +@Slf4j +public class Main { + + public static void main(String[] args) { + final String endpoint = System.getProperty("chEndpoint", "http://localhost:8123"); + final String user = System.getProperty("chUser", "default"); + final String password = System.getProperty("chPassword", null); + final String database = System.getProperty("chDatabase", "default"); + + SimpleWriter writer = new SimpleWriter(endpoint, user, password, database); + + if (writer.isServerAlive()) { + log.info("ClickHouse server is alive"); + } else { + log.error("ClickHouse server is not alive"); + Runtime.getRuntime().exit(-503); + } + + writer.resetTable(); + + + log.info("Inserting data from resources/sample_hacker_news_posts.json"); + try (InputStream is = Main.class.getResourceAsStream("/sample_hacker_news_posts.json")) { + writer.insertData_JSONEachRowFormat(is); + } catch (Exception e) { + log.error("Failed to insert data", e); + } + + SimpleReader reader = new SimpleReader(endpoint, user, password, database); + + reader.readData(); + + log.info("Done"); + + Runtime.getRuntime().exit(0); + } +} diff --git a/examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/SimpleReader.java b/examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/SimpleReader.java new file mode 100644 index 000000000..43a7a414a --- /dev/null +++ b/examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/SimpleReader.java @@ -0,0 +1,68 @@ +package com.clickhouse.examples.client_v2; + +import com.clickhouse.client.api.Client; +import com.clickhouse.client.api.data_formats.ClickHouseBinaryFormatReader; +import com.clickhouse.client.api.data_formats.RowBinaryWithNamesAndTypesFormatReader; +import com.clickhouse.client.api.query.QueryResponse; +import com.clickhouse.client.api.query.QuerySettings; +import com.clickhouse.data.ClickHouseFormat; +import lombok.extern.slf4j.Slf4j; + +import java.util.Collections; +import java.util.concurrent.TimeUnit; + +@Slf4j +public class SimpleReader { + + private static final String TABLE_NAME = "hacker_news_articles"; + + Client client; + + String database; + + public SimpleReader(String endpoint, String user, String password, String database) { + // Create a lightweight object to interact with ClickHouse server + this.client = new Client.Builder() + .addEndpoint(endpoint) + .addUsername(user).addPassword(password) + .enableCompression(false) + .enableDecompression(false) + .build(); + this.database = database; + } + + public boolean isServerAlive() { + return client.ping(); + } + + public void readData() { + try { + // Read data from the table + log.info("Reading data from table: {}", TABLE_NAME); + QuerySettings settings = new QuerySettings(); + settings.setDatabase(database); + settings.setFormat(ClickHouseFormat.RowBinaryWithNamesAndTypes.name()); + settings.setCompress(false); + final String sql = "select * from " + TABLE_NAME + " where title <> '' limit 10"; + QueryResponse response = client.query(sql, Collections.emptyMap(), settings) + .get(10, TimeUnit.SECONDS); + response.ensureDone(); + + ClickHouseBinaryFormatReader reader = new RowBinaryWithNamesAndTypesFormatReader(response.getInputStream(), + settings); + + while (reader.next()) { + double id = reader.getDouble("id"); + String title = reader.getString("title"); + String url = reader.getString("url"); + + log.info("id: {}, title: {}, url: {}", id, title, url); + } + + log.info("Data read successfully: {}", response.getOperationStatistics()); + + } catch (Exception e) { + log.error("Failed to read data", e); + } + } +} diff --git a/examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/SimpleWriter.java b/examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/SimpleWriter.java new file mode 100644 index 000000000..26453d212 --- /dev/null +++ b/examples/client-v2/src/main/java/com/clickhouse/examples/client_v2/SimpleWriter.java @@ -0,0 +1,74 @@ +package com.clickhouse.examples.client_v2; + +import com.clickhouse.client.api.Client; +import com.clickhouse.client.api.exception.ClientException; +import com.clickhouse.client.api.insert.InsertResponse; +import com.clickhouse.client.api.insert.InsertSettings; +import com.clickhouse.client.api.query.QueryResponse; +import com.clickhouse.client.api.query.QuerySettings; +import com.clickhouse.data.ClickHouseFormat; +import lombok.extern.slf4j.Slf4j; + +import java.io.BufferedReader; +import java.io.InputStream; +import java.io.InputStreamReader; +import java.util.Collections; +import java.util.concurrent.TimeUnit; +import java.util.stream.Collectors; + +@Slf4j +public class SimpleWriter { + + private static final String TABLE_NAME = "hacker_news_articles"; + + Client client; + + String database; + + public SimpleWriter(String endpoint, String user, String password, String database) { + // Create a lightweight object to interact with ClickHouse server + this.client = new Client.Builder() + .addEndpoint(endpoint) + .addUsername(user).addPassword(password) + .enableCompression(false) + .enableDecompression(false) + .build(); + this.database = database; + } + + public boolean isServerAlive() { + return client.ping(); + } + + // Initializes a table for the specific dataset + public void resetTable() { + try (InputStream initSql = SimpleWriter.class.getResourceAsStream("/simple_writer_init.sql")) { + BufferedReader reader = new BufferedReader(new InputStreamReader(initSql)); + String sql = reader.lines().collect(Collectors.joining("\n")); + log.debug("Executing Create Table: {}", sql); + QuerySettings settings = new QuerySettings(); + settings.setDatabase(database); + settings.setFormat(ClickHouseFormat.TabSeparated.name()); + settings.setCompress(false); + client.query("drop table if exists hacker_news_articles", Collections.emptyMap(), settings).get(10, TimeUnit.SECONDS); + QueryResponse response = client.query(sql, Collections.emptyMap(), settings).get(10, TimeUnit.SECONDS); + response.ensureDone(); + log.info("Table initialized"); + } catch (Exception e) { + log.error("Failed to initialize table", e); + } + } + + public void insertData_JSONEachRowFormat(InputStream inputStream) { + InsertSettings insertSettings = new InsertSettings(); + insertSettings.setFormat(ClickHouseFormat.JSONEachRow); + + try { + InsertResponse response = client.insert(TABLE_NAME, inputStream, insertSettings); + log.info("Insert finished: {}", response.getOperationStatistics()); + } catch (Exception | ClientException e) { + log.error("Failed to write JSONEachRow data", e); + throw new RuntimeException(e); + } + } +} diff --git a/examples/client-v2/src/main/resources/sample_hacker_news_posts.json b/examples/client-v2/src/main/resources/sample_hacker_news_posts.json new file mode 100644 index 000000000..d29d80cea --- /dev/null +++ b/examples/client-v2/src/main/resources/sample_hacker_news_posts.json @@ -0,0 +1,10 @@ +{"id":11132929,"deleted":0,"type":"story","by":"brakmic","time":"2016-02-19 11:40:55","text":"","dead":1,"parent":0,"poll":0,"kids":[11133170],"url":"https:\/\/www.washingtonpost.com\/news\/morning-mix\/wp\/2016\/02\/18\/s-f-tech-bro-writes-open-letter-to-mayor-i-shouldnt-have-to-see-the-pain-struggle-and-despair-of-homeless-people\/","score":3,"title":"SF ‘tech bro’ writes to mayor: ‘I shouldn’t have to see the despair of homeless’","parts":[],"descendants":0} +{"id":11132930,"deleted":0,"type":"comment","by":"kherraza","time":"2016-02-19 11:41:07","text":"a new way to travel without moving through a new system!","dead":0,"parent":11132928,"poll":0,"kids":[],"url":"","score":0,"title":"","parts":[],"descendants":0} +{"id":11132931,"deleted":0,"type":"comment","by":"DanielBMarkham","time":"2016-02-19 11:41:50","text":"I don't want huge denominations, but assuming inflation over the last 50 years, seems to me that we should have a $500US note.

Yes, I understand if LE had their way we'd be paying with 5-spots, but convenience has to mean something also. Let's bring back the $500 bill and then index the entire thing to inflation such that there isn't more than a dozen or so steps between the smallest and largest monetary units.

I don't think this is a political discussion/argument we need to have constantly. Fix it once and then leave it alone. I'm also completely opposed to eliminating all currency and going totally electronic, which sounds like an even more dystopian state than the current one our betters have created for us over the last decade or two.","dead":0,"parent":11132443,"poll":0,"kids":[11132960],"url":"","score":0,"title":"","parts":[],"descendants":0} +{"id":11132932,"deleted":0,"type":"comment","by":"choosername","time":"2016-02-19 11:41:50","text":"> With math, there are no docs, no playground, no explanations, no nothing. Either you are the part of mathematical community and can hope for explanations and knowledge transfer, or you are not

what's the difference to CS here, where did you get that coding sample you mentioned? The playground is your mind, pebbles or pen and paper, maths are a projection of the real world.

Mathematics is not inconsistent. Isn't this the premisses of rigor? Math has mathematical consistency and that's not the same as literal consistency, but maybe that's a problem in semiotics and language. The language and the literature is inconsistent.","dead":0,"parent":11132614,"poll":0,"kids":[11132963],"url":"","score":0,"title":"","parts":[],"descendants":0} +{"id":11132933,"deleted":0,"type":"story","by":"mt_caret","time":"2016-02-19 11:42:32","text":"","dead":0,"parent":0,"poll":0,"kids":[],"url":"https:\/\/github.com\/php\/php-src\/commit\/a0724d30817600540946b41e40f4cfc2a0c30f80","score":3,"title":"Commit to revert to mistaken implementation of mt_rand()","parts":[],"descendants":0} +{"id":11132934,"deleted":0,"type":"story","by":"rayascott","time":"2016-02-19 11:42:34","text":"","dead":0,"parent":0,"poll":0,"kids":[],"url":"http:\/\/www.wired.com\/2015\/10\/this-living-clothing-morphs-when-you-sweat\/","score":1,"title":"This Living Clothing Morphs When You Sweat","parts":[],"descendants":0} +{"id":11132935,"deleted":0,"type":"comment","by":"refal","time":"2016-02-19 11:43:03","text":"I'm basically with you.

However, as a German, I find it interesting why you mention Germany as "less-principled". Less-principled than the US? Do you talk about current Germany or about former east Germany or even Nazi Germany?

Of course, there are enough problems here as anywhere else regarding privacy, etc. But I have difficulties seeing current Germany as "less-principled" than the US wrt. privacy, human rights etc.

Just curious what your impression is based on.","dead":0,"parent":11132330,"poll":0,"kids":[],"url":"","score":0,"title":"","parts":[],"descendants":0} +{"id":11132936,"deleted":0,"type":"comment","by":"rwmj","time":"2016-02-19 11:43:32","text":"Against an insufficiently smart attacker. How about a computer that can learn to imitate any speech, perhaps by modelling the shape of the vocal tract (which is how modern speech synthesizers actually work, so it's not a huge leap).","dead":0,"parent":11132918,"poll":0,"kids":[11133186,11132982],"url":"","score":0,"title":"","parts":[],"descendants":0} +{"id":11132937,"deleted":0,"type":"comment","by":"noir_lord","time":"2016-02-19 11:43:56","text":"Except the note size is kind of secondary here.

Having limits on transactions over 1000 euro (like France) does forces law-abiding citizens to handle transactions in a way that can be monitored by the security services (the non-law abiding citizens of course just ignore the damn law or wash the money through a front).

It's yet another attack on privacy, if I want to buy a bike from someone legally and pay in cash for it what business is it of the governments, it's a private transaction between two parties for a legally owned and sold item.

I'm rapidly coming to the view that Orwell wasn't far of the mark with "If you want a vision of the future, imagine a boot stamping on a human face - forever." except instead of an actual jackboot we just have the complete removal of privacy, fear of the boot stamping on your face instead of an actual boot.","dead":0,"parent":11132917,"poll":0,"kids":[],"url":"","score":0,"title":"","parts":[],"descendants":0} +{"id":11132938,"deleted":0,"type":"story","by":"vickymehta","time":"2016-02-19 11:44:07","text":"http://goo.gl/tW7F2j","dead":1,"parent":0,"poll":0,"kids":[],"url":"","score":1,"title":"Letting You Online Stores to Find the Best Brand for You and Your Family","parts":[],"descendants":0} diff --git a/examples/client-v2/src/main/resources/simple_writer_init.sql b/examples/client-v2/src/main/resources/simple_writer_init.sql new file mode 100644 index 000000000..209fbd5e7 --- /dev/null +++ b/examples/client-v2/src/main/resources/simple_writer_init.sql @@ -0,0 +1,17 @@ +create table hacker_news_articles ( +id Nullable(Float64), +deleted Nullable(Float64), +type Nullable(String), +by Nullable(String), +time Nullable(String), +text Nullable(String), +dead Nullable(Float64), +parent Nullable(Float64), +poll Nullable(Float64), +kids Array(Nullable(Float64)), +url Nullable(String), +score Nullable(Float64), +title Nullable(String), +parts Array(Nullable(Float64)), +descendants Nullable(Float64) +) engine = MergeTree order by (); \ No newline at end of file