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

io.nats.streaming.examples.StanBench Maven / Gradle / Ivy

// Copyright 2015-2018 The NATS Authors
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at:
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package io.nats.streaming.examples;

import io.nats.streaming.examples.benchmark.Benchmark;
import io.nats.streaming.examples.benchmark.Sample;
import io.nats.client.Connection;
import io.nats.client.Consumer;
import io.nats.client.ErrorListener;
import io.nats.client.NUID;
import io.nats.client.Nats;
import io.nats.streaming.AckHandler;
import io.nats.streaming.Message;
import io.nats.streaming.MessageHandler;
import io.nats.streaming.NatsStreaming;
import io.nats.streaming.Options;
import io.nats.streaming.StreamingConnection;
import io.nats.streaming.Subscription;
import io.nats.streaming.SubscriptionOptions;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.Charset;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Properties;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Phaser;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * A utility class for measuring NATS performance.
 *
 */
public class StanBench {

    // Default test values
    private int numMsgs = 100000;
    private int numPubs = 1;
    private int numSubs = 0;
    private boolean async = false;
    private int size = 128;
    private boolean ignoreOld = false;
    private int maxPubAcksInFlight = 1000;
    private String clientId = "benchmark";
    private String clusterId = "test-cluster";

    private String urls = io.nats.client.Options.DEFAULT_URL;
    private String subject;
    private final AtomicInteger published = new AtomicInteger();
    private final AtomicInteger received = new AtomicInteger();
    private String csvFileName;

    private io.nats.client.Options natsOptions;

    private Thread shutdownHook;
    private final AtomicBoolean shutdown = new AtomicBoolean(false);

    private boolean secure;
    private Benchmark bench;

    private static final String usageString =
            "\nUsage: nats-bench [-s server] [--tls] [-c clusterid] [-id clientid] [-np #pubs] "
                    + "[-ns #subs] [-n #msg] [-mpa #pubacks] [-ms size] "
                    + "[-io] [-a] [-csv file] \n\nOptions:\n"
                    + "    -s                        NATS Streaming server URLs (separated by"
                    + " comma)\n"
                    + "    -cid                            NATS Streaming cluster ID\n"
                    + "    -id                             Benchmark process base client ID\n"
                    + "    -tls                            Use TLS secure connection\n"
                    + "    -np                             Number of concurrent publishers\n"
                    + "    -ns                             Number of concurrent subscribers\n"
                    + "    -n                              Number of messages to publish\n"
                    + "    -a                              Async message publishing\n"
                    + "    -ms                             Message size in bytes\n"
                    + "    -io                             Subscribers ignore old messages\n"
                    + "    -ms                             Message size in bytes\n"
                    + "    -mpa                            Max number of published acks in flight\n"
                    + "    -csv                            Save bench data to csv file\n";

    /**
     * Main constructor for StanBench.
     * 
     * @param args configuration parameters
     */
    public StanBench(String[] args) {
        if (args == null || args.length < 1) {
            usage();
            return;
        }
        parseArgs(args);
    }

    /**
     * Properties-based constructor for StanBench.
     * 
     * @param properties configuration properties
     */
    public StanBench(Properties properties) {
        urls = properties.getProperty("bench.stan.servers", urls);
        clientId = properties.getProperty("bench.streaming.client.id", clientId);
        clusterId = properties.getProperty("bench.stan.cluster.id", clusterId);
        secure = Boolean.parseBoolean(
                properties.getProperty("bench.stan.secure", Boolean.toString(secure)));
        numMsgs = Integer.parseInt(
                properties.getProperty("bench.stan.msg.count", Integer.toString(numMsgs)));
        maxPubAcksInFlight = Integer.parseInt(properties.getProperty("bench.stan.pub.maxpubacks",
                Integer.toString(maxPubAcksInFlight)));
        size = Integer
                .parseInt(properties.getProperty("bench.stan.msg.size", Integer.toString(numSubs)));
        numPubs = Integer
                .parseInt(properties.getProperty("bench.stan.pubs", Integer.toString(numPubs)));
        numSubs = Integer
                .parseInt(properties.getProperty("bench.stan.subs", Integer.toString(numSubs)));
        csvFileName = properties.getProperty("bench.stan.csv.filename", null);
        subject = properties.getProperty("bench.stan.subject", NUID.nextGlobal());
        async = Boolean.parseBoolean(
                properties.getProperty("bench.stan.pub.async", Boolean.toString(async)));
        ignoreOld = Boolean.parseBoolean(
                properties.getProperty("bench.stan.sub.ignoreold", Boolean.toString(ignoreOld)));
    }

    class Worker implements Runnable {
        final Phaser phaser;
        final int numMsgs;
        final int size;
        final String workerClientId;

        Worker(Phaser phaser, int numMsgs, int size, String workerClientId) {
            Thread.currentThread().setName(workerClientId);
            this.phaser = phaser;
            this.numMsgs = numMsgs;
            this.size = size;
            this.workerClientId = workerClientId;
        }

        @Override
        public void run() {}
    }

    class SubWorker extends Worker {
        private final boolean ignoreOld;

        SubWorker(Phaser phaser, int numMsgs, int size, boolean ignoreOld, String subId) {
            super(phaser, numMsgs, size, subId);
            this.ignoreOld = ignoreOld;
        }

        @Override
        public void run() {
            try {
                runSubscriber();
            } catch (Exception e) {
                e.printStackTrace();
                phaser.arrive();
            }
        }

        public void runSubscriber() throws Exception {
            try (final io.nats.client.Connection nc = Nats.connect(natsOptions)) {
                io.nats.streaming.Options opts = new io.nats.streaming.Options.Builder()
                        .natsConn(nc)
                        .build();

                final StreamingConnection sc = NatsStreaming.connect(clusterId, clientId, opts);
                final SubscriptionOptions sopts;

                if (ignoreOld) {
                    sopts = new SubscriptionOptions.Builder().deliverAllAvailable().build();
                } else {
                    sopts = new SubscriptionOptions.Builder().build();
                }

                final long start = System.nanoTime();

                final Subscription sub = sc.subscribe(subject, new MessageHandler() {
                    @Override
                    public void onMessage(Message msg) {
                        received.incrementAndGet();
                        if (received.get() >= numMsgs) {
                            bench.addSubSample(new Sample(numMsgs, size, start, System.nanoTime(), nc.getStatistics()));
                            System.out.printf("Subscriber connection stats: " + nc.getStatistics());
                            phaser.arrive();
                            try {
                                sc.close();
                            } catch (IOException | TimeoutException e) {
                                System.err.printf(
                                        "streaming-bench: "
                                                + "exception thrown during subscriber connection close",
                                        e);
                            } catch (InterruptedException e) {
                                System.err.printf("Interrupted during subscriber connection close", e);
                                Thread.currentThread().interrupt();
                            }
                        }
                    }
                }, sopts);

                phaser.arrive();
                while (received.get() < numMsgs) {
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        break;
                    }
                }
            }
        }
    }

    class PubWorker extends Worker {
        private final boolean async;

        PubWorker(Phaser phaser, int numMsgs, int size, boolean async, String pubId) {
            super(phaser, numMsgs, size, pubId);
            this.async = async;
        }

        @Override
        public void run() {
            try {
                runPublisher();
                phaser.arrive();
            } catch (Exception e) {
                e.printStackTrace();
                phaser.arrive();
            }
        }

        public void runPublisher() throws Exception {
            try (final io.nats.client.Connection nc = Nats.connect(natsOptions)) {
                Options pubOpts;
                if (maxPubAcksInFlight > 0) {
                    pubOpts = new Options.Builder()
                            .maxPubAcksInFlight(maxPubAcksInFlight)
                            .natsConn(nc)
                            .build();
                } else {
                    pubOpts = new Options.Builder()
                            .natsConn(nc)
                            .build();
                }
                try (StreamingConnection sc =
                            NatsStreaming.connect(clusterId, workerClientId, pubOpts)) {

                    byte[] msg = null;
                    if (size > 0) {
                        msg = new byte[size];
                    }

                    final long start = System.nanoTime();

                    if (async) {
                        CountDownLatch latch = new CountDownLatch(1);
                        AckHandler acb = new AckHandler() {
                            public void onAck(String nuid, Exception ex) {
                                if (published.incrementAndGet() >= numMsgs) {
                                    latch.countDown();
                                }
                            }
                        };
                        for (int i = 0; i < numMsgs; i++) {
                            try {
                                sc.publish(subject, msg, acb);
                            } catch (Exception e) {
                                System.err.printf("streaming-bench: error during publish", e);
                            }
                        }
                        latch.await();
                    } else {
                        for (int i = 0; i < numMsgs; i++) {
                            try {
                                sc.publish(subject, msg);
                                published.incrementAndGet();
                            } catch (Exception e) {
                                System.err.printf("streaming-bench: error during publish", e);
                            }
                        }
                    }

                    bench.addPubSample(new Sample(numMsgs, size, start, System.nanoTime(), nc.getStatistics()));
                    System.out.printf("Publisher connection stats: \n" + nc.getStatistics());
                } // StreamingConnection
            }
        }
    }

    /**
     * Runs the benchmark.
     * 
     * @throws Exception if an exception occurs
     */
    public void run() throws Exception {
        ExecutorService exec = Executors.newCachedThreadPool();
        final Phaser phaser = new Phaser();

        installShutdownHook();

        phaser.register();
        
        String[] servers = urls.split(",");
        io.nats.client.Options.Builder builder = new io.nats.client.Options.Builder();
        builder.noReconnect();
        builder.connectionName("StanBench");
        builder.servers(servers);

        if (secure) {
            builder.secure();
        }

        builder.errorListener(new ErrorListener(){
            @Override
            public void slowConsumerDetected(Connection conn, Consumer consumer) {
                System.err.println("Slow consumer detected on client side.");
            }
        
            @Override
            public void exceptionOccurred(Connection conn, Exception ex) {
                System.err.printf("Connection exception %s, connection status is %s\n", ex, conn.getStatus());
                System.err.printf("Sent=%d, Received=%d\n", published.get(), received.get());
            }
        
            @Override
            public void errorOccurred(Connection conn, String err) {
                System.err.println("Error message from server "+err);
            }
        });

        natsOptions = builder.build();

        bench = new Benchmark("NATS Streaming");

        // Run Subscribers first
        for (int i = 0; i < numSubs; i++) {
            phaser.register();
            String subId = String.format("%s-sub-%d", clientId, i);
            exec.execute(new SubWorker(phaser, numMsgs, size, ignoreOld, subId));
        }

        // Wait for subscribers threads to initialize
        phaser.arriveAndAwaitAdvance();

        // Now publishers
        List pubCounts = io.nats.streaming.examples.benchmark.Utils.msgsPerClient(numMsgs, numPubs);
        for (int i = 0; i < numPubs; i++) {
            phaser.register();
            String pubId = String.format("%s-pub-%d", clientId, i);
            exec.execute(new PubWorker(phaser, pubCounts.get(i), size, async, pubId));
        }

        System.out.printf("Starting benchmark [msgs=%d, msgsize=%d, pubs=%d, subs=%d]\n", numMsgs,
                size, numPubs, numSubs);

        // Wait for subscribers and publishers to finish
        phaser.arriveAndAwaitAdvance();

        // We're done. Clean up and report.
        Runtime.getRuntime().removeShutdownHook(shutdownHook);

        bench.close();
        System.out.println(bench.report());

        if (csvFileName != null) {
            String csv = bench.csv();
            Path csvFile = Paths.get(csvFileName);
            Files.write(csvFile, Collections.singletonList(csv), Charset.forName("UTF-8"));
        }

        exec.shutdown();
    }

    private void installShutdownHook() {
        shutdownHook = new Thread(new Runnable() {
            @Override
            public void run() {
                System.err.println("\nCaught CTRL-C, shutting down gracefully...\n");
                shutdown.set(true);
                System.err.printf("Sent=%d\n", published.get());
                System.err.printf("Received=%d\n", received.get());

            }
        });

        Runtime.getRuntime().addShutdownHook(shutdownHook);
    }

    private void usage() {
        System.err.println(usageString);
        System.exit(-1);
    }

    private void parseArgs(String[] args) {
        List argList = new ArrayList(Arrays.asList(args));

        subject = argList.get(argList.size() - 1);
        argList.remove(argList.size() - 1);

        if (subject.startsWith("-")) {
            usage();
        }

        // Anything left is flags + args
        Iterator it = argList.iterator();
        while (it.hasNext()) {
            String arg = it.next();
            switch (arg) {
                case "-s":
                case "--server":
                    if (!it.hasNext()) {
                        usage();
                    }
                    it.remove();
                    urls = it.next();
                    it.remove();
                    continue;
                case "--tls":
                    secure = true;
                    it.remove();
                    continue;
                case "-np":
                    if (!it.hasNext()) {
                        usage();
                    }
                    it.remove();
                    numPubs = Integer.parseInt(it.next());
                    it.remove();
                    continue;
                case "-ns":
                    if (!it.hasNext()) {
                        usage();
                    }
                    it.remove();
                    numSubs = Integer.parseInt(it.next());
                    it.remove();
                    continue;
                case "-n":
                    if (!it.hasNext()) {
                        usage();
                    }
                    it.remove();
                    numMsgs = Integer.parseInt(it.next());
                    it.remove();
                    continue;
                case "-a":
                    async = true;
                    it.remove();
                    continue;
                case "-ms":
                    if (!it.hasNext()) {
                        usage();
                    }
                    it.remove();
                    size = Integer.parseInt(it.next());
                    it.remove();
                    continue;
                case "-io":
                    ignoreOld = true;
                    it.remove();
                    continue;
                case "-mpa":
                    if (!it.hasNext()) {
                        usage();
                    }
                    it.remove();
                    maxPubAcksInFlight = Integer.parseInt(it.next());
                    it.remove();
                    continue;
                case "-id":
                    if (!it.hasNext()) {
                        usage();
                    }
                    it.remove();
                    clientId = it.next();
                    it.remove();
                    continue;
                case "-c":
                case "-cid":
                    if (!it.hasNext()) {
                        usage();
                    }
                    it.remove();
                    clusterId = it.next();
                    it.remove();
                    continue;
                case "-csv":
                    if (!it.hasNext()) {
                        usage();
                    }
                    it.remove();
                    csvFileName = it.next();
                    it.remove();
                    continue;

                default:
                    System.err.printf("Unexpected token: '%s'\n", arg);
                    usage();
                    break;
            }
        }
    }

    private static Properties loadProperties(String configPath) {
        try {
            InputStream is = new FileInputStream(configPath);
            Properties prop = new Properties();
            prop.load(is);
            return prop;
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }

    /**
     * The main program executive.
     * 
     * @param args command line arguments
     */
    public static void main(String[] args) {
        try {
            if (args.length == 1 && args[0].endsWith(".properties")) {
                Properties properties = loadProperties(args[0]);
                new StanBench(properties).run();
            } else {
                new StanBench(args).run();
            }
        } catch (Exception e) {
            e.printStackTrace();
            System.exit(-1);
        }
        System.exit(0);
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy