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

swim.system.router.HostTable Maven / Gradle / Ivy

The newest version!
// Copyright 2015-2024 Nstream, inc.
//
// 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 swim.system.router;

import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
import java.util.concurrent.atomic.AtomicLongFieldUpdater;
import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;
import swim.api.Downlink;
import swim.api.lane.DemandLane;
import swim.api.lane.DemandMapLane;
import swim.api.lane.SupplyLane;
import swim.api.lane.function.OnCue;
import swim.api.lane.function.OnCueKey;
import swim.api.lane.function.OnSyncKeys;
import swim.api.policy.Policy;
import swim.api.warp.WarpUplink;
import swim.collections.FingerTrieSeq;
import swim.concurrent.Cont;
import swim.concurrent.Schedule;
import swim.concurrent.Stage;
import swim.store.StoreBinding;
import swim.structure.Value;
import swim.system.AbstractTierBinding;
import swim.system.HostAddress;
import swim.system.HostBinding;
import swim.system.HostContext;
import swim.system.HostException;
import swim.system.LaneBinding;
import swim.system.LinkBinding;
import swim.system.Metric;
import swim.system.NodeAddress;
import swim.system.NodeBinding;
import swim.system.NodeContext;
import swim.system.PartBinding;
import swim.system.Push;
import swim.system.TierContext;
import swim.system.UplinkError;
import swim.system.agent.AgentNode;
import swim.system.profile.HostProfile;
import swim.system.profile.NodeProfile;
import swim.system.profile.WarpDownlinkProfile;
import swim.system.reflect.AgentPulse;
import swim.system.reflect.HostPulse;
import swim.system.reflect.LogEntry;
import swim.system.reflect.NodeInfo;
import swim.system.reflect.SystemPulse;
import swim.system.reflect.WarpDownlinkPulse;
import swim.system.reflect.WarpUplinkPulse;
import swim.uri.Uri;
import swim.uri.UriFragment;
import swim.uri.UriMapper;
import swim.uri.UriPart;
import swim.uri.UriPath;
import swim.uri.UriPathBuilder;

public class HostTable extends AbstractTierBinding implements HostBinding {

  protected HostContext hostContext;
  volatile UriMapper nodes;
  volatile int flags;

  volatile int nodeOpenDelta;
  volatile long nodeOpenCount;
  volatile int nodeCloseDelta;
  volatile long nodeCloseCount;
  volatile int agentOpenDelta;
  volatile long agentOpenCount;
  volatile int agentCloseDelta;
  volatile long agentCloseCount;
  volatile long agentExecDelta;
  volatile long agentExecRate;
  volatile long agentExecTime;
  volatile int timerEventDelta;
  volatile int timerEventRate;
  volatile long timerEventCount;
  volatile int downlinkOpenDelta;
  volatile long downlinkOpenCount;
  volatile int downlinkCloseDelta;
  volatile long downlinkCloseCount;
  volatile int downlinkEventDelta;
  volatile int downlinkEventRate;
  volatile long downlinkEventCount;
  volatile int downlinkCommandDelta;
  volatile int downlinkCommandRate;
  volatile long downlinkCommandCount;
  volatile int uplinkOpenDelta;
  volatile long uplinkOpenCount;
  volatile int uplinkCloseDelta;
  volatile long uplinkCloseCount;
  volatile int uplinkEventDelta;
  volatile int uplinkEventRate;
  volatile long uplinkEventCount;
  volatile int uplinkCommandDelta;
  volatile int uplinkCommandRate;
  volatile long uplinkCommandCount;
  volatile long lastReportTime;

  HostPulse pulse;
  AgentNode metaNode;
  DemandMapLane metaNodes;
  DemandLane metaPulse;
  SupplyLane metaTraceLog;
  SupplyLane metaDebugLog;
  SupplyLane metaInfoLog;
  SupplyLane metaWarnLog;
  SupplyLane metaErrorLog;
  SupplyLane metaFailLog;

  public HostTable() {
    this.hostContext = null;
    this.nodes = UriMapper.empty();
    this.flags = 0;

    this.nodeOpenDelta = 0;
    this.nodeOpenCount = 0L;
    this.nodeCloseDelta = 0;
    this.nodeCloseCount = 0L;
    this.agentOpenDelta = 0;
    this.agentOpenCount = 0L;
    this.agentCloseDelta = 0;
    this.agentCloseCount = 0L;
    this.agentExecDelta = 0L;
    this.agentExecRate = 0L;
    this.agentExecTime = 0L;
    this.timerEventDelta = 0;
    this.timerEventRate = 0;
    this.timerEventCount = 0L;
    this.downlinkOpenDelta = 0;
    this.downlinkOpenCount = 0L;
    this.downlinkCloseDelta = 0;
    this.downlinkCloseCount = 0L;
    this.downlinkEventDelta = 0;
    this.downlinkEventRate = 0;
    this.downlinkEventCount = 0L;
    this.downlinkCommandDelta = 0;
    this.downlinkCommandRate = 0;
    this.downlinkCommandCount = 0L;
    this.uplinkOpenDelta = 0;
    this.uplinkOpenCount = 0L;
    this.uplinkCloseDelta = 0;
    this.uplinkCloseCount = 0L;
    this.uplinkEventDelta = 0;
    this.uplinkEventRate = 0;
    this.uplinkEventCount = 0L;
    this.uplinkCommandDelta = 0;
    this.uplinkCommandRate = 0;
    this.uplinkCommandCount = 0L;
    this.lastReportTime = 0L;

    this.pulse = null;
    this.metaNode = null;
    this.metaNodes = null;
    this.metaPulse = null;
    this.metaTraceLog = null;
    this.metaDebugLog = null;
    this.metaInfoLog = null;
    this.metaWarnLog = null;
    this.metaErrorLog = null;
    this.metaFailLog = null;
  }

  @Override
  public final TierContext tierContext() {
    return this.hostContext;
  }

  @Override
  public final PartBinding part() {
    return this.hostContext.part();
  }

  @Override
  public final HostBinding hostWrapper() {
    return this;
  }

  @Override
  public final HostContext hostContext() {
    return this.hostContext;
  }

  @Override
  public void setHostContext(HostContext hostContext) {
    this.hostContext = hostContext;
  }

  @SuppressWarnings("unchecked")
  @Override
  public  T unwrapHost(Class hostClass) {
    if (hostClass.isAssignableFrom(this.getClass())) {
      return (T) this;
    } else {
      return this.hostContext.unwrapHost(hostClass);
    }
  }

  @SuppressWarnings("unchecked")
  @Override
  public  T bottomHost(Class hostClass) {
    T host = this.hostContext.bottomHost(hostClass);
    if (host == null && hostClass.isAssignableFrom(this.getClass())) {
      host = (T) this;
    }
    return host;
  }

  protected NodeContext createNodeContext(NodeAddress nodeAddress, NodeBinding node) {
    return new HostTableNode(this, node, nodeAddress);
  }

  @Override
  public final HostAddress cellAddress() {
    return this.hostContext.cellAddress();
  }

  @Override
  public final String edgeName() {
    return this.hostContext.edgeName();
  }

  @Override
  public final Uri meshUri() {
    return this.hostContext.meshUri();
  }

  @Override
  public final Value partKey() {
    return this.hostContext.partKey();
  }

  @Override
  public final Uri hostUri() {
    return this.hostContext.hostUri();
  }

  @Override
  public Policy policy() {
    return this.hostContext.policy();
  }

  @Override
  public Schedule schedule() {
    return this.hostContext.schedule();
  }

  @Override
  public Stage stage() {
    return this.hostContext.stage();
  }

  @Override
  public StoreBinding store() {
    return this.hostContext.store();
  }

  @Override
  public boolean isConnected() {
    return true;
  }

  @Override
  public boolean isRemote() {
    return false;
  }

  @Override
  public boolean isSecure() {
    return true;
  }

  @Override
  public boolean isPrimary() {
    return (HostTable.FLAGS.get(this) & HostTable.PRIMARY) != 0;
  }

  @Override
  public void setPrimary(boolean isPrimary) {
    do {
      final int oldFlags = HostTable.FLAGS.get(this);
      final int newFlags = oldFlags | HostTable.PRIMARY;
      if (HostTable.FLAGS.compareAndSet(this, oldFlags, newFlags)) {
        break;
      }
    } while (true);
  }

  @Override
  public boolean isReplica() {
    return (HostTable.FLAGS.get(this) & HostTable.REPLICA) != 0;
  }

  @Override
  public void setReplica(boolean isReplica) {
    do {
      final int oldFlags = HostTable.FLAGS.get(this);
      final int newFlags = oldFlags | HostTable.REPLICA;
      if (HostTable.FLAGS.compareAndSet(this, oldFlags, newFlags)) {
        break;
      }
    } while (true);
  }

  @Override
  public boolean isMaster() {
    return (HostTable.FLAGS.get(this) & HostTable.MASTER) != 0;
  }

  @Override
  public boolean isSlave() {
    return (HostTable.FLAGS.get(this) & HostTable.SLAVE) != 0;
  }

  @Override
  public void didBecomeMaster() {
    do {
      final int oldFlags = HostTable.FLAGS.get(this);
      final int newFlags = oldFlags & ~HostTable.SLAVE | HostTable.MASTER;
      if (HostTable.FLAGS.compareAndSet(this, oldFlags, newFlags)) {
        break;
      }
    } while (true);
  }

  @Override
  public void didBecomeSlave() {
    do {
      final int oldFlags = HostTable.FLAGS.get(this);
      final int newFlags = oldFlags & ~HostTable.MASTER | HostTable.SLAVE;
      if (HostTable.FLAGS.compareAndSet(this, oldFlags, newFlags)) {
        if (oldFlags != newFlags) {
          this.closeNodes();
        }
      }
    } while (true);
  }

  @Override
  public void openMetaHost(HostBinding host, NodeBinding metaHost) {
    if (metaHost instanceof AgentNode) {
      this.metaNode = (AgentNode) metaHost;
      this.openMetaLanes(host, (AgentNode) metaHost);
    }
    this.hostContext.openMetaHost(host, metaHost);
  }

  protected void openMetaLanes(HostBinding host, AgentNode metaHost) {
    this.openReflectLanes(host, metaHost);
    this.openLogLanes(host, metaHost);
  }

  protected void openReflectLanes(HostBinding host, AgentNode metaHost) {
    this.metaNodes = metaHost.demandMapLane()
                             .keyForm(Uri.form())
                             .valueForm(NodeInfo.form())
                             .observe(new HostTableNodesController(host));
    metaHost.openLane(HostTable.NODES_URI, this.metaNodes);

    this.metaPulse = this.metaNode.demandLane()
                                  .valueForm(HostPulse.form())
                                  .observe(new HostTablePulseController(this));
    this.metaNode.openLane(HostPulse.PULSE_URI, this.metaPulse);
  }

  protected void openLogLanes(HostBinding host, AgentNode metaHost) {
    this.metaTraceLog = metaHost.supplyLane().valueForm(LogEntry.form());
    metaHost.openLane(LogEntry.TRACE_LOG_URI, this.metaTraceLog);

    this.metaDebugLog = metaHost.supplyLane().valueForm(LogEntry.form());
    metaHost.openLane(LogEntry.DEBUG_LOG_URI, this.metaDebugLog);

    this.metaInfoLog = metaHost.supplyLane().valueForm(LogEntry.form());
    metaHost.openLane(LogEntry.INFO_LOG_URI, this.metaInfoLog);

    this.metaWarnLog = metaHost.supplyLane().valueForm(LogEntry.form());
    metaHost.openLane(LogEntry.WARN_LOG_URI, this.metaWarnLog);

    this.metaErrorLog = metaHost.supplyLane().valueForm(LogEntry.form());
    metaHost.openLane(LogEntry.ERROR_LOG_URI, this.metaErrorLog);

    this.metaFailLog = metaHost.supplyLane().valueForm(LogEntry.form());
    metaHost.openLane(LogEntry.FAIL_LOG_URI, this.metaFailLog);
  }

  @Override
  public UriMapper nodes() {
    return HostTable.NODES.get(this);
  }

  @Override
  public NodeBinding getNode(Uri nodeUri) {
    return HostTable.NODES.get(this).get(nodeUri);
  }

  @Override
  public NodeBinding openNode(Uri nodeUri) {
    NodeBinding nodeBinding = null;
    do {
      final UriMapper oldNodes = HostTable.NODES.get(this);
      final NodeBinding node = oldNodes.get(nodeUri);
      if (node != null) {
        if (nodeBinding != null) {
          // Lost creation race.
          nodeBinding.close();
        }
        if (node.isStarted()) {
          nodeBinding = node;
          break;
        } else if (node.isClosed()) {
          nodeBinding = null;
          break;
        }
        // try again
      } else {
        if (nodeBinding == null) {
          final NodeAddress nodeAddress = this.cellAddress().nodeUri(nodeUri);
          nodeBinding = this.hostContext.createNode(nodeAddress);
          if (nodeBinding != null) {
            nodeBinding = this.hostContext.injectNode(nodeAddress, nodeBinding);
            final NodeContext nodeContext = this.createNodeContext(nodeAddress, nodeBinding);
            nodeBinding.setNodeContext(nodeContext);
            nodeBinding = nodeBinding.nodeWrapper();
            nodeBinding.openLanes(nodeBinding);
            nodeBinding.openAgents(nodeBinding);
          } else {
            break;
          }
        }
        final UriMapper newNodes = oldNodes.updated(nodeUri, nodeBinding);
        if (HostTable.NODES.compareAndSet(this, oldNodes, newNodes)) {
          this.activate(nodeBinding);
          this.didOpenNode(nodeBinding);
          break;
        }
      }
    } while (true);
    return nodeBinding;
  }

  @Override
  public NodeBinding openNode(Uri nodeUri, NodeBinding node) {
    NodeBinding nodeBinding = null;
    do {
      final UriMapper oldNodes = HostTable.NODES.get(this);
      if (oldNodes.containsKey(nodeUri)) {
        nodeBinding = null;
        break;
      } else {
        if (nodeBinding == null) {
          final NodeAddress nodeAddress = this.cellAddress().nodeUri(nodeUri);
          nodeBinding = this.hostContext.injectNode(nodeAddress, node);
          final NodeContext nodeContext = this.createNodeContext(nodeAddress, nodeBinding);
          nodeBinding.setNodeContext(nodeContext);
          nodeBinding = nodeBinding.nodeWrapper();
          nodeBinding.openLanes(nodeBinding);
          nodeBinding.openAgents(nodeBinding);
        }
        final UriMapper newNodes = oldNodes.updated(nodeUri, nodeBinding);
        if (HostTable.NODES.compareAndSet(this, oldNodes, newNodes)) {
          this.activate(nodeBinding);
          this.didOpenNode(nodeBinding);
          break;
        }
      }
    } while (true);
    return nodeBinding;
  }

  public void closeNode(Uri nodeUri) {
    do {
      final UriMapper oldNodes = HostTable.NODES.get(this);
      final NodeBinding nodeBinding = oldNodes.get(nodeUri);
      if (nodeBinding != null) {
        final UriMapper newNodes = oldNodes.removed(nodeUri);
        if (HostTable.NODES.compareAndSet(this, oldNodes, newNodes)) {
          nodeBinding.didClose();
          this.didCloseNode(nodeBinding);
          break;
        }
      } else {
        break;
      }
    } while (true);
  }

  public void closeNodes() {
    do {
      final UriMapper oldNodes = HostTable.NODES.get(this);
      if (!oldNodes.isEmpty()) {
        final UriMapper newNodes = UriMapper.empty();
        if (HostTable.NODES.compareAndSet(this, oldNodes, newNodes)) {
          final DemandMapLane metaNodes = this.metaNodes;
          for (NodeBinding nodeBinding : oldNodes.values()) {
            nodeBinding.close();
            nodeBinding.didClose();
            if (metaNodes != null) {
              metaNodes.cue(nodeBinding.nodeUri());
            }
          }
          this.flushMetrics();
          break;
        }
      } else {
        break;
      }
    } while (true);
  }

  protected void didOpenNode(NodeBinding node) {
    final DemandMapLane metaNodes = this.metaNodes;
    if (metaNodes != null) {
      final Uri nodeUri = node.nodeUri();
      metaNodes.cue(nodeUri);
      this.cueAncestorNodes(nodeUri);
    }
    HostTable.NODE_OPEN_DELTA.incrementAndGet(this);
    this.flushMetrics();
  }

  protected void didCloseNode(NodeBinding node) {
    final DemandMapLane metaNodes = this.metaNodes;
    if (metaNodes != null) {
      final Uri nodeUri = node.nodeUri();
      metaNodes.remove(nodeUri);
      this.cueAncestorNodes(nodeUri);
    }
    HostTable.NODE_CLOSE_DELTA.incrementAndGet(this);
    this.flushMetrics();
  }

  protected void cueAncestorNodes(Uri nodeUri) {
    final DemandMapLane metaNodes = this.metaNodes;
    if (metaNodes != null) {
      UriPath nodePath = nodeUri.path();
      final UriPathBuilder ancestorPathBuilder = UriPath.builder();
      while (!nodePath.isEmpty()) {
        if (nodePath.isAbsolute()) {
          ancestorPathBuilder.addSlash();
          nodePath = nodePath.tail();
        } else {
          ancestorPathBuilder.addSegment(nodePath.head());
          nodePath = nodePath.tail();
          if (nodePath.isAbsolute()) {
            final Uri ancestorUri = nodeUri.path(ancestorPathBuilder.bind());
            metaNodes.cue(ancestorUri);
            ancestorPathBuilder.addSlash();
            nodePath = nodePath.tail();
          }
        }
      }
    }
  }

  @Override
  public void openMetaNode(NodeBinding node, NodeBinding metaNode) {
    this.hostContext.openMetaNode(node, metaNode);
  }

  @Override
  public void openMetaLane(LaneBinding lane, NodeBinding metaLane) {
    this.hostContext.openMetaLane(lane, metaLane);
  }

  @Override
  public void openMetaUplink(LinkBinding uplink, NodeBinding metaUplink) {
    this.hostContext.openMetaUplink(uplink, metaUplink);
  }

  @Override
  public void openMetaDownlink(LinkBinding downlink, NodeBinding metaDownlink) {
    this.hostContext.openMetaDownlink(downlink, metaDownlink);
  }

  @Override
  public LinkBinding bindDownlink(Downlink downlink) {
    final LinkBinding link = this.hostContext.bindDownlink(downlink);
    link.setCellContext(this);
    return link;
  }

  @Override
  public void openDownlink(LinkBinding link) {
    this.hostContext.openDownlink(link);
    link.setCellContext(this);
  }

  @Override
  public void closeDownlink(LinkBinding link) {
    this.hostContext.closeDownlink(link);
  }

  @Override
  public void pushDown(Push push) {
    this.hostContext.pushDown(push);
  }

  @Override
  public void openUplink(LinkBinding link) {
    NodeBinding nodeBinding = this.openNode(link.nodeUri());
    if (nodeBinding != null) {
      nodeBinding = nodeBinding.bottomNode(NodeBinding.class);
    }
    if (nodeBinding != null) {
      nodeBinding.openUplink(link);
    } else {
      UplinkError.rejectNodeNotFound(link);
    }
  }

  @Override
  public void pushUp(Push push) {
    final Uri nodeUri = push.nodeUri();
    NodeBinding nodeBinding = this.openNode(nodeUri);
    if (nodeBinding != null) {
      nodeBinding = nodeBinding.bottomNode(NodeBinding.class);
    }
    if (nodeBinding != null) {
      nodeBinding.pushUp(push);
    } else {
      push.trap(new HostException("unknown node: " + nodeUri));
    }
  }

  @Override
  public void trace(Object message) {
    final SupplyLane metaTraceLog = this.metaTraceLog;
    if (metaTraceLog != null) {
      metaTraceLog.push(LogEntry.trace(message));
    }
    this.hostContext.trace(message);
  }

  @Override
  public void debug(Object message) {
    final SupplyLane metaDebugLog = this.metaDebugLog;
    if (metaDebugLog != null) {
      metaDebugLog.push(LogEntry.debug(message));
    }
    this.hostContext.debug(message);
  }

  @Override
  public void info(Object message) {
    final SupplyLane metaInfoLog = this.metaInfoLog;
    if (metaInfoLog != null) {
      metaInfoLog.push(LogEntry.info(message));
    }
    this.hostContext.info(message);
  }

  @Override
  public void warn(Object message) {
    final SupplyLane metaWarnLog = this.metaWarnLog;
    if (metaWarnLog != null) {
      metaWarnLog.push(LogEntry.warn(message));
    }
    this.hostContext.warn(message);
  }

  @Override
  public void error(Object message) {
    final SupplyLane metaErrorLog = this.metaErrorLog;
    if (metaErrorLog != null) {
      metaErrorLog.push(LogEntry.error(message));
    }
    this.hostContext.error(message);
  }

  @Override
  public void fail(Object message) {
    final SupplyLane metaFailLog = this.metaFailLog;
    if (metaFailLog != null) {
      metaFailLog.push(LogEntry.fail(message));
    }
    this.hostContext.fail(message);
  }

  @Override
  protected void willOpen() {
    super.willOpen();
    final Iterator nodesIterator = HostTable.NODES.get(this).valueIterator();
    while (nodesIterator.hasNext()) {
      nodesIterator.next().open();
    }
  }

  @Override
  protected void willLoad() {
    super.willLoad();
    final Iterator nodesIterator = HostTable.NODES.get(this).valueIterator();
    while (nodesIterator.hasNext()) {
      nodesIterator.next().load();
    }
  }

  @Override
  protected void willStart() {
    super.willStart();
    final Iterator nodesIterator = HostTable.NODES.get(this).valueIterator();
    while (nodesIterator.hasNext()) {
      nodesIterator.next().start();
    }
  }

  @Override
  protected void willStop() {
    super.willStop();
    final Iterator nodesIterator = HostTable.NODES.get(this).valueIterator();
    while (nodesIterator.hasNext()) {
      nodesIterator.next().stop();
    }
  }

  @Override
  protected void willUnload() {
    super.willUnload();
    final Iterator nodesIterator = HostTable.NODES.get(this).valueIterator();
    while (nodesIterator.hasNext()) {
      nodesIterator.next().unload();
    }
  }

  @Override
  protected void willClose() {
    super.willClose();
    final Iterator nodesIterator = HostTable.NODES.get(this).valueIterator();
    while (nodesIterator.hasNext()) {
      nodesIterator.next().close();
    }
  }

  @Override
  public void didClose() {
    super.didClose();
    final AgentNode metaNode = this.metaNode;
    if (metaNode != null) {
      metaNode.close();
      this.metaNode = null;
      this.metaNodes = null;
      this.metaTraceLog = null;
      this.metaDebugLog = null;
      this.metaInfoLog = null;
      this.metaWarnLog = null;
      this.metaErrorLog = null;
      this.metaFailLog = null;
    }
    this.flushMetrics();
  }

  @Override
  public void didFail(Throwable error) {
    if (Cont.isNonFatal(error)) {
      this.fail(error);
    } else {
      error.printStackTrace();
    }
  }

  @Override
  public void reportDown(Metric metric) {
    if (metric instanceof NodeProfile) {
      this.accumulateNodeProfile((NodeProfile) metric);
    } else if (metric instanceof WarpDownlinkProfile) {
      this.accumulateWarpDownlinkProfile((WarpDownlinkProfile) metric);
    } else {
      this.hostContext.reportDown(metric);
    }
  }

  protected void accumulateNodeProfile(NodeProfile profile) {
    HostTable.AGENT_OPEN_DELTA.addAndGet(this, profile.agentOpenDelta());
    HostTable.AGENT_CLOSE_DELTA.addAndGet(this, profile.agentCloseDelta());
    HostTable.AGENT_EXEC_DELTA.addAndGet(this, profile.agentExecDelta());
    HostTable.AGENT_EXEC_RATE.addAndGet(this, profile.agentExecRate());
    HostTable.TIMER_EVENT_DELTA.addAndGet(this, profile.timerEventDelta());
    HostTable.TIMER_EVENT_RATE.addAndGet(this, profile.timerEventRate());
    HostTable.DOWNLINK_OPEN_DELTA.addAndGet(this, profile.downlinkOpenDelta());
    HostTable.DOWNLINK_CLOSE_DELTA.addAndGet(this, profile.downlinkCloseDelta());
    HostTable.DOWNLINK_EVENT_DELTA.addAndGet(this, profile.downlinkEventDelta());
    HostTable.DOWNLINK_EVENT_RATE.addAndGet(this, profile.downlinkEventRate());
    HostTable.DOWNLINK_COMMAND_DELTA.addAndGet(this, profile.downlinkCommandDelta());
    HostTable.DOWNLINK_COMMAND_RATE.addAndGet(this, profile.downlinkCommandRate());
    HostTable.UPLINK_OPEN_DELTA.addAndGet(this, profile.uplinkOpenDelta());
    HostTable.UPLINK_CLOSE_DELTA.addAndGet(this, profile.uplinkCloseDelta());
    HostTable.UPLINK_EVENT_DELTA.addAndGet(this, profile.uplinkEventDelta());
    HostTable.UPLINK_EVENT_RATE.addAndGet(this, profile.uplinkEventRate());
    HostTable.UPLINK_COMMAND_DELTA.addAndGet(this, profile.uplinkCommandDelta());
    HostTable.UPLINK_COMMAND_RATE.addAndGet(this, profile.uplinkCommandRate());
    this.didUpdateMetrics();
  }

  protected void accumulateWarpDownlinkProfile(WarpDownlinkProfile profile) {
    HostTable.DOWNLINK_OPEN_DELTA.addAndGet(this, profile.openDelta());
    HostTable.DOWNLINK_CLOSE_DELTA.addAndGet(this, profile.closeDelta());
    HostTable.DOWNLINK_EVENT_DELTA.addAndGet(this, profile.eventDelta());
    HostTable.DOWNLINK_EVENT_RATE.addAndGet(this, profile.eventRate());
    HostTable.DOWNLINK_COMMAND_DELTA.addAndGet(this, profile.commandDelta());
    HostTable.DOWNLINK_COMMAND_RATE.addAndGet(this, profile.commandRate());
    this.didUpdateMetrics();
  }

  protected void didUpdateMetrics() {
    do {
      final long newReportTime = System.currentTimeMillis();
      final long oldReportTime = HostTable.LAST_REPORT_TIME.get(this);
      final long dt = newReportTime - oldReportTime;
      if (dt >= Metric.REPORT_INTERVAL) {
        if (HostTable.LAST_REPORT_TIME.compareAndSet(this, oldReportTime, newReportTime)) {
          try {
            this.reportMetrics(dt);
          } catch (Throwable error) {
            if (Cont.isNonFatal(error)) {
              this.didFail(error);
            } else {
              throw error;
            }
          }
          break;
        }
      } else {
        break;
      }
    } while (true);
  }

  protected void flushMetrics() {
    final long newReportTime = System.currentTimeMillis();
    final long oldReportTime = HostTable.LAST_REPORT_TIME.getAndSet(this, newReportTime);
    final long dt = newReportTime - oldReportTime;
    try {
      this.reportMetrics(dt);
    } catch (Throwable error) {
      if (Cont.isNonFatal(error)) {
        this.didFail(error);
      } else {
        throw error;
      }
    }
  }

  protected void reportMetrics(long dt) {
    final HostProfile profile = this.collectProfile(dt);
    this.hostContext.reportDown(profile);
  }

  protected HostProfile collectProfile(long dt) {
    final int nodeOpenDelta = HostTable.NODE_OPEN_DELTA.getAndSet(this, 0);
    final long nodeOpenCount = HostTable.NODE_OPEN_COUNT.addAndGet(this, (long) nodeOpenDelta);
    final int nodeCloseDelta = HostTable.NODE_CLOSE_DELTA.getAndSet(this, 0);
    final long nodeCloseCount = HostTable.NODE_CLOSE_COUNT.addAndGet(this, (long) nodeCloseDelta);

    final int agentOpenDelta = HostTable.AGENT_OPEN_DELTA.getAndSet(this, 0);
    final long agentOpenCount = HostTable.AGENT_OPEN_COUNT.addAndGet(this, (long) agentOpenDelta);
    final int agentCloseDelta = HostTable.AGENT_CLOSE_DELTA.getAndSet(this, 0);
    final long agentCloseCount = HostTable.AGENT_CLOSE_COUNT.addAndGet(this, (long) agentCloseDelta);
    final long agentExecDelta = HostTable.AGENT_EXEC_DELTA.getAndSet(this, 0L);
    final long agentExecRate = HostTable.AGENT_EXEC_RATE.getAndSet(this, 0L);
    final long agentExecTime = HostTable.AGENT_EXEC_TIME.addAndGet(this, agentExecDelta);

    final int timerEventDelta = HostTable.TIMER_EVENT_DELTA.getAndSet(this, 0);
    final int timerEventRate = HostTable.TIMER_EVENT_RATE.getAndSet(this, 0);
    final long timerEventCount = HostTable.TIMER_EVENT_COUNT.addAndGet(this, (long) timerEventDelta);

    final int downlinkOpenDelta = HostTable.DOWNLINK_OPEN_DELTA.getAndSet(this, 0);
    final long downlinkOpenCount = HostTable.DOWNLINK_OPEN_COUNT.addAndGet(this, (long) downlinkOpenDelta);
    final int downlinkCloseDelta = HostTable.DOWNLINK_CLOSE_DELTA.getAndSet(this, 0);
    final long downlinkCloseCount = HostTable.DOWNLINK_CLOSE_COUNT.addAndGet(this, (long) downlinkCloseDelta);
    final int downlinkEventDelta = HostTable.DOWNLINK_EVENT_DELTA.getAndSet(this, 0);
    final int downlinkEventRate = HostTable.DOWNLINK_EVENT_RATE.getAndSet(this, 0);
    final long downlinkEventCount = HostTable.DOWNLINK_EVENT_COUNT.addAndGet(this, (long) downlinkEventDelta);
    final int downlinkCommandDelta = HostTable.DOWNLINK_COMMAND_DELTA.getAndSet(this, 0);
    final int downlinkCommandRate = HostTable.DOWNLINK_COMMAND_RATE.getAndSet(this, 0);
    final long downlinkCommandCount = HostTable.DOWNLINK_COMMAND_COUNT.addAndGet(this, (long) downlinkCommandDelta);

    final int uplinkOpenDelta = HostTable.UPLINK_OPEN_DELTA.getAndSet(this, 0);
    final long uplinkOpenCount = HostTable.UPLINK_OPEN_COUNT.addAndGet(this, (long) uplinkOpenDelta);
    final int uplinkCloseDelta = HostTable.UPLINK_CLOSE_DELTA.getAndSet(this, 0);
    final long uplinkCloseCount = HostTable.UPLINK_CLOSE_COUNT.addAndGet(this, (long) uplinkCloseDelta);
    final int uplinkEventDelta = HostTable.UPLINK_EVENT_DELTA.getAndSet(this, 0);
    final int uplinkEventRate = HostTable.UPLINK_EVENT_RATE.getAndSet(this, 0);
    final long uplinkEventCount = HostTable.UPLINK_EVENT_COUNT.addAndGet(this, (long) uplinkEventDelta);
    final int uplinkCommandDelta = HostTable.UPLINK_COMMAND_DELTA.getAndSet(this, 0);
    final int uplinkCommandRate = HostTable.UPLINK_COMMAND_RATE.getAndSet(this, 0);
    final long uplinkCommandCount = HostTable.UPLINK_COMMAND_COUNT.addAndGet(this, (long) uplinkCommandDelta);

    final long nodeCount = nodeOpenCount - nodeCloseCount;
    final long agentCount = agentOpenCount - agentCloseCount;
    final AgentPulse agentPulse = new AgentPulse(agentCount, agentExecRate, agentExecTime, timerEventRate, timerEventCount);
    final long downlinkCount = downlinkOpenCount - downlinkCloseCount;
    final WarpDownlinkPulse downlinkPulse = new WarpDownlinkPulse(downlinkCount, downlinkEventRate, downlinkEventCount,
                                                                  downlinkCommandRate, downlinkCommandCount);
    final long uplinkCount = uplinkOpenCount - uplinkCloseCount;
    final WarpUplinkPulse uplinkPulse = new WarpUplinkPulse(uplinkCount, uplinkEventRate, uplinkEventCount,
                                                            uplinkCommandRate, uplinkCommandCount);
    this.pulse = new HostPulse(nodeCount, agentPulse, downlinkPulse, uplinkPulse, SystemPulse.latest());
    final DemandLane metaPulse = this.metaPulse;
    if (metaPulse != null) {
      metaPulse.cue();
    }

    return new HostProfile(this.cellAddress(),
                           nodeOpenDelta, nodeOpenCount, nodeCloseDelta, nodeCloseCount,
                           agentOpenDelta, agentOpenCount, agentCloseDelta, agentCloseCount,
                           agentExecDelta, agentExecRate, agentExecTime,
                           timerEventDelta, timerEventRate, timerEventCount,
                           downlinkOpenDelta, downlinkOpenCount, downlinkCloseDelta, downlinkCloseCount,
                           downlinkEventDelta, downlinkEventRate, downlinkEventCount,
                           downlinkCommandDelta, downlinkCommandRate, downlinkCommandCount,
                           uplinkOpenDelta, uplinkOpenCount, uplinkCloseDelta, uplinkCloseCount,
                           uplinkEventDelta, uplinkEventRate, uplinkEventCount,
                           uplinkCommandDelta, uplinkCommandRate, uplinkCommandCount);
  }

  static final int PRIMARY = 1 << 0;
  static final int REPLICA = 1 << 1;
  static final int MASTER = 1 << 2;
  static final int SLAVE = 1 << 3;

  static final Uri NODES_URI = Uri.parse("nodes");

  @SuppressWarnings("unchecked")
  static final AtomicReferenceFieldUpdater> NODES =
      AtomicReferenceFieldUpdater.newUpdater(HostTable.class, (Class>) (Class) UriMapper.class, "nodes");

  static final AtomicIntegerFieldUpdater FLAGS =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "flags");

  static final AtomicIntegerFieldUpdater NODE_OPEN_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "nodeOpenDelta");
  static final AtomicLongFieldUpdater NODE_OPEN_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "nodeOpenCount");
  static final AtomicIntegerFieldUpdater NODE_CLOSE_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "nodeCloseDelta");
  static final AtomicLongFieldUpdater NODE_CLOSE_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "nodeCloseCount");
  static final AtomicIntegerFieldUpdater AGENT_OPEN_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "agentOpenDelta");
  static final AtomicLongFieldUpdater AGENT_OPEN_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "agentOpenCount");
  static final AtomicIntegerFieldUpdater AGENT_CLOSE_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "agentCloseDelta");
  static final AtomicLongFieldUpdater AGENT_CLOSE_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "agentCloseCount");
  static final AtomicLongFieldUpdater AGENT_EXEC_DELTA =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "agentExecDelta");
  static final AtomicLongFieldUpdater AGENT_EXEC_RATE =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "agentExecRate");
  static final AtomicLongFieldUpdater AGENT_EXEC_TIME =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "agentExecTime");
  static final AtomicIntegerFieldUpdater TIMER_EVENT_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "timerEventDelta");
  static final AtomicIntegerFieldUpdater TIMER_EVENT_RATE =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "timerEventRate");
  static final AtomicLongFieldUpdater TIMER_EVENT_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "timerEventCount");
  static final AtomicIntegerFieldUpdater DOWNLINK_OPEN_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "downlinkOpenDelta");
  static final AtomicLongFieldUpdater DOWNLINK_OPEN_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "downlinkOpenCount");
  static final AtomicIntegerFieldUpdater DOWNLINK_CLOSE_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "downlinkCloseDelta");
  static final AtomicLongFieldUpdater DOWNLINK_CLOSE_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "downlinkCloseCount");
  static final AtomicIntegerFieldUpdater DOWNLINK_EVENT_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "downlinkEventDelta");
  static final AtomicIntegerFieldUpdater DOWNLINK_EVENT_RATE =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "downlinkEventRate");
  static final AtomicLongFieldUpdater DOWNLINK_EVENT_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "downlinkEventCount");
  static final AtomicIntegerFieldUpdater DOWNLINK_COMMAND_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "downlinkCommandDelta");
  static final AtomicIntegerFieldUpdater DOWNLINK_COMMAND_RATE =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "downlinkCommandRate");
  static final AtomicLongFieldUpdater DOWNLINK_COMMAND_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "downlinkCommandCount");
  static final AtomicIntegerFieldUpdater UPLINK_OPEN_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "uplinkOpenDelta");
  static final AtomicLongFieldUpdater UPLINK_OPEN_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "uplinkOpenCount");
  static final AtomicIntegerFieldUpdater UPLINK_CLOSE_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "uplinkCloseDelta");
  static final AtomicLongFieldUpdater UPLINK_CLOSE_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "uplinkCloseCount");
  static final AtomicIntegerFieldUpdater UPLINK_EVENT_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "uplinkEventDelta");
  static final AtomicIntegerFieldUpdater UPLINK_EVENT_RATE =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "uplinkEventRate");
  static final AtomicLongFieldUpdater UPLINK_EVENT_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "uplinkEventCount");
  static final AtomicIntegerFieldUpdater UPLINK_COMMAND_DELTA =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "uplinkCommandDelta");
  static final AtomicIntegerFieldUpdater UPLINK_COMMAND_RATE =
      AtomicIntegerFieldUpdater.newUpdater(HostTable.class, "uplinkCommandRate");
  static final AtomicLongFieldUpdater UPLINK_COMMAND_COUNT =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "uplinkCommandCount");
  static final AtomicLongFieldUpdater LAST_REPORT_TIME =
      AtomicLongFieldUpdater.newUpdater(HostTable.class, "lastReportTime");

}

final class HostTableNodesController implements OnCueKey, OnSyncKeys {

  final HostBinding host;

  HostTableNodesController(HostBinding host) {
    this.host = host;
  }

  @Override
  public NodeInfo onCue(Uri nodeUri, WarpUplink uplink) {
    final NodeBinding nodeBinding;
    final String laneQuery = uplink.laneUri().query().get("q");
    final UriFragment laneFragment = uplink.laneUri().fragment();
    if (laneQuery != null) {
      if (nodeUri.toString().contains(laneQuery)) {
        nodeBinding = this.host.getNode(nodeUri);
        if (nodeBinding != null) {
          return NodeInfo.create(nodeBinding);
        }
      }
    } else if (laneFragment.isDefined()) {
      final Uri parentUri = Uri.parse(laneFragment.identifier());
      if (nodeUri.isChildOf(parentUri)) {
        nodeBinding = this.host.getNode(nodeUri);
        final Uri directoryUri = nodeUri.appendedSlash();
        final UriMapper directory = this.host.nodes().getSuffix(directoryUri);
        if (nodeBinding != null) {
          return NodeInfo.create(nodeBinding, directory.childCount());
        } else if (!directory.isEmpty()) {
          return new NodeInfo(nodeUri, 0L, FingerTrieSeq.empty(), directory.childCount()); // synthetic directory node
        }
      }
    } else {
      nodeBinding = this.host.getNode(nodeUri);
      if (nodeBinding != null) {
        return NodeInfo.create(nodeBinding);
      }
    }
    return null;
  }

  @Override
  public Iterator onSync(WarpUplink uplink) {
    final String laneQuery = uplink.laneUri().query().get("q");
    final UriFragment laneFragment = uplink.laneUri().fragment();
    if (laneQuery != null) {
      return new HostTableNodesQueryIterator(laneQuery, this.host.nodes().iterator());
    } else if (laneFragment.isDefined()) {
      final Uri parentUri = Uri.parse(laneFragment.identifier());
      final UriMapper parentSuffix = this.host.nodes().getSuffix(parentUri);
      return new HostTableNodesChildIterator(parentUri, parentSuffix.childIterator());
    }
    return this.host.nodes().keyIterator();
  }

}

final class HostTableNodesQueryIterator implements Iterator {

  final String query;
  final Iterator> nodes;
  Uri nextNodeUri;

  HostTableNodesQueryIterator(String query, Iterator> nodes) {
    this.query = query;
    this.nodes = nodes;
  }

  Uri nextNodeUri() {
    if (this.nextNodeUri == null) {
      while (this.nodes.hasNext()) {
        final Map.Entry entry = this.nodes.next();
        final Uri nodeUri = entry.getKey();
        if (nodeUri.toString().contains(this.query)) {
          this.nextNodeUri = nodeUri;
          break;
        }
      }
    }
    return this.nextNodeUri;
  }

  @Override
  public boolean hasNext() {
    return this.nextNodeUri() != null;
  }

  @Override
  public Uri next() {
    final Uri nextNodeUri = this.nextNodeUri();
    this.nextNodeUri = null;
    return nextNodeUri;
  }

  @Override
  public void remove() {
    throw new UnsupportedOperationException();
  }

}

final class HostTableNodesChildIterator implements Iterator {

  final Uri parentUri;
  final Iterator childSegments;

  HostTableNodesChildIterator(Uri parentUri, Iterator childSegments) {
    this.parentUri = parentUri;
    this.childSegments = childSegments;
  }

  @Override
  public boolean hasNext() {
    return this.childSegments.hasNext();
  }

  @Override
  public Uri next() {
    final UriPart childSegment = this.childSegments.next();
    return this.parentUri.appendedPath((UriPath) childSegment);
  }

  @Override
  public void remove() {
    throw new UnsupportedOperationException();
  }

}

final class HostTablePulseController implements OnCue {

  final HostTable host;

  HostTablePulseController(HostTable host) {
    this.host = host;
  }

  @Override
  public HostPulse onCue(WarpUplink uplink) {
    return this.host.pulse;
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy