com.arangodb.ArangoDB Maven / Gradle / Ivy
/*
* DISCLAIMER
*
* Copyright 2016 ArangoDB GmbH, Cologne, Germany
*
* 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.
*
* Copyright holder is ArangoDB GmbH, Cologne, Germany
*/
package com.arangodb;
import java.io.InputStream;
import java.lang.annotation.Annotation;
import java.util.Collection;
import java.util.Properties;
import javax.net.ssl.SSLContext;
import com.arangodb.entity.ArangoDBVersion;
import com.arangodb.entity.LoadBalancingStrategy;
import com.arangodb.entity.LogEntity;
import com.arangodb.entity.LogLevelEntity;
import com.arangodb.entity.Permissions;
import com.arangodb.entity.ServerRole;
import com.arangodb.entity.UserEntity;
import com.arangodb.internal.ArangoContext;
import com.arangodb.internal.ArangoDBImpl;
import com.arangodb.internal.ArangoDefaults;
import com.arangodb.internal.InternalArangoDBBuilder;
import com.arangodb.internal.http.HttpCommunication;
import com.arangodb.internal.http.HttpConnectionFactory;
import com.arangodb.internal.net.ConnectionFactory;
import com.arangodb.internal.net.Host;
import com.arangodb.internal.net.HostHandle;
import com.arangodb.internal.net.HostHandler;
import com.arangodb.internal.net.HostResolver;
import com.arangodb.internal.util.ArangoDeserializerImpl;
import com.arangodb.internal.util.ArangoSerializationFactory;
import com.arangodb.internal.util.ArangoSerializerImpl;
import com.arangodb.internal.util.DefaultArangoSerialization;
import com.arangodb.internal.velocystream.VstCommunicationSync;
import com.arangodb.internal.velocystream.VstConnectionFactorySync;
import com.arangodb.model.LogOptions;
import com.arangodb.model.UserCreateOptions;
import com.arangodb.model.UserUpdateOptions;
import com.arangodb.util.ArangoCursorInitializer;
import com.arangodb.util.ArangoDeserializer;
import com.arangodb.util.ArangoSerialization;
import com.arangodb.util.ArangoSerializer;
import com.arangodb.velocypack.VPack;
import com.arangodb.velocypack.VPackAnnotationFieldFilter;
import com.arangodb.velocypack.VPackAnnotationFieldNaming;
import com.arangodb.velocypack.VPackDeserializer;
import com.arangodb.velocypack.VPackInstanceCreator;
import com.arangodb.velocypack.VPackJsonDeserializer;
import com.arangodb.velocypack.VPackJsonSerializer;
import com.arangodb.velocypack.VPackModule;
import com.arangodb.velocypack.VPackParser;
import com.arangodb.velocypack.VPackParserModule;
import com.arangodb.velocypack.VPackSerializer;
import com.arangodb.velocypack.ValueType;
import com.arangodb.velocystream.Request;
import com.arangodb.velocystream.Response;
/**
* Central access point for applications to communicate with an ArangoDB server.
*
*
* Will be instantiated through {@link ArangoDB.Builder}
*
*
*
* ArangoDB arango = new ArangoDB.Builder().build();
* ArangoDB arango = new ArangoDB.Builder().host("127.0.0.1", 8529).build();
*
*
* @author Mark Vollmary
*/
public interface ArangoDB extends ArangoSerializationAccessor {
/**
* Builder class to build an instance of {@link ArangoDB}.
*
* @author Mark Vollmary
*/
public static class Builder extends InternalArangoDBBuilder {
private static String PROPERTY_KEY_PROTOCOL = "arangodb.protocol";
protected Protocol protocol;
public Builder() {
super();
}
@Override
protected void loadProperties(final Properties properties) {
super.loadProperties(properties);
protocol = loadProtocol(properties, protocol);
}
private static Protocol loadProtocol(final Properties properties, final Protocol currentValue) {
return Protocol.valueOf(
getProperty(properties, PROPERTY_KEY_PROTOCOL, currentValue, ArangoDefaults.DEFAULT_NETWORK_PROTOCOL)
.toUpperCase());
}
public Builder useProtocol(final Protocol protocol) {
this.protocol = protocol;
return this;
}
@Override
public Builder loadProperties(final InputStream in) throws ArangoDBException {
super.loadProperties(in);
return this;
}
/**
* Adds a host to connect to. Multiple hosts can be added to provide fallbacks.
*
* @param host
* address of the host
* @param port
* port of the host
* @return {@link ArangoDB.Builder}
*/
public Builder host(final String host, final int port) {
setHost(host, port);
return this;
}
/**
* Sets the timeout in milliseconds.
*
*
* For network protocol {@link Protocol#VST} it is used as socket timeout when opening a VecloyStream
* connection.
*
*
* For network protocol {@link Protocol#HTTP_JSON} and {@link Protocol#HTTP_VPACK} it is used as timeout for a
* single request.
*
*
* @param timeout
* timeout in milliseconds
* @return {@link ArangoDB.Builder}
*/
public Builder timeout(final Integer timeout) {
setTimeout(timeout);
return this;
}
/**
* Sets the username to use for authentication.
*
* @param user
* the user in the database (default: {@code root})
* @return {@link ArangoDB.Builder}
*/
public Builder user(final String user) {
setUser(user);
return this;
}
/**
* Sets the password for the user for authentication.
*
* @param password
* the password of the user in the database (default: {@code null})
* @return {@link ArangoDB.Builder}
*/
public Builder password(final String password) {
setPassword(password);
return this;
}
/**
* If set to {@code true} SSL will be used when connecting to an ArangoDB server.
*
* @param useSsl
* whether or not use SSL (default: {@code false})
* @return {@link ArangoDB.Builder}
*/
public Builder useSsl(final Boolean useSsl) {
setUseSsl(useSsl);
return this;
}
/**
* Sets the SSL context to be used when {@code true} is passed through {@link #useSsl(Boolean)}.
*
* @param sslContext
* SSL context to be used
* @return {@link ArangoDB.Builder}
*/
public Builder sslContext(final SSLContext sslContext) {
setSslContext(sslContext);
return this;
}
/**
* Sets the chunk size when {@link Protocol#VST} is used.
*
* @param chunksize
* size of a chunk in bytes
* @return {@link ArangoDB.Builder}
*/
public Builder chunksize(final Integer chunksize) {
setChunksize(chunksize);
return this;
}
/**
* Sets the maximum number of connections the built in connection pool will open per host.
*
*
* Defaults:
*
*
*
* {@link Protocol#VST} == 1
* {@link Protocol#HTTP_JSON} == 20
* {@link Protocol#HTTP_VPACK} == 20
*
*
* @param maxConnections
* max number of connections
* @return {@link ArangoDB.Builder}
*/
public Builder maxConnections(final Integer maxConnections) {
setMaxConnections(maxConnections);
return this;
}
/**
* Set the maximum time to life of a connection. After this time the connection will be closed automatically.
*
* @param connectionTtl
* the maximum time to life of a connection.
* @return {@link ArangoDB.Builder}
*/
public Builder connectionTtl(final Long connectionTtl) {
setConnectionTtl(connectionTtl);
return this;
}
/**
* Whether or not the driver should acquire a list of available coordinators in an ArangoDB cluster or a single
* server with active failover.
*
*
* The host list will be used for failover and load balancing.
*
*
* @param acquireHostList
* whether or not automatically acquire a list of available hosts (default: false)
* @return {@link ArangoDB.Builder}
*/
public Builder acquireHostList(final Boolean acquireHostList) {
setAcquireHostList(acquireHostList);
return this;
}
/**
* Sets the load balancing strategy to be used in an ArangoDB cluster setup.
*
* @param loadBalancingStrategy
* the load balancing strategy to be used (default: {@link LoadBalancingStrategy#NONE}
* @return {@link ArangoDB.Builder}
*/
public Builder loadBalancingStrategy(final LoadBalancingStrategy loadBalancingStrategy) {
setLoadBalancingStrategy(loadBalancingStrategy);
return this;
}
/**
* Register a custom {@link VPackSerializer} for a specific type to be used within the internal serialization
* process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param clazz
* the type the serializer should be registered for
* @param serializer
* serializer to register
* @return {@link ArangoDB.Builder}
*/
public Builder registerSerializer(final Class clazz, final VPackSerializer serializer) {
vpackBuilder.registerSerializer(clazz, serializer);
return this;
}
/**
* Register a special serializer for a member class which can only be identified by its enclosing class.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param clazz
* the type of the enclosing class
* @param serializer
* serializer to register
* @return {@link ArangoDB.Builder}
*/
public Builder registerEnclosingSerializer(final Class clazz, final VPackSerializer serializer) {
vpackBuilder.registerEnclosingSerializer(clazz, serializer);
return this;
}
/**
* Register a custom {@link VPackDeserializer} for a specific type to be used within the internal serialization
* process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param clazz
* the type the serializer should be registered for
* @param deserializer
* @return {@link ArangoDB.Builder}
*/
public Builder registerDeserializer(final Class clazz, final VPackDeserializer deserializer) {
vpackBuilder.registerDeserializer(clazz, deserializer);
return this;
}
/**
* Register a custom {@link VPackInstanceCreator} for a specific type to be used within the internal
* serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param clazz
* the type the instance creator should be registered for
* @param creator
* @return {@link ArangoDB.Builder}
*/
public Builder registerInstanceCreator(final Class clazz, final VPackInstanceCreator creator) {
vpackBuilder.registerInstanceCreator(clazz, creator);
return this;
}
/**
* Register a custom {@link VPackJsonDeserializer} for a specific type to be used within the internal
* serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param type
* the type the serializer should be registered for
* @param deserializer
* @return {@link ArangoDB.Builder}
*/
public Builder registerJsonDeserializer(final ValueType type, final VPackJsonDeserializer deserializer) {
vpackParserBuilder.registerDeserializer(type, deserializer);
return this;
}
/**
* Register a custom {@link VPackJsonDeserializer} for a specific type and attribute name to be used within the
* internal serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param attribute
* @param type
* the type the serializer should be registered for
* @param deserializer
* @return {@link ArangoDB.Builder}
*/
public Builder registerJsonDeserializer(
final String attribute,
final ValueType type,
final VPackJsonDeserializer deserializer) {
vpackParserBuilder.registerDeserializer(attribute, type, deserializer);
return this;
}
/**
* Register a custom {@link VPackJsonSerializer} for a specific type to be used within the internal
* serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param clazz
* the type the serializer should be registered for
* @param serializer
* @return {@link ArangoDB.Builder}
*/
public Builder registerJsonSerializer(final Class clazz, final VPackJsonSerializer serializer) {
vpackParserBuilder.registerSerializer(clazz, serializer);
return this;
}
/**
* Register a custom {@link VPackJsonSerializer} for a specific type and attribute name to be used within the
* internal serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param attribute
* @param clazz
* the type the serializer should be registered for
* @param serializer
* @return {@link ArangoDB.Builder}
*/
public Builder registerJsonSerializer(
final String attribute,
final Class clazz,
final VPackJsonSerializer serializer) {
vpackParserBuilder.registerSerializer(attribute, clazz, serializer);
return this;
}
/**
* Register a custom {@link VPackAnnotationFieldFilter} for a specific type to be used within the internal
* serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param type
* the type the serializer should be registered for
* @param fieldFilter
* @return {@link ArangoDB.Builder}
*/
public Builder annotationFieldFilter(
final Class type,
final VPackAnnotationFieldFilter fieldFilter) {
vpackBuilder.annotationFieldFilter(type, fieldFilter);
return this;
}
/**
* Register a custom {@link VPackAnnotationFieldNaming} for a specific type to be used within the internal
* serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param type
* the type the serializer should be registered for
* @param fieldNaming
* @return {@link ArangoDB.Builder}
*/
public Builder annotationFieldNaming(
final Class type,
final VPackAnnotationFieldNaming fieldNaming) {
vpackBuilder.annotationFieldNaming(type, fieldNaming);
return this;
}
/**
* Register a {@link VPackModule} to be used within the internal serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param module
* module to register
* @return {@link ArangoDB.Builder}
*/
public Builder registerModule(final VPackModule module) {
vpackBuilder.registerModule(module);
return this;
}
/**
* Register a list of {@link VPackModule} to be used within the internal serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param modules
* modules to register
* @return {@link ArangoDB.Builder}
*/
public Builder registerModules(final VPackModule... modules) {
vpackBuilder.registerModules(modules);
return this;
}
/**
* Register a {@link VPackParserModule} to be used within the internal serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param module
* module to register
* @return {@link ArangoDB.Builder}
*/
public Builder registerJsonModule(final VPackParserModule module) {
vpackParserBuilder.registerModule(module);
return this;
}
/**
* Register a list of {@link VPackParserModule} to be used within the internal serialization process.
*
*
* Attention:can not be used together with {@link #serializer(ArangoSerialization)}
*
*
* @param modules
* modules to register
* @return {@link ArangoDB.Builder}
*/
public Builder registerJsonModules(final VPackParserModule... modules) {
vpackParserBuilder.registerModules(modules);
return this;
}
/**
* Replace the built-in serializer with the given serializer.
*
*
* ATTENTION!: Use at your own risk
*
* @param serializer
* custom serializer
* @deprecated use {@link #serializer(ArangoSerialization)} instead
* @return {@link ArangoDB.Builder}
*/
@Deprecated
public Builder setSerializer(final ArangoSerializer serializer) {
serializer(serializer);
return this;
}
/**
* Replace the built-in deserializer with the given deserializer.
*
*
* ATTENTION!: Use at your own risk
*
* @param deserializer
* custom deserializer
* @deprecated use {@link #serializer(ArangoSerialization)} instead
* @return {@link ArangoDB.Builder}
*/
@Deprecated
public Builder setDeserializer(final ArangoDeserializer deserializer) {
deserializer(deserializer);
return this;
}
/**
* Replace the built-in serializer/deserializer with the given one.
*
*
* ATTENTION!: Any registered custom serializer/deserializer or module will be ignored.
*
* @param serialization
* custom serializer/deserializer
* @return {@link ArangoDB.Builder}
*/
public Builder serializer(final ArangoSerialization serialization) {
setSerializer(serialization);
return this;
}
/**
* Returns an instance of {@link ArangoDB}.
*
* @return {@link ArangoDB}
*/
public synchronized ArangoDB build() {
if (hosts.isEmpty()) {
hosts.add(host);
}
final VPack vpacker = vpackBuilder.serializeNullValues(false).build();
final VPack vpackerNull = vpackBuilder.serializeNullValues(true).build();
final VPackParser vpackParser = vpackParserBuilder.build();
final ArangoSerializer serializerTemp = serializer != null ? serializer
: new ArangoSerializerImpl(vpacker, vpackerNull, vpackParser);
final ArangoDeserializer deserializerTemp = deserializer != null ? deserializer
: new ArangoDeserializerImpl(vpackerNull, vpackParser);
final DefaultArangoSerialization internal = new DefaultArangoSerialization(serializerTemp,
deserializerTemp);
final ArangoSerialization custom = customSerializer != null ? customSerializer : internal;
final ArangoSerializationFactory util = new ArangoSerializationFactory(internal, custom);
final int max = maxConnections != null ? Math.max(1, maxConnections)
: protocol == Protocol.VST ? ArangoDefaults.MAX_CONNECTIONS_VST_DEFAULT
: ArangoDefaults.MAX_CONNECTIONS_HTTP_DEFAULT;
final ConnectionFactory connectionFactory = (protocol == null || Protocol.VST == protocol)
? new VstConnectionFactorySync(host, timeout, connectionTtl, useSsl, sslContext)
: new HttpConnectionFactory(timeout, user, password, useSsl, sslContext, custom, protocol,
connectionTtl);
final Collection hostList = createHostList(max, connectionFactory);
final HostResolver hostResolver = createHostResolver(hostList, max, connectionFactory);
final HostHandler hostHandler = createHostHandler(hostResolver);
return new ArangoDBImpl(
new VstCommunicationSync.Builder(hostHandler).timeout(timeout).user(user).password(password)
.useSsl(useSsl).sslContext(sslContext).chunksize(chunksize).maxConnections(maxConnections)
.connectionTtl(connectionTtl),
new HttpCommunication.Builder(hostHandler), util, protocol, hostResolver, new ArangoContext());
}
}
/**
* Releases all connections to the server and clear the connection pool.
*
* @throws ArangoDBException
*/
void shutdown() throws ArangoDBException;
/**
* Returns a {@code ArangoDatabase} instance for the {@code _system} database.
*
* @return database handler
*/
ArangoDatabase db();
/**
* Returns a {@code ArangoDatabase} instance for the given database name.
*
* @param name
* Name of the database
* @return database handler
*/
ArangoDatabase db(String name);
/**
* Creates a new database with the given name.
*
* @see API
* Documentation
* @param name
* Name of the database to create
* @return true if the database was created successfully.
* @throws ArangoDBException
*/
Boolean createDatabase(String name) throws ArangoDBException;
/**
* Retrieves a list of all existing databases
*
* @see API
* Documentation
* @return a list of all existing databases
* @throws ArangoDBException
*/
Collection getDatabases() throws ArangoDBException;
/**
* Retrieves a list of all databases the current user can access
*
* @see API
* Documentation
* @return a list of all databases the current user can access
* @throws ArangoDBException
*/
Collection getAccessibleDatabases() throws ArangoDBException;
/**
* List available database to the specified user
*
* @see API
* Documentation
* @param user
* The name of the user for which you want to query the databases
* @return
* @throws ArangoDBException
*/
Collection getAccessibleDatabasesFor(String user) throws ArangoDBException;
/**
* Returns the server name and version number.
*
* @see API
* Documentation
* @return the server version, number
* @throws ArangoDBException
*/
ArangoDBVersion getVersion() throws ArangoDBException;
/**
* Returns the server role.
*
* @return the server role
* @throws ArangoDBException
*/
ServerRole getRole() throws ArangoDBException;
/**
* Create a new user. This user will not have access to any database. You need permission to the _system database in
* order to execute this call.
*
* @see API Documentation
* @param user
* The name of the user
* @param passwd
* The user password
* @return information about the user
* @throws ArangoDBException
*/
UserEntity createUser(String user, String passwd) throws ArangoDBException;
/**
* Create a new user. This user will not have access to any database. You need permission to the _system database in
* order to execute this call.
*
* @see API Documentation
* @param user
* The name of the user
* @param passwd
* The user password
* @param options
* Additional options, can be null
* @return information about the user
* @throws ArangoDBException
*/
UserEntity createUser(String user, String passwd, UserCreateOptions options) throws ArangoDBException;
/**
* Removes an existing user, identified by user. You need access to the _system database.
*
* @see API Documentation
* @param user
* The name of the user
* @throws ArangoDBException
*/
void deleteUser(String user) throws ArangoDBException;
/**
* Fetches data about the specified user. You can fetch information about yourself or you need permission to the
* _system database in order to execute this call.
*
* @see API Documentation
* @param user
* The name of the user
* @return information about the user
* @throws ArangoDBException
*/
UserEntity getUser(String user) throws ArangoDBException;
/**
* Fetches data about all users. You can only execute this call if you have access to the _system database.
*
* @see API
* Documentation
* @return informations about all users
* @throws ArangoDBException
*/
Collection getUsers() throws ArangoDBException;
/**
* Partially updates the data of an existing user. The name of an existing user must be specified in user. You can
* only change the password of your self. You need access to the _system database to change the active flag.
*
* @see API Documentation
* @param user
* The name of the user
* @param options
* Properties of the user to be changed
* @return information about the user
* @throws ArangoDBException
*/
UserEntity updateUser(String user, UserUpdateOptions options) throws ArangoDBException;
/**
* Replaces the data of an existing user. The name of an existing user must be specified in user. You can only
* change the password of your self. You need access to the _system database to change the active flag.
*
* @see API
* Documentation
* @param user
* The name of the user
* @param options
* Additional properties of the user, can be null
* @return information about the user
* @throws ArangoDBException
*/
UserEntity replaceUser(String user, UserUpdateOptions options) throws ArangoDBException;
/**
* Sets the default access level for databases for the user {@code user}. You need permission to the _system
* database in order to execute this call.
*
* @param user
* The name of the user
* @param permissions
* The permissions the user grant
* @since ArangoDB 3.2.0
* @throws ArangoDBException
*/
void grantDefaultDatabaseAccess(String user, Permissions permissions) throws ArangoDBException;
/**
* Sets the default access level for collections for the user {@code user}. You need permission to the _system
* database in order to execute this call.
*
* @param user
* The name of the user
* @param permissions
* The permissions the user grant
* @since ArangoDB 3.2.0
* @throws ArangoDBException
*/
void grantDefaultCollectionAccess(String user, Permissions permissions) throws ArangoDBException;
/**
* Generic Execute. Use this method to execute custom FOXX services.
*
* @param request
* VelocyStream request
* @return VelocyStream response
* @throws ArangoDBException
*/
Response execute(Request request) throws ArangoDBException;
/**
* Generic Execute. Use this method to execute custom FOXX services.
*
* @param request
* VelocyStream request
* @param hostHandle
* Used to stick to a specific host when using {@link LoadBalancingStrategy#ROUND_ROBIN}
* @return VelocyStream response
* @throws ArangoDBException
*/
Response execute(Request request, HostHandle hostHandle) throws ArangoDBException;
/**
* Returns fatal, error, warning or info log messages from the server's global log.
*
* @see API
* Documentation
* @param options
* Additional options, can be null
* @return the log messages
* @throws ArangoDBException
*/
LogEntity getLogs(LogOptions options) throws ArangoDBException;
/**
* Returns the server's current loglevel settings.
*
* @return the server's current loglevel settings
* @since ArangoDB 3.1.0
* @throws ArangoDBException
*/
LogLevelEntity getLogLevel() throws ArangoDBException;
/**
* Modifies and returns the server's current loglevel settings.
*
* @param entity
* loglevel settings
* @return the server's current loglevel settings
* @since ArangoDB 3.1.0
* @throws ArangoDBException
*/
LogLevelEntity setLogLevel(LogLevelEntity entity) throws ArangoDBException;
/**
* Attention: Please do not use!
*
* @param cursorInitializer
* @return ArangoDB
*/
ArangoDB _setCursorInitializer(ArangoCursorInitializer cursorInitializer);
}