All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.clickhouse.jdbc.Main Maven / Gradle / Ivy

There is a newer version: 0.7.1-patch1
Show newest version
package com.clickhouse.jdbc;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.time.LocalDateTime;
import java.util.Arrays;
import java.util.List;
import java.util.Locale;
import java.util.Objects;

import com.clickhouse.client.ClickHouseClient;
import com.clickhouse.client.ClickHouseConfig;
import com.clickhouse.client.ClickHouseException;
import com.clickhouse.client.ClickHouseNode;
import com.clickhouse.client.ClickHouseRequest;
import com.clickhouse.client.ClickHouseResponse;
import com.clickhouse.client.ClickHouseRequest.Mutation;
import com.clickhouse.client.config.ClickHouseClientOption;
import com.clickhouse.data.ClickHouseByteBuffer;
import com.clickhouse.data.ClickHouseColumn;
import com.clickhouse.data.ClickHouseDataProcessor;
import com.clickhouse.data.ClickHouseDataStreamFactory;
import com.clickhouse.data.ClickHouseDataType;
import com.clickhouse.data.ClickHouseDeserializer;
import com.clickhouse.data.ClickHouseFormat;
import com.clickhouse.data.ClickHouseInputStream;
import com.clickhouse.data.ClickHouseOutputStream;
import com.clickhouse.data.ClickHouseRecord;
import com.clickhouse.data.ClickHouseSerializer;
import com.clickhouse.data.ClickHouseValue;
import com.clickhouse.data.format.BinaryStreamUtils;
import com.clickhouse.data.value.ClickHouseByteValue;
import com.clickhouse.data.value.ClickHouseLongValue;
import com.clickhouse.data.value.ClickHouseStringValue;
import com.clickhouse.jdbc.internal.ClickHouseConnectionImpl;

public final class Main {
    public static class Pojo {
        private byte b;
        private long l;
        private BigDecimal d;
        private LocalDateTime t;
        private long[] a;
        private List p;
        private Object[][] n;
        private Object j;

        public void setByte(byte b) {
            this.b = b;
        }

        public byte getByte() {
            return b;
        }

        public void setLong(long l) {
            this.l = l;
        }

        public long getLong() {
            return l;
        }

        public void setDecimal(BigDecimal d) {
            this.d = d;
        }

        public BigDecimal getDecimal() {
            return d;
        }

        public void setDateTime(LocalDateTime t) {
            this.t = t;
        }

        public LocalDateTime getDateTime() {
            return t;
        }

        public void setArray(long[] a) {
            this.a = a;
        }

        public long[] getArray() {
            return a;
        }

        public void setTuple(List p) {
            this.p = p;
        }

        public List getTuple() {
            return p;
        }

        public void setNested(Object[][] n) {
            this.n = n;
        }

        public Object[][] getNested() {
            return n;
        }

        public void setJson(Object j) {
            this.j = j;
        }

        public Object getJson() {
            return j;
        }
    }

    static class Options {
        final String action;
        final int batch;
        final boolean mapping;
        final boolean output;
        final int samples;
        final boolean serde;
        final String type;
        final boolean verbose;

        final String url;
        final String query;
        final String file;

        final boolean requiresJdbc;

        private Options(String url, String query, String file) {
            action = System.getProperty("action", "read").toLowerCase();
            batch = Integer.getInteger("batch", 1000);
            mapping = Boolean.getBoolean("mapping");
            output = Boolean.getBoolean("output");
            samples = Integer.getInteger("samples", 500000000);
            serde = !"false".equalsIgnoreCase(System.getProperty("serde", ""));
            type = System.getProperty("type", "").toLowerCase();
            verbose = Boolean.getBoolean("verbose");

            this.url = url;
            this.requiresJdbc = url.length() > 5 && "jdbc:".equalsIgnoreCase(url.substring(0, 5));

            if (query == null || query.isEmpty()) {
                this.query = isLoadAction() || isWriteAction() ? getInsertQuery() : getSelectQuery();
            } else {
                this.query = query;
            }
            if (file == null || file.isEmpty()) {
                if (output) {
                    this.file = requiresJdbc ? "jdbc.out" : "java.out";
                } else {
                    this.file = "";
                }
            } else {
                this.file = file;
            }

            if (verbose) {
                println("Arguments:");
                println("  -   url=%s", this.url);
                println("  - query=%s", this.query);
                println("  -  file=%s", this.file);
                println();
                println("Options:\n  - action=%s, batch=%d, mapping=%s,\n  - output=%s, samples=%d, serde=%s, type=%s",
                        action, batch, mapping, output, samples, serde, type);
            }
        }

        int getSamples() {
            // final int s;
            // if (isMixed() || isTuple() || isNested()) {
            // s = samples / 5;
            // } else if (isArray()) {
            // s = samples / 1000;
            // } else if (isJson()) {
            // s = samples / 500;
            // } else {
            // s = samples;
            // }
            // return s;
            return samples;
        }

        boolean hasFile() {
            return !file.isEmpty();
        }

        boolean hasMapping() {
            return mapping;
        }

        boolean isDumpAction() {
            return "dump".equals(action);
        }

        boolean isLoadAction() {
            return "load".equals(action);
        }

        boolean isWriteAction() {
            return "write".equals(action);
        }

        boolean isInt8() {
            return "int8".equals(type);
        }

        boolean isUInt64() {
            return "uint64".equals(type);
        }

        boolean isString() {
            return "string".equals(type);
        }

        boolean isDateTime() {
            return "datetime".equals(type);
        }

        boolean isDecimal() {
            return "decimal".equals(type);
        }

        boolean isMixed() {
            return "mixed".equals(type);
        }

        boolean isArray() {
            return "array".equals(type);
        }

        boolean isTuple() {
            return "tuple".equals(type);
        }

        boolean isNested() {
            return "nested".equals(type);
        }

        boolean isJson() {
            return "json".equals(type);
        }

        List getColumns() {
            final List columns;
            if (isInt8()) {
                columns = Arrays.asList(ClickHouseColumn.of(null, ClickHouseDataType.Int8, false));
            } else if (isUInt64()) {
                columns = Arrays.asList(ClickHouseColumn.of(null, ClickHouseDataType.UInt64, false));
            } else if (isString()) {
                columns = Arrays.asList(ClickHouseColumn.of(null, ClickHouseDataType.String, false));
            } else if (isDateTime()) {
                columns = Arrays.asList(ClickHouseColumn.of(null, ClickHouseDataType.DateTime, false));
            } else if (isDecimal()) {
                columns = Arrays.asList(ClickHouseColumn.of(null, ClickHouseDataType.Decimal128, false, 0, 6));
            } else if (isMixed()) {
                columns = Arrays.asList(ClickHouseColumn.of(null, ClickHouseDataType.Int8, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.UInt64, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.String, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.DateTime, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.Decimal128, false, 0, 6));
            } else if (isArray()) {
                columns = Arrays.asList(ClickHouseColumn.of(null, "Array(Int32)"));
            } else if (isTuple()) {
                columns = Arrays.asList(ClickHouseColumn.of(null, ClickHouseDataType.Tuple, false,
                        ClickHouseColumn.of(null, ClickHouseDataType.Int8, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.UInt64, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.String, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.DateTime, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.Decimal128, false, 0, 6)));
            } else if (isNested()) {
                columns = Arrays.asList(ClickHouseColumn.of(null, ClickHouseDataType.Nested, false,
                        ClickHouseColumn.of(null, ClickHouseDataType.Int8, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.UInt64, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.String, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.DateTime, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.Decimal128, false, 0, 6)));
            } else if (isJson()) {
                columns = Arrays.asList(ClickHouseColumn.of(null, ClickHouseDataType.Tuple, false,
                        ClickHouseColumn.of(null, ClickHouseDataType.Int8, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.UInt64, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.String, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.DateTime, false),
                        ClickHouseColumn.of(null, ClickHouseDataType.Decimal128, false, 0, 6)));
            } else {
                columns = null;
            }
            return columns;
        }

        ClickHouseDeserializer getDerializer(ClickHouseConfig config) throws IOException {
            final List columns = getColumns();
            if (columns == null || columns.isEmpty()) {
                throw new IllegalStateException("Not column information available for query: " + query);
            }

            final ClickHouseDataProcessor processor = ClickHouseDataStreamFactory.getInstance().getProcessor(config,
                    null, ClickHouseOutputStream.empty(), null, columns);
            final ClickHouseDeserializer[] deserializers = processor.getDeserializers(config, columns);
            return deserializers.length == 1 ? deserializers[0]
                    : ClickHouseDeserializer.of(Arrays.asList(deserializers));
        }

        ClickHouseSerializer getSerializer(ClickHouseConfig config) throws IOException {
            final ClickHouseSerializer[] serializers = getSerializers(config);
            return serializers.length == 1 ? serializers[0] : ClickHouseSerializer.of(Arrays.asList(serializers));
        }

        ClickHouseSerializer[] getSerializers(ClickHouseConfig config) throws IOException {
            final List columns = getColumns();
            if (columns == null || columns.isEmpty()) {
                throw new IllegalStateException("Not column information available for query: " + query);
            }

            final ClickHouseDataProcessor processor = ClickHouseDataStreamFactory.getInstance().getProcessor(config,
                    null, ClickHouseOutputStream.empty(), null, columns);
            return processor.getSerializers(config, columns);
        }

        String getSelectQuery() {
            final String selectQuery;
            if (isInt8()) {
                selectQuery = "select number::Int8 `byte` from numbers(%d)";
            } else if (isUInt64()) {
                selectQuery = "select number `long` from numbers(%d)";
            } else if (isString()) {
                selectQuery = "select toString(number) `string` from numbers(%d)";
            } else if (isDateTime()) {
                selectQuery = "select toDateTime(number) `datetime` from numbers(%d)";
            } else if (isDecimal()) {
                selectQuery = "select toDecimal128(number, 6) `decimal` from numbers(%d)";
            } else if (isMixed()) {
                selectQuery = "select number::Int8 `byte`, number `long`, toString(number) `string`, toDateTime(number) `datetime`, toDecimal128(number, 6) `decimal` from numbers(%d)";
            } else if (isArray()) {
                selectQuery = "select range(100000, 101000 + number %% 1000) as `array` from numbers(%d)";
            } else if (isTuple()) {
                selectQuery = "select tuple(number::Int8, number, toString(number), toDateTime(number), toDecimal128(number, 6)) `tuple` from numbers(%d)";
            } else if (isNested()) {
                selectQuery = "select [(number::Int8, number, toString(number), toDateTime(number), toDecimal128(number, 6))]::Nested(a Int8, b UInt64, c String, d DateTime, e Decimal128(6)) `nested` from numbers(%d)";
            } else if (isJson()) {
                selectQuery = "select (number::Int8, number, toString(number), toDateTime(number), toDecimal128(number, 6), range(1000,1005), [tuple(number, number+1)])::Tuple(a Int8, b UInt64, c String, d DateTime, e Decimal128(6), f Array(UInt16), g Nested(x UInt64, y UInt64)) `json` from numbers(%d)";
            } else {
                selectQuery = "select %d";
            }
            return String.format(selectQuery, getSamples());
        }

        String getInsertQuery() {
            return type.isEmpty() ? "insert into test_insert" : "insert into test_insert_" + type;
        }
    }

    static class GenericQuery {
        static final ClickHouseFormat defaultFormat = ClickHouseFormat.RowBinaryWithNamesAndTypes;

        protected final Options options;

        protected GenericQuery(Options options) {
            this.options = options;
        }

        final long run() throws ClickHouseException, SQLException {
            final long rows;
            if (options.isDumpAction()) {
                rows = dump();
            } else if (options.isLoadAction()) {
                rows = load(options);
            } else if (options.isWriteAction()) {
                rows = write(options);
            } else {
                rows = read(options);
            }
            return rows;
        }

        long read(ResultSet rs) throws SQLException {
            long count = 0L;
            final int len = rs.getMetaData().getColumnCount();
            while (rs.next()) {
                Object obj = null;
                for (int i = 1; i <= len; i++) {
                    // autoboxing to ensure we "got" the value
                    obj = rs.getObject(i);
                }
                if (obj != null) {
                    count++;
                }
            }
            return count;
        }

        long read(ClickHouseResponse response) throws ClickHouseException {
            long count = 0L;
            int len = response.getColumns().size();
            for (ClickHouseRecord r : response.records()) {
                Object obj = null;
                for (int i = 0; i < len; i++) {
                    // autoboxing just for comparison
                    obj = r.getValue(i).asObject();
                }
                if (obj != null) {
                    count++;
                }
            }
            return count;
        }

        long write(Connection conn) throws SQLException {
            throw new UnsupportedOperationException("No idea how to write data for custom query");
        }

        long write(Mutation request) throws ClickHouseException {
            throw new UnsupportedOperationException("No idea how to write data for custom query");
        }

        final long dump() throws ClickHouseException, SQLException {
            final long rows;
            if (options.requiresJdbc) {
                try (ClickHouseConnection conn = new ClickHouseConnectionImpl(options.url)) {
                    ClickHouseRequest request = conn.unwrap(ClickHouseRequest.class).query(options.query);
                    if (!request.getServer().getConfig().hasOption(ClickHouseClientOption.FORMAT)) {
                        request.format(defaultFormat);
                    }
                    request.output(options.file);
                    try (ClickHouseResponse response = request.executeAndWait()) {
                        rows = response.getSummary().getReadRows();
                    }
                }
            } else { // java client
                final ClickHouseNode server = ClickHouseNode.of(options.url);
                try (ClickHouseClient client = ClickHouseClient.newInstance(server.getProtocol())) {
                    ClickHouseRequest request = client.read(server).query(options.query);
                    if (!server.getConfig().hasOption(ClickHouseClientOption.FORMAT)) {
                        request.format(defaultFormat);
                    }
                    request.output(options.file);
                    try (ClickHouseResponse response = request.query(options.query).executeAndWait()) {
                        rows = response.getSummary().getReadRows();
                    }
                }
            }
            return rows;
        }

        final long load(Options options) throws ClickHouseException, SQLException {
            final long rows;
            if (options.requiresJdbc) {
                try (ClickHouseConnection conn = new ClickHouseConnectionImpl(options.url)) {
                    ClickHouseFormat format = conn.getConfig().getFormat();
                    if (!conn.unwrap(ClickHouseRequest.class).getServer().getConfig()
                            .hasOption(ClickHouseClientOption.FORMAT)) {
                        format = defaultFormat.defaultInputFormat();
                    }
                    try (PreparedStatement stmt = conn.prepareStatement(options.query + " format " + format.name())) { // NOSONAR
                        stmt.setObject(1, new File(options.file));
                        rows = stmt.executeLargeUpdate();
                    }
                }
            } else { // java client
                final ClickHouseNode server = ClickHouseNode.of(options.url);
                try (ClickHouseClient client = ClickHouseClient.newInstance(server.getProtocol())) {
                    Mutation request = client.write(server).data(options.file);
                    if (!server.getConfig().hasOption(ClickHouseClientOption.FORMAT)) {
                        request.format(defaultFormat.defaultInputFormat());
                    }
                    try (ClickHouseResponse response = request
                            .query(options.query + " format " + request.getConfig().getFormat().name())
                            .executeAndWait()) {
                        rows = response.getSummary().getWrittenRows();
                    }
                }
            }
            return rows;
        }

        final long read(Options options) throws ClickHouseException, SQLException {
            final long rows;
            if (options.requiresJdbc) {
                try (ClickHouseConnection conn = new ClickHouseConnectionImpl(options.url);
                        ClickHouseStatement stmt = conn.createStatement()) {
                    if (options.hasFile()) {
                        try {
                            stmt.setMirroredOutput(
                                    !"-".equals(options.file) ? new FileOutputStream(options.file, false) : System.out); // NOSONAR
                        } catch (IOException e) {
                            throw SqlExceptionUtils.clientError(e);
                        }
                    }
                    try (ResultSet rs = stmt.executeQuery(options.query)) {
                        rows = read(rs);
                    }
                }
            } else { // java client
                final ClickHouseNode server = ClickHouseNode.of(options.url);
                try (ClickHouseClient client = ClickHouseClient.newInstance(server.getProtocol())) {
                    ClickHouseRequest request = client.read(server).query(options.query);
                    if (!server.getConfig().hasOption(ClickHouseClientOption.FORMAT)) {
                        request.format(defaultFormat);
                    }
                    try (ClickHouseResponse response = request.executeAndWait()) {
                        if (options.hasFile()) {
                            try {
                                response.getInputStream().setCopyToTarget(
                                        !"-".equals(options.file) ? new FileOutputStream(options.file, false) // NOSONAR
                                                : System.out); // NOSONAR
                            } catch (IOException e) {
                                throw ClickHouseException.of(e, server);
                            }
                        }

                        if (options.hasMapping()) {
                            long count = 0L;
                            for (Pojo p : response.records(Pojo.class)) {
                                if (p != null) {
                                    count++;
                                }
                            }
                            rows = count;
                        } else {
                            rows = read(response);
                        }
                    }
                }
            }
            return rows;
        }

        final long write(Options options) throws ClickHouseException, SQLException {
            final long rows;
            if (options.requiresJdbc) {
                try (ClickHouseConnection conn = new ClickHouseConnectionImpl(options.url)) {
                    rows = write(conn);
                }
            } else { // java client
                final ClickHouseNode server = ClickHouseNode.of(options.url);
                try (ClickHouseClient client = ClickHouseClient.newInstance(server.getProtocol())) {
                    Mutation request = client.write(server).query(options.query).data(options.file);
                    if (!server.getConfig().hasOption(ClickHouseClientOption.FORMAT)) {
                        request.format(defaultFormat.defaultInputFormat());
                    }
                    rows = write(request);
                }
            }
            return rows;
        }
    }

    static class Int8Query extends GenericQuery {
        Int8Query(Options options) {
            super(options);
        }

        @Override
        long read(ResultSet rs) throws SQLException {
            long count = 0L;
            final int len = rs.getMetaData().getColumnCount();
            byte v = (byte) 0;
            while (rs.next()) {
                for (int i = 1; i <= len; i++) {
                    v = rs.getByte(i);
                }
                count++;
            }
            long lastValue = 0xFFL & v;
            return count >= lastValue ? count : lastValue;
        }

        @Override
        long read(ClickHouseResponse response) throws ClickHouseException {
            long count = 0L;
            byte v = (byte) 0;
            if (options.serde) {
                if (options.verbose) {
                    println("Deserialization: records");
                }
                for (ClickHouseRecord r : response.records()) {
                    // only one column
                    v = r.getValue(0).asByte();
                    count++;
                }
            } else {
                if (options.verbose) {
                    println("Deserialization: readByte");
                }
                try (ClickHouseInputStream in = response.getInputStream()) {
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        v = in.readByte();
                        count++;
                    }
                } catch (IOException e) {
                    throw new UncheckedIOException(e);
                }
            }
            long lastValue = 0xFFL & v;
            return count >= lastValue ? count : lastValue;
        }

        @Override
        long write(Connection conn) throws SQLException {
            try (PreparedStatement stmt = conn.prepareStatement(options.query)) {
                final int batchSize = options.batch;
                long count = 0L;
                long rows = 0L;
                for (long i = 0, len = options.getSamples(); i < len; i++) {
                    stmt.setByte(1, (byte) i);
                    stmt.addBatch();
                    if ((count = (i + 1) % batchSize) == 0L) {
                        rows += stmt.executeLargeBatch().length;
                    }
                }
                if (count > 0L) {
                    rows += stmt.executeLargeBatch().length;
                }
                return rows;
            }
        }

        @Override
        long write(Mutation request) throws ClickHouseException {
            try (ClickHouseResponse response = request.data(o -> {
                if (options.serde) {
                    ClickHouseConfig config = request.getConfig();
                    ClickHouseSerializer serializer = options.getSerializer(config);
                    ClickHouseValue value = ClickHouseByteValue.ofNull();
                    if (options.verbose) {
                        println("Serialization: %s -> %s", serializer, value);
                    }
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        serializer.serialize(value.update(i), o);
                    }
                } else {
                    if (options.verbose) {
                        println("Serialization: writeByte");
                    }
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        o.writeByte((byte) i);
                    }
                }
            }).executeAndWait()) {
                return response.getSummary().getWrittenRows();
            }
        }
    }

    static class UInt64Query extends GenericQuery {
        UInt64Query(Options options) {
            super(options);
        }

        @Override
        long read(ResultSet rs) throws SQLException {
            long count = 0L;
            final int len = rs.getMetaData().getColumnCount();
            long v = 0L;
            while (rs.next()) {
                for (int i = 1; i <= len; i++) {
                    v = rs.getLong(i);
                }
                count++;
            }
            return count >= v ? count : v;
        }

        @Override
        long read(ClickHouseResponse response) throws ClickHouseException {
            long count = 0L;
            long v = 0L;
            if (options.serde) {
                if (options.verbose) {
                    println("Deserialization: records");
                }
                for (ClickHouseRecord r : response.records()) {
                    // only one column
                    v = r.getValue(0).asLong();
                    count++;
                }
            } else {
                if (options.verbose) {
                    println("Deserialization: readByte");
                }
                try (ClickHouseInputStream in = response.getInputStream()) {
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        v = in.readBuffer(8).asLong();
                        count++;
                    }
                } catch (IOException e) {
                    throw new UncheckedIOException(e);
                }
            }
            return count >= v ? count : v;
        }

        @Override
        long write(Connection conn) throws SQLException {
            try (PreparedStatement stmt = conn.prepareStatement(options.query)) {
                final int batchSize = options.batch;
                long count = 0L;
                long rows = 0L;
                for (long i = 0, len = options.getSamples(); i < len; i++) {
                    stmt.setLong(1, i);
                    stmt.addBatch();
                    if ((count = (i + 1) % batchSize) == 0L) {
                        rows += stmt.executeLargeBatch().length;
                    }
                }
                if (count > 0L) {
                    rows += stmt.executeLargeBatch().length;
                }
                return rows;
            }
        }

        @Override
        long write(Mutation request) throws ClickHouseException {
            try (ClickHouseResponse response = request.data(o -> {
                if (options.serde) {
                    ClickHouseConfig config = request.getConfig();
                    ClickHouseSerializer serializer = options.getSerializer(config);
                    ClickHouseValue value = ClickHouseLongValue.ofUnsignedNull();
                    if (options.verbose) {
                        println("Serialization: %s -> %s", serializer, value);
                    }
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        serializer.serialize(value.update(i), o);
                    }
                } else {
                    if (options.verbose) {
                        println("Serialization: writeLong");
                    }
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        BinaryStreamUtils.writeUnsignedInt64(o, i);
                    }
                }
            }).executeAndWait()) {
                return response.getSummary().getWrittenRows();
            }
        }
    }

    static class StringQuery extends GenericQuery {
        StringQuery(Options options) {
            super(options);
        }

        @Override
        long read(ResultSet rs) throws SQLException {
            long count = 0L;
            final int len = rs.getMetaData().getColumnCount();
            String v = null;
            while (rs.next()) {
                for (int i = 1; i <= len; i++) {
                    v = rs.getString(i);
                }
                count++;
            }
            return v != null ? count : 0L;
        }

        @Override
        long read(ClickHouseResponse response) throws ClickHouseException {
            long count = 0L;
            String v = null;
            if (options.serde) {
                if (options.verbose) {
                    println("Deserialization: records");
                }
                for (ClickHouseRecord r : response.records()) {
                    // only one column
                    v = r.getValue(0).asString();
                    count++;
                }
            } else {
                if (options.verbose) {
                    println("Deserialization: readByte");
                }
                try (ClickHouseInputStream in = response.getInputStream()) {
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        v = in.readUnicodeString();
                        count++;
                    }
                } catch (IOException e) {
                    throw new UncheckedIOException(e);
                }
            }
            return v != null ? count : 0L;
        }

        @Override
        long write(Connection conn) throws SQLException {
            try (PreparedStatement stmt = conn.prepareStatement(options.query)) {
                final int batchSize = options.batch;
                long count = 0L;
                long rows = 0L;
                for (long i = 0, len = options.getSamples(); i < len; i++) {
                    stmt.setString(1, Long.toString(i));
                    stmt.addBatch();
                    if ((count = (i + 1) % batchSize) == 0L) {
                        rows += stmt.executeLargeBatch().length;
                    }
                }
                if (count > 0L) {
                    rows += stmt.executeLargeBatch().length;
                }
                return rows;
            }
        }

        @Override
        long write(Mutation request) throws ClickHouseException {
            try (ClickHouseResponse response = request.data(o -> {
                if (options.serde) {
                    ClickHouseConfig config = request.getConfig();
                    ClickHouseSerializer serializer = options.getSerializer(config);
                    ClickHouseValue value = ClickHouseStringValue.ofNull();
                    if (options.verbose) {
                        println("Serialization: %s -> %s", serializer, value);
                    }
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        serializer.serialize(value.update(i), o);
                    }
                } else {
                    if (options.verbose) {
                        println("Serialization: writeString");
                    }
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        o.writeUnicodeString(Long.toString(i));
                    }
                }
            }).executeAndWait()) {
                return response.getSummary().getWrittenRows();
            }
        }
    }

    static class MixedQuery extends GenericQuery {
        MixedQuery(Options options) {
            super(options);
        }

        @Override
        long read(ResultSet rs) throws SQLException {
            long count = 0L;
            byte b = (byte) 0;
            long l = 0L;
            String s = null;
            Object t = null;
            BigDecimal d = null;
            while (rs.next()) {
                b = rs.getByte(1);
                l = rs.getLong(2);
                s = rs.getString(3);
                t = rs.getObject(4);
                d = rs.getBigDecimal(5);
                count++;
            }
            return l > b && l > 0L && s != null && t != null && d != null ? count : 0L;
        }

        @Override
        long read(ClickHouseResponse response) throws ClickHouseException {
            long count = 0L;
            byte b = (byte) 0;
            long l = 0L;
            String s = null;
            Object t = null;
            BigDecimal d = null;
            if (options.serde) {
                if (options.verbose) {
                    println("Deserialization: records");
                }
                for (ClickHouseRecord r : response.records()) {
                    b = r.getValue(0).asByte();
                    l = r.getValue(1).asLong();
                    s = r.getValue(2).asString();
                    t = r.getValue(3).asDateTime();
                    d = r.getValue(4).asBigDecimal();
                    count++;
                }
                if (l > b && l > 0L && t != null && d != null) {
                    // ignore
                } else {
                    s = null;
                }
            } else {
                if (options.verbose) {
                    println("Deserialization: read(Byte, Long, String, DateTime, Decimal)");
                }
                try (ClickHouseInputStream in = response.getInputStream()) {
                    response.getColumns();
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        b = in.readByte();
                        l = in.readBuffer(8).asLong();
                        s = in.readUnicodeString();
                        t = in.readBuffer(4).asDateTime();
                        d = in.readBuffer(16).asBigDecimal(6);
                        count++;
                    }
                    if (l > b && l > 0L && t != null && d != null) {
                        // ignore
                    } else {
                        s = null;
                    }
                } catch (IOException e) {
                    throw new UncheckedIOException(e);
                }
            }
            return s != null ? count : 0L;
        }

        @Override
        long write(Connection conn) throws SQLException {
            try (PreparedStatement stmt = conn.prepareStatement(options.query)) {
                final int batchSize = options.batch;
                long count = 0L;
                long rows = 0L;
                for (long i = 0, len = options.getSamples(); i < len; i++) {
                    stmt.setByte(1, (byte) (i % 256));
                    stmt.setLong(2, i);
                    stmt.setString(3, Long.toString(i));
                    stmt.setLong(4, i);
                    stmt.setLong(5, i);
                    stmt.addBatch();
                    if ((count = (i + 1) % batchSize) == 0L) {
                        rows += stmt.executeLargeBatch().length;
                    }
                }
                if (count > 0L) {
                    rows += stmt.executeLargeBatch().length;
                }
                return rows;
            }
        }

        @Override
        long write(Mutation request) throws ClickHouseException {
            try (ClickHouseResponse response = request.data(o -> {
                if (options.serde) {
                    ClickHouseConfig config = request.getConfig();
                    ClickHouseSerializer[] serializers = options.getSerializers(config);
                    ClickHouseValue value = ClickHouseLongValue.ofNull();
                    if (options.verbose) {
                        println("Serialization: %s -> %s", serializers, value);
                    }
                    for (long i = 0L, len = options.samples, l = serializers.length; i < len; i++) {
                        for (int j = 0; j < l; j++) {
                            serializers[j].serialize(value.update(i), o);
                        }
                    }
                } else {
                    if (options.verbose) {
                        println("Serialization: read(Byte, Long, String, DateTime, Decimal)");
                    }
                    for (long i = 0L, len = options.samples; i < len; i++) {
                        o.writeByte((byte) (i % 256));
                        BinaryStreamUtils.writeUnsignedInt64(o, i);
                        o.writeUnicodeString(Long.toString(i));
                        BinaryStreamUtils.writeUnsignedInt32(o, i);
                        BinaryStreamUtils.writeInt128(o, BigInteger.valueOf(i));
                    }
                }
            }).executeAndWait()) {
                return response.getSummary().getWrittenRows();
            }
        }
    }

    private static void println() {
        System.out.println(); // NOSONAR
    }

    private static void println(Object msg, Object... args) {
        if (args == null || args.length == 0) {
            System.out.println(msg); // NOSONAR
        } else {
            System.out.println(String.format(Locale.ROOT, Objects.toString(msg), args)); // NOSONAR
        }
    }

    private static void printUsage() {
        String execFile = "clickhouse-jdbc-bin";
        try {
            File file = Paths.get(Main.class.getProtectionDomain().getCodeSource().getLocation().toURI())
                    .toFile();
            if (file.isFile()) {
                execFile = file.getName();
                if (!Files.isExecutable(file.toPath())) {
                    execFile = "java -jar " + execFile;
                }
            } else {
                execFile = "java -cp " + file.getCanonicalPath() + " " + Main.class.getName();
            }
        } catch (Exception e) {
            // ignore
        }

        final int index = execFile.indexOf(' ');
        println("Usage: %s  [QUERY] [FILE]",
                index > 0 ? (execFile.substring(0, index) + " [PROPERTIES]" + execFile.substring(index))
                        : (execFile + " [PROPERTIES]"));
        println();
        println("Properties: -Dkey=value [-Dkey=value]*");
        println("  action \tAction, one of read(default), write, dump(no deserialization), and load(no serialization)");
        println("  batch  \tBatch size for JDBC writing, defaults to 1000");
        println("  mapping\tWhether to map record into POJO, defaults to false");
        println("  output \tWhether to write raw response into stdout or a file(java.out or jdbc.out), defaults to false");
        println("  samples\tSamples, defaults to 500000000");
        println("  serde  \tWhether to use default serialization/deserializion mechanism in Java client, defaults to true");
        println("  type   \tPredefined QUERY, one of Int8, UInt64, String, Array, Tuple, Nested, and Mixed");
        println("  verbose\tWhether to show logs, defaults to false");
        println();
        println("Examples:");
        println("  -  %s 'https://localhost?sslmode=none' 'select 1' -",
                index > 0 ? (execFile.substring(0, index) + " -Dverbose=true" + execFile.substring(index))
                        : (execFile + " -Dverbose=true"));
        println("  -  %s 'jdbc:ch://user:password@localhost:8123/default' 'select 1' output.file", execFile);
        println("  -  %s 'jdbc:ch:http://node1,node2,node3/default' 'insert into table1' input.file", execFile);
    }

    public static void main(String[] args) throws Exception {
        if ((args == null || args.length < 1) || args.length > 3) {
            printUsage();
            System.exit(0);
        }

        final Options options = new Options(args[0].trim(), args.length > 1 ? args[1].trim() : null,
                args.length > 2 ? args[2].trim() : null);

        final GenericQuery query;
        if (options.isInt8()) {
            query = new Int8Query(options);
        } else if (options.isUInt64()) {
            query = new UInt64Query(options);
        } else if (options.isString()) {
            query = new StringQuery(options);
        } else if (options.isMixed()) {
            query = new MixedQuery(options);
        } else {
            query = new GenericQuery(options);
        }

        final long startTime = options.verbose ? System.nanoTime() : 0L;
        final long rows = query.run();
        if (options.verbose) {
            long elapsedNanos = System.nanoTime() - startTime;
            println("\nProcessed %,d rows in %,.2f ms (%,.2f rows/s)", rows, elapsedNanos / 1_000_000D,
                    rows * 1_000_000_000D / elapsedNanos);
        }
        System.exit(rows > 0L ? 0 : 1);
    }

    private Main() {
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy