/*
* Copyright 2015 JBoss 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 io.apiman.gateway.engine.es;
import io.apiman.gateway.engine.async.AsyncResultImpl;
import io.apiman.gateway.engine.async.IAsyncResultHandler;
import io.apiman.gateway.engine.components.ISharedStateComponent;
import io.apiman.gateway.engine.es.beans.PrimitiveBean;
import io.searchbox.client.JestResult;
import io.searchbox.core.Delete;
import io.searchbox.core.Get;
import io.searchbox.core.Index;
import java.util.Map;
import javax.xml.namespace.QName;
import org.apache.commons.codec.binary.Base64;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
/**
* An elasticsearch implementation of the shared state component.
*
* @author eric.wittmann@redhat.com
*/
public class ESSharedStateComponent extends AbstractESComponent implements ISharedStateComponent {
private static final ObjectMapper mapper = new ObjectMapper();
static {
mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
}
/**
* Constructor.
* @param config the configuration
*/
public ESSharedStateComponent(Map<String, String> config) {
super(config);
}
/**
* @see io.apiman.gateway.engine.components.ISharedStateComponent#getProperty(java.lang.String, java.lang.String, java.lang.Object, io.apiman.gateway.engine.async.IAsyncResultHandler)
*/
@Override
public <T> void getProperty(final String namespace, final String propertyName, final T defaultValue,
final IAsyncResultHandler<T> handler) {
if (defaultValue == null) {
handler.handle(AsyncResultImpl.<T>create(new Exception("Null defaultValue is not allowed."))); //$NON-NLS-1$
return;
}
String id = getPropertyId(namespace, propertyName);
try {
Get get = new Get.Builder(getIndexName(), id).type("sharedStateProperty").build(); //$NON-NLS-1$
JestResult result = getClient().execute(get);
if (result.isSucceeded()) {
try {
T value;
if (defaultValue.getClass().isPrimitive() || defaultValue instanceof String) {
value = (T) readPrimitive(result);
} else {
value = (T) result.getSourceAsObject(defaultValue.getClass());
}
handler.handle(AsyncResultImpl.create(value));
} catch (Exception e) {
handler.handle(AsyncResultImpl.<T>create(e));
}
} else {
handler.handle(AsyncResultImpl.create(defaultValue));
}
} catch (Throwable e) {
handler.handle(AsyncResultImpl.<T>create(e));
}
}
/**
* @see io.apiman.gateway.engine.components.ISharedStateComponent#setProperty(java.lang.String, java.lang.String, java.lang.Object, io.apiman.gateway.engine.async.IAsyncResultHandler)
*/
@Override
public <T> void setProperty(final String namespace, final String propertyName, final T value, final IAsyncResultHandler<Void> handler) {
if (value == null) {
handler.handle(AsyncResultImpl.<Void>create(new Exception("Null value is not allowed."))); //$NON-NLS-1$
return;
}
String source;
try {
if (value.getClass().isPrimitive() || value instanceof String) {
PrimitiveBean pb = new PrimitiveBean();
pb.setValue(String.valueOf(value));
pb.setType(value.getClass().getName());
source = mapper.writeValueAsString(pb);
} else {
source = mapper.writeValueAsString(value);
}
} catch (Exception e) {
handler.handle(AsyncResultImpl.<Void>create(e));
return;
}
String id = getPropertyId(namespace, propertyName);
String json = source;
Index index = new Index.Builder(json).refresh(false).index(getIndexName())
.type("sharedStateProperty").id(id).build(); //$NON-NLS-1$
try {
getClient().execute(index);
handler.handle(AsyncResultImpl.create((Void) null));
} catch (Throwable e) {
handler.handle(AsyncResultImpl.<Void>create(e));
}
}
/**
* @see io.apiman.gateway.engine.components.ISharedStateComponent#clearProperty(java.lang.String, java.lang.String, io.apiman.gateway.engine.async.IAsyncResultHandler)
*/
@Override
public <T> void clearProperty(final String namespace, final String propertyName, final IAsyncResultHandler<Void> handler) {
String id = getPropertyId(namespace, propertyName);
Delete delete = new Delete.Builder(id).index(getIndexName()).type("sharedStateProperty").build(); //$NON-NLS-1$
try {
getClient().execute(delete);
handler.handle(AsyncResultImpl.create((Void) null));
} catch (Throwable e) {
handler.handle(AsyncResultImpl.<Void>create(e));
}
}
/**
* @param namespace
* @param propertyName
*/
private String getPropertyId(String namespace, String propertyName) {
String qn = new QName(namespace, propertyName).toString();
return Base64.encodeBase64String(qn.getBytes());
}
/**
* Reads a stored primitive.
* @param result
*/
protected Object readPrimitive(JestResult result) throws Exception {
PrimitiveBean pb = result.getSourceAsObject(PrimitiveBean.class);
String value = pb.getValue();
Class<?> c = Class.forName(pb.getType());
if (c == String.class) {
return value;
} else if (c == Long.class) {
return Long.parseLong(value);
} else if (c == Integer.class) {
return Integer.parseInt(value);
} else if (c == Double.class) {
return Double.parseDouble(value);
} else if (c == Boolean.class) {
return Boolean.parseBoolean(value);
} else if (c == Byte.class) {
return Byte.parseByte(value);
} else if (c == Short.class) {
return Short.parseShort(value);
} else if (c == Float.class) {
return Float.parseFloat(value);
} else {
throw new Exception("Unsupported primitive: " + c); //$NON-NLS-1$
}
}
/**
* @see io.apiman.gateway.engine.es.AbstractESComponent#getDefaultIndexName()
*/
@Override
protected String getDefaultIndexName() {
return ESConstants.GATEWAY_INDEX_NAME;
}
}