org.opendaylight.netconf.test.tool.client.stress.StressClient.java Source code

Java tutorial

Introduction

Here is the source code for org.opendaylight.netconf.test.tool.client.stress.StressClient.java

Source

/*
 * Copyright (c) 2015 Cisco Systems, Inc. and others.  All rights reserved.
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License v1.0 which accompanies this distribution,
 * and is available at http://www.eclipse.org/legal/epl-v10.html
 */

package org.opendaylight.netconf.test.tool.client.stress;

import ch.qos.logback.classic.Level;
import com.google.common.base.Charsets;
import com.google.common.base.Stopwatch;
import com.google.common.io.Files;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.util.HashedWheelTimer;
import io.netty.util.Timer;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import net.sourceforge.argparse4j.inf.ArgumentParser;
import net.sourceforge.argparse4j.inf.ArgumentParserException;
import org.opendaylight.controller.config.util.xml.XmlUtil;
import org.opendaylight.netconf.api.NetconfMessage;
import org.opendaylight.netconf.client.NetconfClientDispatcherImpl;
import org.opendaylight.netconf.nettyutil.handler.ssh.client.AsyncSshHandler;
import org.opendaylight.netconf.sal.connect.api.RemoteDevice;
import org.opendaylight.netconf.sal.connect.netconf.listener.NetconfDeviceCommunicator;
import org.opendaylight.netconf.sal.connect.netconf.listener.NetconfSessionPreferences;
import org.opendaylight.netconf.test.tool.TestToolUtils;
import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.netconf.base._1._0.rev110601.CommitInput;
import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.netconf.base._1._0.rev110601.EditConfigInput;
import org.opendaylight.yangtools.yang.common.QName;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.xml.sax.SAXException;

public final class StressClient {

    private static final Logger LOG = LoggerFactory.getLogger(StressClient.class);

    static final QName COMMIT_QNAME = QName.create(CommitInput.QNAME, "commit");
    public static final NetconfMessage COMMIT_MSG;

    static {
        try {
            COMMIT_MSG = new NetconfMessage(XmlUtil.readXmlToDocument(
                    "<rpc message-id=\"commit-batch\" xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
                            + "    <commit/>\n" + "</rpc>"));
        } catch (SAXException | IOException e) {
            throw new ExceptionInInitializerError(e);
        }
    }

    static final QName EDIT_QNAME = QName.create(EditConfigInput.QNAME, "edit-config");
    static final org.w3c.dom.Document editCandidateBlueprint;
    static final org.w3c.dom.Document editRunningBlueprint;

    static {
        try {
            editCandidateBlueprint = XmlUtil
                    .readXmlToDocument("<rpc xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
                            + "    <edit-config xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
                            + "        <target>\n" + "            <candidate/>\n" + "        </target>\n"
                            + "        <default-operation>none</default-operation>" + "        <config/>\n"
                            + "    </edit-config>\n" + "</rpc>");

            editRunningBlueprint = XmlUtil
                    .readXmlToDocument("<rpc xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
                            + "    <edit-config xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
                            + "        <target>\n" + "            <running/>\n" + "        </target>\n"
                            + "        <default-operation>none</default-operation>" + "        <config/>\n"
                            + "    </edit-config>\n" + "</rpc>");
        } catch (SAXException | IOException e) {
            throw new ExceptionInInitializerError(e);
        }
    }

    private static final String MSG_ID_PLACEHOLDER_REGEX = "\\{MSG_ID\\}";
    private static final String PHYS_ADDR_PLACEHOLDER = "{PHYS_ADDR}";

    private static long macStart = 0xAABBCCDD0000L;

    private static Parameters params;

    public static void main(final String[] args) {

        params = parseArgs(args, Parameters.getParser());
        params.validate();

        final ch.qos.logback.classic.Logger root = (ch.qos.logback.classic.Logger) LoggerFactory
                .getLogger(Logger.ROOT_LOGGER_NAME);
        root.setLevel(params.debug ? Level.DEBUG : Level.INFO);

        final int threadAmount = params.threadAmount;
        LOG.info("thread amount: " + threadAmount);
        final int requestsPerThread = params.editCount / params.threadAmount;
        LOG.info("requestsPerThread: " + requestsPerThread);
        final int leftoverRequests = params.editCount % params.threadAmount;
        LOG.info("leftoverRequests: " + leftoverRequests);

        LOG.info("Preparing messages");
        // Prepare all msgs up front
        final List<List<NetconfMessage>> allPreparedMessages = new ArrayList<>(threadAmount);
        for (int i = 0; i < threadAmount; i++) {
            if (i != threadAmount - 1) {
                allPreparedMessages.add(new ArrayList<NetconfMessage>(requestsPerThread));
            } else {
                allPreparedMessages.add(new ArrayList<NetconfMessage>(requestsPerThread + leftoverRequests));
            }
        }

        final String editContentString;
        try {
            editContentString = Files.toString(params.editContent, Charsets.UTF_8);
        } catch (final IOException e) {
            throw new IllegalArgumentException("Cannot read content of " + params.editContent);
        }

        for (int i = 0; i < threadAmount; i++) {
            final List<NetconfMessage> preparedMessages = allPreparedMessages.get(i);
            int padding = 0;
            if (i == threadAmount - 1) {
                padding = leftoverRequests;
            }
            for (int j = 0; j < requestsPerThread + padding; j++) {
                LOG.debug("id: " + (i * requestsPerThread + j));
                preparedMessages.add(prepareMessage(i * requestsPerThread + j, editContentString));
            }
        }

        final NioEventLoopGroup nioGroup = new NioEventLoopGroup();
        final Timer timer = new HashedWheelTimer();

        final NetconfClientDispatcherImpl netconfClientDispatcher = configureClientDispatcher(params, nioGroup,
                timer);

        final List<StressClientCallable> callables = new ArrayList<>(threadAmount);
        for (final List<NetconfMessage> messages : allPreparedMessages) {
            callables.add(new StressClientCallable(params, netconfClientDispatcher, messages));
        }

        final ExecutorService executorService = Executors.newFixedThreadPool(threadAmount);

        LOG.info("Starting stress test");
        final Stopwatch started = Stopwatch.createStarted();
        try {
            final List<Future<Boolean>> futures = executorService.invokeAll(callables);
            for (final Future<Boolean> future : futures) {
                try {
                    future.get(4L, TimeUnit.MINUTES);
                } catch (ExecutionException | TimeoutException e) {
                    throw new RuntimeException(e);
                }
            }
            executorService.shutdownNow();
        } catch (final InterruptedException e) {
            throw new RuntimeException("Unable to execute requests", e);
        }
        started.stop();

        LOG.info("FINISHED. Execution time: {}", started);
        LOG.info("Requests per second: {}", (params.editCount * 1000.0 / started.elapsed(TimeUnit.MILLISECONDS)));

        // Cleanup
        timer.stop();
        try {
            nioGroup.shutdownGracefully().get(20L, TimeUnit.SECONDS);
        } catch (InterruptedException | ExecutionException | TimeoutException e) {
            LOG.warn("Unable to close executor properly", e);
        }
        //stop the underlying ssh thread that gets spawned if we use ssh
        if (params.ssh) {
            AsyncSshHandler.DEFAULT_CLIENT.stop();
        }
    }

    static NetconfMessage prepareMessage(final int id, final String editContentString) {
        final Document msg = XmlUtil
                .createDocumentCopy(params.candidateDatastore ? editCandidateBlueprint : editRunningBlueprint);
        msg.getDocumentElement().setAttribute("message-id", Integer.toString(id));
        final NetconfMessage netconfMessage = new NetconfMessage(msg);

        final Element editContentElement;
        try {
            // Insert message id where needed
            String specificEditContent = editContentString.replaceAll(MSG_ID_PLACEHOLDER_REGEX,
                    Integer.toString(id));

            final StringBuilder stringBuilder = new StringBuilder(specificEditContent);
            int idx = stringBuilder.indexOf(PHYS_ADDR_PLACEHOLDER);
            while (idx != -1) {
                stringBuilder.replace(idx, idx + PHYS_ADDR_PLACEHOLDER.length(), TestToolUtils.getMac(macStart++));
                idx = stringBuilder.indexOf(PHYS_ADDR_PLACEHOLDER);
            }
            specificEditContent = stringBuilder.toString();

            editContentElement = XmlUtil.readXmlToElement(specificEditContent);
            final Node config = ((Element) msg.getDocumentElement().getElementsByTagName("edit-config").item(0))
                    .getElementsByTagName("config").item(0);
            config.appendChild(msg.importNode(editContentElement, true));
        } catch (final IOException | SAXException e) {
            throw new IllegalArgumentException("Edit content file is unreadable", e);
        }

        return netconfMessage;
    }

    private static NetconfClientDispatcherImpl configureClientDispatcher(final Parameters params,
            final NioEventLoopGroup nioGroup, final Timer timer) {
        final NetconfClientDispatcherImpl netconfClientDispatcher;
        if (params.exi) {
            if (params.legacyFraming) {
                netconfClientDispatcher = ConfigurableClientDispatcher.createLegacyExi(nioGroup, nioGroup, timer);
            } else {
                netconfClientDispatcher = ConfigurableClientDispatcher.createChunkedExi(nioGroup, nioGroup, timer);
            }
        } else {
            if (params.legacyFraming) {
                netconfClientDispatcher = ConfigurableClientDispatcher.createLegacy(nioGroup, nioGroup, timer);
            } else {
                netconfClientDispatcher = ConfigurableClientDispatcher.createChunked(nioGroup, nioGroup, timer);
            }
        }
        return netconfClientDispatcher;
    }

    private static Parameters parseArgs(final String[] args, final ArgumentParser parser) {
        final Parameters opt = new Parameters();
        try {
            parser.parseArgs(args, opt);
            return opt;
        } catch (final ArgumentParserException e) {
            parser.handleError(e);
        }

        System.exit(1);
        return null;
    }

    static class LoggingRemoteDevice
            implements RemoteDevice<NetconfSessionPreferences, NetconfMessage, NetconfDeviceCommunicator> {
        @Override
        public void onRemoteSessionUp(final NetconfSessionPreferences remoteSessionCapabilities,
                final NetconfDeviceCommunicator netconfDeviceCommunicator) {
            LOG.info("Session established");
        }

        @Override
        public void onRemoteSessionDown() {
            LOG.info("Session down");
        }

        @Override
        public void onRemoteSessionFailed(final Throwable throwable) {
            LOG.info("Session failed");
        }

        @Override
        public void onNotification(final NetconfMessage notification) {
            LOG.info("Notification received: {}", notification.toString());
        }
    }

}