/*
* Copyright 2004-2006 Stefan Reuter and others
*
* 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 org.asteriskjava.manager.internal;
import java.lang.reflect.Method;
import java.util.Iterator;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import java.util.HashSet;
import org.asteriskjava.AsteriskVersion;
import org.asteriskjava.manager.action.ManagerAction;
import org.asteriskjava.util.Log;
import org.asteriskjava.util.LogFactory;
import org.asteriskjava.util.ReflectionUtil;
import org.asteriskjava.manager.action.UserEventAction;
import org.asteriskjava.manager.event.UserEvent;
/**
* Default implementation of the ActionBuilder interface.
*
* @author srt
* @version $Id: ActionBuilderImpl.java 976 2008-02-03 22:08:11Z srt $
*/
public class ActionBuilderImpl implements ActionBuilder
{
protected static final String LINE_SEPARATOR = "\r\n";
/**
* Instance logger.
*/
private final Log logger = LogFactory.getLog(getClass());
private AsteriskVersion targetVersion;
/**
* Creates a new ActionBuilder for Asterisk 1.0.
*/
public ActionBuilderImpl()
{
this.targetVersion = AsteriskVersion.ASTERISK_1_0;
}
public void setTargetVersion(AsteriskVersion targetVersion)
{
this.targetVersion = targetVersion;
}
public String buildAction(final ManagerAction action)
{
return buildAction(action, null);
}
@SuppressWarnings("unchecked")
public String buildAction(final ManagerAction action, final String internalActionId)
{
StringBuffer sb = new StringBuffer();
sb.append("action: ");
sb.append(action.getAction());
sb.append(LINE_SEPARATOR);
if (internalActionId != null)
{
sb.append("actionid: ");
sb.append(ManagerUtil.addInternalActionId(action.getActionId(), internalActionId));
sb.append(LINE_SEPARATOR);
}
else if (action.getActionId() != null)
{
sb.append("actionid: ");
sb.append(action.getActionId());
sb.append(LINE_SEPARATOR);
}
/*
* When using the Reflection API to get all of the getters for building
* actions to send, we ignore some of the getters
*/
Set<String> ignore = new HashSet<String>();
ignore.add("class");
ignore.add("action");
ignore.add("actionid");
// if this is a user event action, we need to grab the internal event,
// otherwise do below as normal
if (action instanceof UserEventAction)
{
UserEvent userEvent = ((UserEventAction) action).getUserEvent();
appendUserEvent(sb, userEvent);
// eventually we may want to add more Map keys for events to ignore
// when appending
appendGetters(sb, userEvent, ignore);
}
else
{
appendGetters(sb, action, ignore);
}
sb.append(LINE_SEPARATOR);
return sb.toString();
}
private void appendMap(StringBuffer sb, String key, Map<String, String> values)
{
String singularKey;
// strip plural s (i.e. use "variable: " instead of "variables: "
if (key.endsWith("s"))
{
singularKey = key.substring(0, key.length() - 1);
}
else
{
singularKey = key;
}
if (targetVersion.isAtLeast(AsteriskVersion.ASTERISK_1_2))
{
appendMap12(sb, singularKey, values);
}
else
{
appendMap10(sb, singularKey, values);
}
}
private void appendMap10(StringBuffer sb, String singularKey, Map<String, String> values)
{
Iterator<Map.Entry<String, String>> entryIterator;
sb.append(singularKey);
sb.append(": ");
entryIterator = values.entrySet().iterator();
while (entryIterator.hasNext())
{
Map.Entry entry;
entry = entryIterator.next();
sb.append(entry.getKey());
sb.append("=");
if (entry.getValue() != null)
{
sb.append(entry.getValue());
}
if (entryIterator.hasNext())
{
sb.append("|");
}
}
sb.append(LINE_SEPARATOR);
}
private void appendMap12(StringBuffer sb, String singularKey, Map<String, String> values)
{
for (Map.Entry entry : values.entrySet())
{
sb.append(singularKey);
sb.append(": ");
sb.append(entry.getKey());
sb.append("=");
if (entry.getValue() != null)
{
sb.append(entry.getValue());
}
sb.append(LINE_SEPARATOR);
}
}
private void appendString(StringBuffer sb, String key, String value)
{
sb.append(key);
sb.append(": ");
sb.append(value);
sb.append(LINE_SEPARATOR);
}
private void appendUserEvent(StringBuffer sb, UserEvent event)
{
Class clazz = event.getClass();
String className = clazz.getName();
String eventType = className.substring(className.lastIndexOf('.') + 1).toLowerCase(Locale.ENGLISH);
if (eventType.endsWith("event"))
{
eventType = eventType.substring(0, eventType.length() - "event".length());
}
appendString(sb, "UserEvent", eventType);
}
@SuppressWarnings("unchecked")
private void appendGetters(StringBuffer sb, Object action, Set<String> membersToIgnore)
{
Map<String, Method> getters = ReflectionUtil.getGetters(action.getClass());
for (String name : getters.keySet())
{
Method getter;
Object value;
if (membersToIgnore.contains(name))
continue;
getter = getters.get(name);
try
{
value = getter.invoke(action);
}
catch (Exception ex)
{
logger.error("Unable to retrieve property '" + name + "' of " + action.getClass(), ex);
continue;
}
if (value == null)
{
// continue
}
else if (value instanceof Class)
{
// continue
}
else if (value instanceof Map)
{
appendMap(sb, name, (Map) value);
}
else if (value instanceof String)
{
appendString(sb, name, (String) value);
}
else
{
appendString(sb, name, value.toString());
}
}
}
}