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

org.apache.brooklyn.policy.enricher.TimeWeightedDeltaEnricher Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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 org.apache.brooklyn.policy.enricher;

import org.apache.brooklyn.api.catalog.Catalog;
import org.apache.brooklyn.api.entity.Entity;
import org.apache.brooklyn.api.sensor.AttributeSensor;
import org.apache.brooklyn.api.sensor.EnricherSpec;
import org.apache.brooklyn.api.sensor.Sensor;
import org.apache.brooklyn.api.sensor.SensorEvent;
import org.apache.brooklyn.enricher.stock.AbstractTypeTransformingEnricher;
import org.apache.brooklyn.enricher.stock.YamlTimeWeightedDeltaEnricher;
import org.apache.brooklyn.util.core.flags.SetFromFlag;
import org.apache.brooklyn.util.groovy.GroovyJavaMethods;
import org.apache.brooklyn.util.javalang.JavaClassNames;
import org.apache.brooklyn.util.time.Duration;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.google.common.annotations.Beta;
import com.google.common.base.Function;
import com.google.common.base.Functions;

import groovy.lang.Closure;

/**
 * Converts an absolute sensor into a delta sensor (i.e. the diff between the current and previous value),
 * presented as a units/timeUnit based on the event timing.
 * 

* NB for time (e.g. "total milliseconds consumed") use {@link TimeFractionDeltaEnricher} *

* See also {@link YamlTimeWeightedDeltaEnricher} designed for use from YAML. *

* TODO this may end up being deprecated in favour of near-duplicate code in YAML-friendly {@link YamlTimeWeightedDeltaEnricher}, * marking as @Beta in 0.7.0 timeframe */ @Beta @Catalog(name="Time-weighted Delta", description="Converts an absolute sensor into a delta sensor " + "(i.e. the diff between the current and previous value), presented as a units/timeUnit " + "based on the event timing.") public class TimeWeightedDeltaEnricher extends AbstractTypeTransformingEnricher { private static final Logger LOG = LoggerFactory.getLogger(TimeWeightedDeltaEnricher.class); Number lastValue; long lastTime = -1; /** unitMillis is the number of milliseconds to apply for the conversion from input to output; * e.g. 1000 for counting things per second; * NB for time (e.g. "total milliseconds consumed") use {@link TimeFractionDeltaEnricher} */ @SetFromFlag int unitMillis; @SetFromFlag Function postProcessor; /** * @deprecated since 0.12.0; use {@link EnricherSpec} */ @Deprecated // default 1 second public static TimeWeightedDeltaEnricher getPerSecondDeltaEnricher(Entity producer, Sensor source, Sensor target) { return new TimeWeightedDeltaEnricher(producer, source, target, 1000); } public TimeWeightedDeltaEnricher() { // for rebind } /** * @deprecated since 0.12.0; use {@link EnricherSpec} */ @Deprecated public TimeWeightedDeltaEnricher(Entity producer, Sensor source, Sensor target, int unitMillis) { this(producer, source, target, unitMillis, Functions.identity()); } /** * @deprecated since 0.11.0; explicit groovy utilities/support will be deleted; also use {@link EnricherSpec} */ @Deprecated public TimeWeightedDeltaEnricher(Entity producer, Sensor source, Sensor target, int unitMillis, Closure postProcessor) { this(producer, source, target, unitMillis, GroovyJavaMethods.functionFromClosure(postProcessor)); } /** * @deprecated since 0.12.0; use {@link EnricherSpec} */ @Deprecated public TimeWeightedDeltaEnricher(Entity producer, Sensor source, Sensor target, int unitMillis, Function postProcessor) { super(producer, source, target); this.unitMillis = unitMillis; this.postProcessor = postProcessor; if (source!=null && target!=null) this.uniqueTag = JavaClassNames.simpleClassName(getClass())+":"+source.getName()+"/"+Duration.millis(unitMillis)+"->"+target.getName(); } @Override public void init() { super.init(); if (uniqueTag == null && source != null && target != null) { uniqueTag = JavaClassNames.simpleClassName(getClass())+":"+source.getName()+"/"+Duration.millis(unitMillis)+"->"+target.getName(); } } @Override public void onEvent(SensorEvent event) { onEvent(event, event.getTimestamp()); } public void onEvent(SensorEvent event, long eventTime) { Number current = event.getValue(); if (current == null) { // Can't compute a delta; // don't assume current=zero because then things like requestCount->requestsPerSecond is negative! // instead assume same as last time, so delta == 0 double deltaPostProcessed = getPostProcessor().apply(0d); entity.sensors().set((AttributeSensor)target, deltaPostProcessed); if (LOG.isTraceEnabled()) LOG.trace("set {} to {}, {} -> {} at {}", new Object[] {this, deltaPostProcessed, lastValue, current, eventTime}); return; } if (eventTime > 0 && eventTime > lastTime) { if (lastValue == null || lastTime <= 0) { // cannot calculate time-based delta with a single value if (LOG.isTraceEnabled()) LOG.trace("{} received event but no last value so will not emit, null -> {} at {}", new Object[] {this, current, eventTime}); } else { double duration = (lastTime < 0) ? unitMillis : eventTime - lastTime; if (eventTime == lastTime) duration = 0.1; // 0.1 of a millisecond is a relatively small number: double delta = (current.doubleValue() - lastValue.doubleValue()) / (duration / unitMillis); double deltaPostProcessed = getPostProcessor().apply(delta); entity.sensors().set((AttributeSensor)target, deltaPostProcessed); if (LOG.isTraceEnabled()) LOG.trace("set {} to {}, {} -> {} at {}", new Object[] {this, deltaPostProcessed, lastValue, current, eventTime}); } lastValue = current; lastTime = eventTime; } else if (lastTime<0) { lastValue = current; lastTime = -1; } } private Function getPostProcessor() { return (postProcessor != null) ? postProcessor : Functions.identity(); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy