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

org.rhq.enterprise.communications.util.CommandTraceUtil Maven / Gradle / Ivy

The newest version!
/*
 * RHQ Management Platform
 * Copyright (C) 2005-2008 Red Hat, Inc.
 * All rights reserved.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation version 2 of the License.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 */
package org.rhq.enterprise.communications.util;

import java.io.NotSerializableException;
import java.io.Serializable;
import java.util.Hashtable;
import java.util.Map;
import java.util.Properties;

import org.rhq.core.util.exception.ThrowableUtil;
import org.rhq.core.util.stream.StreamUtil;
import org.rhq.enterprise.communications.command.Command;
import org.rhq.enterprise.communications.command.CommandResponse;
import org.rhq.enterprise.communications.command.impl.remotepojo.RemotePojoInvocationCommand;
import org.rhq.enterprise.communications.command.impl.stream.RemoteInputStreamCommand;
import org.rhq.enterprise.communications.command.impl.stream.RemoteOutputStreamCommand;

/**
 * Static utility methods useful when logging commands, used mainly for debugging a client and remote endpoint.
 *
 * @author John Mazzitelli
 */
public class CommandTraceUtil {

    // name of system property that, when set to true, dumps the configuration properties of all commands
    private static final String SYSPROP_TRACE_COMMAND_CONFIG = "rhq.trace-command-config";

    // name of system property that, when set to a number, dumps the results of all command responses
    private static final String SYSPROP_TRACE_COMMAND_RESPONSE_RESULTS = "rhq.trace-command-response-results";

    // name of system property that, when set to a number, dumps the size of the command when larger than the prop value
    private static final String SYSPROP_TRACE_COMMAND_SIZE_THRESHOLD = "rhq.trace-command-size-threshold";

    // name of system property that, when set to a number, dumps the size of the command response when larger than the prop value
    private static final String SYSPROP_TRACE_COMMAND_RESPONSE_SIZE_THRESHOLD = "rhq.trace-command-response-size-threshold";

    // just a simple cache so we can avoid having to parse the interface names all the time
    private static Map interfaceSimpleNames = null;

    public static Boolean getSettingTraceCommandConfig() {
        if (System.getProperty(SYSPROP_TRACE_COMMAND_CONFIG) != null) {
            return Boolean.getBoolean(SYSPROP_TRACE_COMMAND_CONFIG);
        }
        return null; // we want to return null if not set
    }

    public static void setSettingTraceCommandConfig(Boolean val) {
        System.setProperty(SYSPROP_TRACE_COMMAND_CONFIG, (val != null) ? Boolean.toString(val) : "false");
    }

    public static Integer getSettingTraceCommandResponseResults() {
        int val = getPropertyInt(SYSPROP_TRACE_COMMAND_RESPONSE_RESULTS);
        return (val != -1) ? Integer.valueOf(val) : null; // returns null if not set, numeric otherwise
    }

    public static void setSettingTraceCommandResponseResults(Integer val) {
        System.setProperty(SYSPROP_TRACE_COMMAND_RESPONSE_RESULTS, (val != null) ? Integer.toString(val) : "-1");
    }

    public static Integer getSettingTraceCommandSizeThreshold() {
        int val = getPropertyInt(SYSPROP_TRACE_COMMAND_SIZE_THRESHOLD);
        return (val != -1) ? Integer.valueOf(val) : null; // returns null if not set, numeric otherwise
    }

    public static void setSettingTraceCommandSizeThreshold(Integer val) {
        System.setProperty(SYSPROP_TRACE_COMMAND_SIZE_THRESHOLD, (val != null) ? Integer.toString(val) : "-1");
    }

    public static Integer getSettingTraceCommandResponseSizeThreshold() {
        int val = getPropertyInt(SYSPROP_TRACE_COMMAND_RESPONSE_SIZE_THRESHOLD);
        return (val != -1) ? Integer.valueOf(val) : null; // returns null if not set, numeric otherwise
    }

    public static void setSettingTraceCommandResponseSizeThreshold(Integer val) {
        System.setProperty(SYSPROP_TRACE_COMMAND_RESPONSE_SIZE_THRESHOLD, (val != null) ? Integer.toString(val) : "-1");
    }

    /**
     * Returns the string that represents the command configuration.
     * 
     * @param command the command whose configuration is stringified and returned
     * 
     * @return the stringified configuration
     */
    public static String getConfigString(Command command) {
        Properties config = null;
        if (command != null && Boolean.getBoolean(SYSPROP_TRACE_COMMAND_CONFIG)) {
            config = command.getConfiguration();
        }
        return (config != null) ? config.toString() : "?";
    }

    /**
     * Returns a good string to use to log the command.  If this is a remote pojo invocation
     * command, this will return the remote pojo method being invoked; otherwise, its the
     * command's type name.
     * 
     * @param command the command to convert to a string
     * 
     * @return the command's string that should be logged
     */
    public static String getCommandString(Command command) {
        String commandString;
        if (command == null) {
            commandString = "null command";
        } else if (command.getCommandType().equals(RemotePojoInvocationCommand.COMMAND_TYPE)) {
            RemotePojoInvocationCommand remoteCmd = (RemotePojoInvocationCommand) command;
            StringBuilder fullMethod = new StringBuilder(getInterfaceSimpleName(remoteCmd.getTargetInterfaceName()));
            fullMethod.append('.');
            fullMethod.append(remoteCmd.getNameBasedInvocation().getMethodName());
            commandString = fullMethod.toString();
        } else if (command.getCommandType().equals(RemoteOutputStreamCommand.COMMAND_TYPE)) {
            RemoteOutputStreamCommand streamCmd = (RemoteOutputStreamCommand) command;
            StringBuilder fullMethod = new StringBuilder("OutputStream.");
            fullMethod.append(streamCmd.getNameBasedInvocation().getMethodName());
            fullMethod.append('.');
            fullMethod.append(streamCmd.getStreamId());
            commandString = fullMethod.toString();
        } else if (command.getCommandType().equals(RemoteInputStreamCommand.COMMAND_TYPE)) {
            RemoteInputStreamCommand streamCmd = (RemoteInputStreamCommand) command;
            StringBuilder fullMethod = new StringBuilder("InputStream.");
            fullMethod.append(streamCmd.getNameBasedInvocation().getMethodName());
            fullMethod.append('.');
            fullMethod.append(streamCmd.getStreamId());
            commandString = fullMethod.toString();
        } else {
            commandString = command.getCommandType().getName();
        }
        return commandString;
    }

    /**
     * Returns a good string to use to log the command response.
     * 
     * @param response the command response to convert to a string
     * 
     * @return the command response string that should be logged
     */
    public static String getCommandResponseString(Object response) {
        String responseString;
        if (response instanceof CommandResponse) {
            CommandResponse cmdResponse = (CommandResponse) response;
            if (cmdResponse.isSuccessful()) {
                if (System.getProperty(SYSPROP_TRACE_COMMAND_RESPONSE_RESULTS) != null) {
                    responseString = "success:" + getCommandResponseResultsString(cmdResponse.getResults());
                } else {
                    responseString = "success";
                }
            } else {
                responseString = "failed:" + ThrowableUtil.getAllMessages(cmdResponse.getException());
            }
        } else if (response instanceof Throwable) {
            responseString = ThrowableUtil.getAllMessages((Throwable) response);
        } else if (response instanceof String) {
            responseString = "STRING:" + response.toString();
        } else if (response == null) {
            responseString = "NULL";
        } else {
            responseString = "UNEXPECTED:" + response.getClass().getName();
        }
        return responseString;
    }

    /**
     * Returns the actual size, in bytes, of the given command but only if it
     * exceeds the given configured threshold (-1 is returned otherwise).
     * If size tracing threshold is disabled, this will not perform any
     * serialization and simply return -1
     * ({@link #SYSPROP_TRACE_COMMAND_SIZE_THRESHOLD} defines this threshold).
     * Note that if the command is not serializable, an exception is thrown.
     * All commands must be serializable; therefore, if we trace a command that is
     * not, it is appropriate for us to draw attention to it by throwing an
     * exception from this method.
     * 
     * @param command
     * 
     * @return the size of the command or -1 if not configured to perform size tracing
     * 
     * @throws NotSerializableException if failed to serialize the command
     */
    public static int getCommandSize(Command command) throws NotSerializableException {
        int threshold = getPropertyInt(SYSPROP_TRACE_COMMAND_SIZE_THRESHOLD);
        if (threshold > -1) {
            try {
                int size = StreamUtil.serialize(command).length;
                if (size > threshold) {
                    return size;
                }
            } catch (Exception e) {
                throw new NotSerializableException(ThrowableUtil.getAllMessages(e));
            }
        }
        return -1;
    }

    /**
     * Returns the actual size, in bytes, of the given response but only if it
     * exceeds the given configured threshold (-1 is returned otherwise).
     * If size tracing threshold is disabled, this will not perform any
     * serialization and simply return -1
     * (see {@link #SYSPROP_TRACE_COMMAND_RESPONSE_SIZE_THRESHOLD} defines this threshold).
     * Note that if the response is not serializable, a runtime exception is thrown.
     * All responses must be serializable; therefore, if we trace a response that is
     * not, it is appropriate for us to draw attention to it by throwing an
     * exception from this method.
     * 
     * @param response
     * 
     * @return the size of the response or -1 if not configured to perform size tracing
     * 
     * @throws NotSerializableException if failed to serialize the response
     */
    public static int getCommandResponseSize(Object response) throws NotSerializableException {
        int threshold = getPropertyInt(SYSPROP_TRACE_COMMAND_RESPONSE_SIZE_THRESHOLD);
        if (threshold > -1) {
            try {
                int size = StreamUtil.serialize((Serializable) response).length;
                if (size > threshold) {
                    return size;
                }
            } catch (Exception e) {
                throw new NotSerializableException(ThrowableUtil.getAllMessages(e));
            }
        }
        return -1;
    }

    /**
     * Returns a stringified form of a command response results object.  This makes it so
     * the string isn't really long and doesn't contain newlines so we can show it on a single
     * line in the log file.
     * 
     * @param results the results object from a command response
     * 
     * @return results string
     */
    private static String getCommandResponseResultsString(Object results) {

        if (results == null) {
            return "null";
        }

        StringBuilder resultsString = new StringBuilder();

        try {
            resultsString.append(results.toString());
        } catch (Throwable t) {
            return "toString failed: [" + t + "]";
        }

        int maxLength;

        try {
            // if the sys prop is a number, use it as our max length - useful if developer wants to see alot or alittle
            maxLength = Integer.parseInt(System.getProperty(SYSPROP_TRACE_COMMAND_RESPONSE_RESULTS));
        } catch (NumberFormatException e) {
            maxLength = 256;
        }

        if (resultsString.length() > maxLength) {
            resultsString.setLength(maxLength);
            resultsString.append("...");
        }

        return resultsString.toString().replaceAll("[\r\n]", ".");
    }

    /**
     * Gets the simple name of the given interface name. The simple name is the name
     * of the interface itself, minus any package names.
     * 
     * @param interfaceName the full interface name
     * 
     * @return the name of the interface class itself, without packages
     */
    private static String getInterfaceSimpleName(String interfaceName) {
        if (interfaceName == null) { // i'm paranoid
            return "unknown";
        }

        if (interfaceSimpleNames == null) {
            interfaceSimpleNames = new Hashtable();
        }

        String simpleName = interfaceSimpleNames.get(interfaceName);

        if (simpleName == null) {
            int period = interfaceName.lastIndexOf('.');
            if (period == -1 || period == (interfaceName.length() - 1)) {
                simpleName = interfaceName;
            } else {
                simpleName = interfaceName.substring(period + 1);
            }
            interfaceSimpleNames.put(interfaceName, simpleName);
        }

        return simpleName;
    }

    /**
     * Returns the given system property's value as a number. If the property
     * isn't set or is invalid, -1 is returned.
     * 
     * @param propertyName
     * 
     * @return the property's value as a number
     */
    private static int getPropertyInt(String propertyName) {
        String propertyValue = System.getProperty(propertyName);
        if (propertyValue != null) {
            try {
                return Integer.parseInt(propertyValue, 10);
            } catch (Exception e) {
                System.setProperty(propertyName, "-1"); // invalid number, just disable it so we don't keep checking
            }
        }
        return -1;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy