Package net.kuujo.vertigo.io.impl

Source Code of net.kuujo.vertigo.io.impl.DefaultInputCollector

/*
* Copyright 2014 the original author or 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 net.kuujo.vertigo.io.impl;

import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;

import net.kuujo.vertigo.io.InputCollector;
import net.kuujo.vertigo.io.InputContext;
import net.kuujo.vertigo.io.port.InputPort;
import net.kuujo.vertigo.io.port.InputPortContext;
import net.kuujo.vertigo.io.port.impl.DefaultInputPort;
import net.kuujo.vertigo.io.port.impl.DefaultInputPortContext;
import net.kuujo.vertigo.util.CountingCompletionHandler;
import net.kuujo.vertigo.util.Observer;
import net.kuujo.vertigo.util.Task;
import net.kuujo.vertigo.util.TaskRunner;

import org.vertx.java.core.AsyncResult;
import org.vertx.java.core.Handler;
import org.vertx.java.core.Vertx;
import org.vertx.java.core.impl.DefaultFutureResult;
import org.vertx.java.core.logging.Logger;
import org.vertx.java.core.logging.impl.LoggerFactory;

/**
* Default input collector implementation.
*
* @author <a href="http://github.com/kuujo">Jordan Halterman</a>
*/
public class DefaultInputCollector implements InputCollector, Observer<InputContext> {
  private final Logger log;
  private final Vertx vertx;
  private InputContext context;
  private final Map<String, InputPort> ports = new HashMap<>();
  private final TaskRunner tasks = new TaskRunner();
  private boolean started;

  public DefaultInputCollector(Vertx vertx) {
    this.vertx = vertx;
    this.log = LoggerFactory.getLogger(DefaultInputCollector.class);
  }

  public DefaultInputCollector(Vertx vertx, InputContext context) {
    this.vertx = vertx;
    this.context = context;
    this.log = LoggerFactory.getLogger(String.format("%s-%s-%d", DefaultInputCollector.class.getName(), context.instance().component().name(), context.instance().number()));
    context.registerObserver(this);
  }

  @Override
  public Collection<InputPort> ports() {
    return ports.values();
  }

  @Override
  public InputPort port(String name) {
    // Attempt to search for the port in the existing context. If the
    // port isn't an explicitly configured port then lazily create
    // and open the port. The lazy port will be empty.
    InputPort port = ports.get(name);
    if (port == null) {
      log.debug(String.format("%s - Lazily creating in port: %s", this, name));

      // Attempt to search for the port in the existing context. If the
      // port isn't an explicitly configured port then lazily create
      // and open the port. The lazy port will be empty.
      InputPortContext portContext = context.port(name);
      if (portContext == null) {
        portContext = DefaultInputPortContext.Builder.newBuilder()
            .setAddress(UUID.randomUUID().toString())
            .setName(name)
            .build();
        DefaultInputContext.Builder.newBuilder((DefaultInputContext) context).addPort(portContext);
      }
      port = new DefaultInputPort(vertx, context.port(name));
      if (started) {
        port.open();
      }
      ports.put(name, port);
    }
    return port;
  }

  @Override
  public void update(InputContext context) {
    log.info(String.format("%s - Input configuration has changed, updating ports", this));

    // Copy the context in order to ensure that future changes via the
    // observer will not effect this update.
    final InputContext update = context.copy();

    // All updates are run sequentially to prevent race conditions
    // during configuration changes. Without essentially locking the
    // object, it could be possible that connections are simultaneously
    // added and removed or opened and closed on the object.
    tasks.runTask(new Handler<Task>() {
      @Override
      public void handle(final Task task) {
        final List<InputPort> newPorts = new ArrayList<>();
        for (InputPortContext input : update.ports()) {
          if (!ports.containsKey(input.name())) {
            InputPortContext port = DefaultInputCollector.this.context.port(input.name());
            if (port != null) {
              log.debug(String.format("%s - Adding in port: %s", DefaultInputCollector.this, input));
              newPorts.add(new DefaultInputPort(vertx, port));
            }
          }
        }

        // If the input has already been started, add each input port
        // only once the port has been started. This ensures that the
        // input cannot receive messages until the port is open.
        if (started) {
          final CountingCompletionHandler<Void> counter = new CountingCompletionHandler<Void>(newPorts.size());
          counter.setHandler(new Handler<AsyncResult<Void>>() {
            @Override
            public void handle(AsyncResult<Void> result) {
              task.complete();
            }
          });

          // Iterate through each new input port and open and add the port.
          for (final InputPort port : newPorts) {
            log.debug(String.format("%s - Opening in port: %s", DefaultInputCollector.this, port));
            port.open(new Handler<AsyncResult<Void>>() {
              @Override
              public void handle(AsyncResult<Void> result) {
                if (result.failed()) {
                  log.error(String.format("%s - Failed to open in port: %s", DefaultInputCollector.this, port));
                } else {
                  log.info(String.format("%s - Opened in port: %s", DefaultInputCollector.this, port));
                  ports.put(port.name(), port);
                }
                counter.succeed();
              }
            });
          }
        } else {
          // If the input is not already started, simply open and add the ports.
          // The ports will be open once the input is started.
          for (InputPort port : newPorts) {
            ports.put(port.name(), port);
          }
          task.complete();
        }
      }
    });
  }

  @Override
  public InputCollector open() {
    return open(null);
  }

  @Override
  public InputCollector open(final Handler<AsyncResult<Void>> doneHandler) {
    // Prevent the object from being opened and closed simultaneously
    // by queueing open/close operations as tasks.
    tasks.runTask(new Handler<Task>() {
      @Override
      public void handle(final Task task) {
        // If the input hasn't already been started, start the input
        // by adding and opening any necessary ports.
        if (!started) {
          final CountingCompletionHandler<Void> startCounter = new CountingCompletionHandler<Void>(context.ports().size());
          startCounter.setHandler(new Handler<AsyncResult<Void>>() {
            @Override
            public void handle(AsyncResult<Void> result) {
              if (result.failed()) {
                new DefaultFutureResult<Void>(result.cause()).setHandler(doneHandler);
              } else {
                new DefaultFutureResult<Void>((Void) null).setHandler(doneHandler);
              }
              task.complete();
            }
          });

          for (final InputPortContext port : context.ports()) {
            log.debug(String.format("%s - Opening in port: %s", DefaultInputCollector.this, port));
            if (ports.containsKey(port.name())) {
              ((DefaultInputPort) ports.get(port.name())).open(new Handler<AsyncResult<Void>>() {
                @Override
                public void handle(AsyncResult<Void> result) {
                  if (result.failed()) {
                    log.error(String.format("%s - Failed to open in port: %s", DefaultInputCollector.this, port));
                    startCounter.fail(result.cause());
                  } else {
                    log.info(String.format("%s - Opened in port: %s", DefaultInputCollector.this, port));
                    startCounter.succeed();
                  }
                }
              });
            } else {
              ports.put(port.name(), new DefaultInputPort(vertx, port).open(new Handler<AsyncResult<Void>>() {
                @Override
                public void handle(AsyncResult<Void> result) {
                  if (result.failed()) {
                    log.error(String.format("%s - Failed to open in port: %s", DefaultInputCollector.this, port));
                    startCounter.fail(result.cause());
                  } else {
                    log.info(String.format("%s - Opened in port: %s", DefaultInputCollector.this, port));
                    startCounter.succeed();
                  }
                }
              }));
            }
          }
          started = true;
        } else {
          new DefaultFutureResult<Void>((Void) null).setHandler(doneHandler);
          task.complete();
        }
      }
    });
    return this;
  }

  @Override
  public void close() {
    close(null);
  }

  @Override
  public void close(final Handler<AsyncResult<Void>> doneHandler) {
    // Prevent the object from being opened and closed simultaneously
    // by queueing open/close operations as tasks.
    tasks.runTask(new Handler<Task>() {
      @Override
      public void handle(final Task task) {
        if (started) {
          final CountingCompletionHandler<Void> stopCounter = new CountingCompletionHandler<Void>(ports.size());
          stopCounter.setHandler(new Handler<AsyncResult<Void>>() {
            @Override
            public void handle(AsyncResult<Void> result) {
              if (result.failed()) {
                new DefaultFutureResult<Void>(result.cause()).setHandler(doneHandler);
              } else {
                ports.clear();
                started = false;
                new DefaultFutureResult<Void>((Void) null).setHandler(doneHandler);
              }
              task.complete();
            }
          });
     
          for (final InputPort port : ports.values()) {
            log.debug(String.format("%s - Closing in port: %s", DefaultInputCollector.this, port));
            port.close(new Handler<AsyncResult<Void>>() {
              @Override
              public void handle(AsyncResult<Void> result) {
                if (result.failed()) {
                  log.warn(String.format("%s - Failed to close in port: %s", DefaultInputCollector.this, port));
                  stopCounter.fail(result.cause());
                } else {
                  log.info(String.format("%s - Closed in port: %s", DefaultInputCollector.this, port));
                  stopCounter.succeed();
                }
              }
            });
          }
        } else {
          new DefaultFutureResult<Void>((Void) null).setHandler(doneHandler);
          task.complete();
        }
      }
    });
  }

  @Override
  public String toString() {
    return context.toString();
  }

}
TOP

Related Classes of net.kuujo.vertigo.io.impl.DefaultInputCollector

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.