/* * 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.catalina.ant.jmx; import org.apache.tools.ant.BuildException; /** * * <b>Definition</b>: * <pre> * <path id="catalina_ant"> * <fileset dir="${catalina.home}/server/lib"> * <include name="catalina-ant.jar"/> * </fileset> * </path> * * <typedef * name="jmxCondition" * classname="org.apache.catalina.ant.jmx.JMXAccessorCondition" * classpathref="catalina_ant"/> * <taskdef * name="jmxOpen" * classname="org.apache.catalina.ant.jmx.JMXAccessorTask" * classpathref="catalina_ant"/> * </pre> * * <b>Usage</b>: Wait for start backup node * <pre> * <target name="wait"> * <jmxOpen * host="${jmx.host}" port="${jmx.port}" username="${jmx.username}" password="${jmx.password}" /> * <waitfor maxwait="${maxwait}" maxwaitunit="second" timeoutproperty="server.timeout" > * <and> * <socket server="${server.name}" port="${server.port}"/> * <http url="${url}"/> * <jmxCondition * name="Catalina:type=IDataSender,host=localhost,senderAddress=192.168.111.1,senderPort=9025" * operation="==" * attribute="connected" value="true" * /> * <jmxCondition * operation="&lt;" * name="Catalina:j2eeType=WebModule,name=//${tomcat.application.host}${tomcat.application.path},J2EEApplication=none,J2EEServer=none" * attribute="startupTime" value="250" * /> * </and> * </waitfor> * <fail if="server.timeout" message="Server ${url} don't answer inside ${maxwait} sec" /> * <echo message="Server ${url} alive" /> * </target> * * </pre> * Allowed operation between jmx attribute and reference value: * <ul> * <li>== equals</li> * <li>!= not equals</li> * <li>> greater than (&gt;)</li> * <li>>= greater than or equals (&gt;=)</li> * <li>< lesser than (&lt;)</li> * <li><= lesser than or equals (&lt;=)</li> * </ul> * <b>NOTE</b>: For numeric expressions the type must be set and use xml entities as operations.<br> * As type we currently support <em>long</em> and <em>double</em>. * @author Peter Rossbach * @since 5.5.10 */ public class JMXAccessorCondition extends JMXAccessorConditionBase { // ----------------------------------------------------- Instance Variables private String operation = "==" ; private String type = "long" ; private String unlessCondition; private String ifCondition; // ----------------------------------------------------- Properties /** * @return Returns the operation. */ public String getOperation() { return operation; } /** * @param operation The operation to set. */ public void setOperation(String operation) { this.operation = operation; } /** * @return Returns the type. */ public String getType() { return type; } /** * @param type The type to set. */ public void setType(String type) { this.type = type; } /** * @return Returns the ifCondition. */ public String getIf() { return ifCondition; } /** * Only execute if a property of the given name exists in the current project. * @param c property name */ public void setIf(String c) { ifCondition = c; } /** * @return Returns the unlessCondition. */ public String getUnless() { return unlessCondition; } /** * Only execute if a property of the given name does not * exist in the current project. * @param c property name */ public void setUnless(String c) { unlessCondition = c; } /** * test the if condition * @return true if there is no if condition, or the named property exists */ protected boolean testIfCondition() { if (ifCondition == null || "".equals(ifCondition)) { return true; } return getProject().getProperty(ifCondition) != null; } /** * test the unless condition * @return true if there is no unless condition, * or there is a named property but it doesn't exist */ protected boolean testUnlessCondition() { if (unlessCondition == null || "".equals(unlessCondition)) { return true; } return getProject().getProperty(unlessCondition) == null; } /** * This method evaluates the condition * It support for operation ">,>=,<,<=" the types <code>long</code> and <code>double</code>. * @return expression <em>jmxValue</em> <em>operation</em> <em>value</em> */ @Override public boolean eval() { String value = getValue(); if (operation == null) { throw new BuildException("operation attribute is not set"); } if (value == null) { throw new BuildException("value attribute is not set"); } if ((getName() == null || getAttribute() == null)) { throw new BuildException( "Must specify an MBean name and attribute for condition"); } if (testIfCondition() && testUnlessCondition()) { String jmxValue = accessJMXValue(); if (jmxValue != null) { String op = getOperation(); if ("==".equals(op)) { return jmxValue.equals(value); } else if ("!=".equals(op)) { return !jmxValue.equals(value); } else { if ("long".equals(type)) { long jvalue = Long.parseLong(jmxValue); long lvalue = Long.parseLong(value); if (">".equals(op)) { return jvalue > lvalue; } else if (">=".equals(op)) { return jvalue >= lvalue; } else if ("<".equals(op)) { return jvalue < lvalue; } else if ("<=".equals(op)) { return jvalue <= lvalue; } } else if ("double".equals(type)) { double jvalue = Double.parseDouble(jmxValue); double dvalue = Double.parseDouble(value); if (">".equals(op)) { return jvalue > dvalue; } else if (">=".equals(op)) { return jvalue >= dvalue; } else if ("<".equals(op)) { return jvalue < dvalue; } else if ("<=".equals(op)) { return jvalue <= dvalue; } } } } return false; } return true; } }