Skip to content

Commit

Permalink
Expose BerkeleyJE configs and setup default lock timeout to 30 second…
Browse files Browse the repository at this point in the history
…s [tp-tests]

Related to #1623 and #4425

Signed-off-by: Oleksandr Porunov <alexandr.porunov@gmail.com>
  • Loading branch information
porunov committed Oct 17, 2024
1 parent 9f09767 commit 67ee2d2
Show file tree
Hide file tree
Showing 9 changed files with 193 additions and 32 deletions.
17 changes: 17 additions & 0 deletions docs/changelog.md
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,23 @@ For more information on features and bug fixes in 1.1.0, see the GitHub mileston
* [JanusGraph zip](https://github.com/JanusGraph/janusgraph/releases/download/v1.1.0/janusgraph-1.1.0.zip)
* [JanusGraph zip with embedded Cassandra and ElasticSearch](https://github.com/JanusGraph/janusgraph/releases/download/v1.1.0/janusgraph-full-1.1.0.zip)

#### Upgrade Instructions

##### BerkeleyJE lock timeout set to 30 seconds instead of default 500 ms

Many users reported that in multi-thread environment the default lock timeout of `500 ms` is not enough and often lead
to lock timeout errors. Thus, the new configuration is set to `30000 ms`.
It's possible to switch back to previous settings by setting configuration option
`storage.berkeleyje.ext.je.lock.timeout` to `500 ms`.

##### BerkeleyJE ability to overwrite arbitrary settings applied at `EnvironmentConfig` creation

The new namespace `storage.berkeleyje.ext` now allows to set custom configurations which were not directly exposed by
JanusGraph.
The full list of possible setting is available inside the Java class `com.sleepycat.je.EnvironmentConfig`.
All configurations values should be specified as `String` and be formated the same as specified in the official sleepycat
[documentation](https://docs.oracle.com/cd/E17277_02/html/java/com/sleepycat/je/EnvironmentConfig.html).

### Version 1.0.1 (Release Date: ???)

/// tab | Maven
Expand Down
12 changes: 12 additions & 0 deletions docs/configs/janusgraph-cfg.md
Original file line number Diff line number Diff line change
Expand Up @@ -423,6 +423,18 @@ BerkeleyDB JE configuration options
| storage.berkeleyje.lock-mode | The BDB record lock mode used for read operations | String | LockMode.DEFAULT | MASKABLE |
| storage.berkeleyje.shared-cache | If true, the shared cache is used for all graph instances | Boolean | true | MASKABLE |

### storage.berkeleyje.ext
Overrides for arbitrary settings applied at `EnvironmentConfig` creation.
The full list of possible setting is available inside the Java class `com.sleepycat.je.EnvironmentConfig`. All configurations values should be specified as `String` and be formated the same as specified in the following [documentation](https://docs.oracle.com/cd/E17277_02/html/java/com/sleepycat/je/EnvironmentConfig.html).
Notice, for compatibility reasons, it's allowed to use `-` character instead of `.` for config keys. All dashes will be replaced by dots when passing those keys to `EnvironmentConfig`.


| Name | Description | Datatype | Default Value | Mutability |
| ---- | ---- | ---- | ---- | ---- |
| storage.berkeleyje.ext.je-lock-timeout | Lock timeout configuration. `0` disabled lock timeout completely. To set lock timeout via this configuration it's required to use String formated time representation. For example: `500 ms`, `5 min`, etc.
See information about value constraints in the official [sleepycat documentation](https://docs.oracle.com/cd/E17277_02/html/java/com/sleepycat/je/EnvironmentConfig.html#LOCK_TIMEOUT).
Notice, this option can be specified as `storage.berkeleyje.ext.je.lock.timeout` which will be treated the same as this configuration option. | String | 30000 ms | MASKABLE |

### storage.cql
CQL storage backend options

Expand Down
46 changes: 46 additions & 0 deletions docs/storage-backend/bdb.md
Original file line number Diff line number Diff line change
Expand Up @@ -85,3 +85,49 @@ In order to not run out of memory, it is advised to disable transactions
transactions enabled requires BerkeleyDB to acquire read locks on the
data it is reading. When iterating over the entire graph, these read
locks can easily require more memory than is available.

## Additional BerkeleyDB JE configuration options

It's possible to set additional BerkeleyDB JE configuration which are not
directly exposed by JanusGraph by leveraging `storage.berkeleyje.ext`
namespace.

JanusGraph iterates over all properties prefixed with
`storage.berkeleyje.ext.`. It strips the prefix from each property key.
Any dash character (`-`) wil be replaced by dot character (`.`) in the
remainder of the stripped key. The final string will be interpreted as a parameter
ke for `com.sleepycat.je.EnvironmentConfig`.
Thus, both options `storage.berkeleyje.ext.je.lock.timeout` and
`storage.berkeleyje.ext.je-lock-timeout` will be treated
the same (as `storage.berkeleyje.ext.je.lock.timeout`).
The value associated with the key is not modified.
This allows embedding arbitrary settings in JanusGraph’s properties. Here’s an
example configuration fragment that customizes three BerkeleyDB settings
using the `storage.berkeleyje.ext.` config mechanism:

```properties
storage.backend=berkeleyje
storage.berkeleyje.ext.je.lock.timeout=5000 ms
storage.berkeleyje.ext.je.lock.deadlockDetect=false
storage.berkeleyje.ext.je.txn.timeout=5000 ms
storage.berkeleyje.ext.je.log.fileMax=100000000
```

## Deadlock troubleshooting

In concurrent environment deadlocks are possible when using BerkeleyDB JE storage
backend.
It may be complicated to deal with deadlocks in use-cases when multiple threads are
modifying same vertices (including edges creation between affected vertices).
More insights on this topic can be found in the GitHub issue
[#1623](https://github.com/JanusGraph/janusgraph/issues/1623).

Some users suggest the following configuration to deal with deadlocks:
```properties
storage.berkeleyje.isolation-level=READ_UNCOMMITTED
storage.berkeleyje.lock-mode=LockMode.READ_UNCOMMITTED
storage.berkeleyje.ext.je.lock.timeout=0
storage.lock.wait-time=5000
ids.authority.wait-time=2000
tx.max-commit-time=30000
```
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,7 @@
import org.janusgraph.graphdb.configuration.GraphDatabaseConfiguration;
import org.janusgraph.graphdb.configuration.PreInitializeConfigOptions;
import org.janusgraph.graphdb.transaction.TransactionConfiguration;
import org.janusgraph.util.system.ConfigurationUtil;
import org.janusgraph.util.system.IOUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
Expand Down Expand Up @@ -93,6 +94,28 @@ public class BerkeleyJEStoreManager extends LocalStoreManager implements Ordered
ConfigOption.Type.MASKABLE, String.class,
IsolationLevel.REPEATABLE_READ.toString(), disallowEmpty(String.class));

public static final ConfigNamespace BERKELEY_EXTRAS_NS =
new ConfigNamespace(BERKELEY_NS, "ext", "Overrides for arbitrary settings applied at `EnvironmentConfig` creation.\n" +
"The full list of possible setting is available inside the Java class `com.sleepycat.je.EnvironmentConfig`. " +
"All configurations values should be specified as `String` and be formated the same as specified in the following " +
"[documentation](https://docs.oracle.com/cd/E17277_02/html/java/com/sleepycat/je/EnvironmentConfig.html).\n" +
"Notice, for compatibility reasons, it's allowed to use `-` character instead of `.` for config keys. All dashes will " +
"be replaced by dots when passing those keys to `EnvironmentConfig`.");

// This setting isn't used directly in Java, but this setting will be picked up indirectly during parsing of the
// subset configuration of `BERKELEY_EXTRAS_NS` namespace
public static final ConfigOption<String> EXT_LOCK_TIMEOUT =
new ConfigOption<>(BERKELEY_EXTRAS_NS, toJanusGraphConfigKey(EnvironmentConfig.LOCK_TIMEOUT),
String.format("Lock timeout configuration. `0` disabled lock timeout completely. " +
"To set lock timeout via this configuration it's required to use " +
"String formated time representation. For example: `500 ms`, `5 min`, etc. \nSee information about value " +
"constraints in the official " +
"[sleepycat documentation](https://docs.oracle.com/cd/E17277_02/html/java/com/sleepycat/je/EnvironmentConfig.html#LOCK_TIMEOUT).\n" +
"Notice, this option can be specified as `%s` which will be treated the same as this configuration option.",
BERKELEY_EXTRAS_NS.toStringWithoutRoot()+"."+EnvironmentConfig.LOCK_TIMEOUT
),
ConfigOption.Type.MASKABLE, String.class, "30000 ms", disallowEmpty(String.class));

private final ConcurrentMap<String, BerkeleyJEKeyValueStore> stores;

protected volatile Environment environment;
Expand Down Expand Up @@ -143,6 +166,9 @@ private synchronized void initialize() throws BackendException {
envConfig.setConfigParam(EnvironmentConfig.ENV_RUN_CLEANER, "false");
}

Map<String, String> extraSettings = getSettingsFromJanusGraphConf(storageConfig);
extraSettings.forEach((key, value) -> envConfig.setConfigParam(toBerkeleyConfigKey(key), value));

// Open the environment
environment = new Environment(directory, envConfig);

Expand All @@ -153,7 +179,23 @@ private synchronized void initialize() throws BackendException {
} catch (DatabaseException e) {
throw new PermanentBackendException("Error during BerkeleyJE initialization: ", e);
}
}

public static String toBerkeleyConfigKey(String janusGraphConfigKey){
return janusGraphConfigKey.replace("-", ".");

Check warning on line 185 in janusgraph-berkeleyje/src/main/java/org/janusgraph/diskstorage/berkeleyje/BerkeleyJEStoreManager.java

View check run for this annotation

Codecov / codecov/patch

janusgraph-berkeleyje/src/main/java/org/janusgraph/diskstorage/berkeleyje/BerkeleyJEStoreManager.java#L185

Added line #L185 was not covered by tests
}

public static String toJanusGraphConfigKey(String berkeleyConfigKey){
return berkeleyConfigKey.replace(".", "-");
}

static Map<String, String> getSettingsFromJanusGraphConf(Configuration config) {
final Map<String, String> settings = ConfigurationUtil.getSettingsFromJanusGraphConf(config, BERKELEY_EXTRAS_NS);
if(log.isDebugEnabled()){
settings.forEach((key, val) -> log.debug("[BERKELEY ext.* cfg] Set {}: {}", key, val));
log.debug("Loaded {} settings from the {} JanusGraph config namespace", settings.size(), BERKELEY_EXTRAS_NS);

Check warning on line 196 in janusgraph-berkeleyje/src/main/java/org/janusgraph/diskstorage/berkeleyje/BerkeleyJEStoreManager.java

View check run for this annotation

Codecov / codecov/patch

janusgraph-berkeleyje/src/main/java/org/janusgraph/diskstorage/berkeleyje/BerkeleyJEStoreManager.java#L195-L196

Added lines #L195 - L196 were not covered by tests
}
return settings;
}

private synchronized void reInitialize(DatabaseException exception) throws BackendException {
Expand Down Expand Up @@ -395,4 +437,8 @@ private TransactionBegin(String msg) {
super(msg);
}
}

public Environment getEnvironment(){
return environment;

Check warning on line 442 in janusgraph-berkeleyje/src/main/java/org/janusgraph/diskstorage/berkeleyje/BerkeleyJEStoreManager.java

View check run for this annotation

Codecov / codecov/patch

janusgraph-berkeleyje/src/main/java/org/janusgraph/diskstorage/berkeleyje/BerkeleyJEStoreManager.java#L442

Added line #L442 was not covered by tests
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@
package org.janusgraph.graphdb.berkeleyje;

import com.google.common.base.Preconditions;
import com.sleepycat.je.Environment;
import com.sleepycat.je.EnvironmentConfig;
import com.sleepycat.je.LockMode;
import org.janusgraph.BerkeleyStorageSetup;
import org.janusgraph.core.JanusGraphException;
Expand All @@ -27,6 +29,7 @@
import org.janusgraph.diskstorage.configuration.ConfigOption;
import org.janusgraph.diskstorage.configuration.ModifiableConfiguration;
import org.janusgraph.diskstorage.configuration.WriteConfiguration;
import org.janusgraph.diskstorage.keycolumnvalue.keyvalue.OrderedKeyValueStoreManagerAdapter;
import org.janusgraph.graphdb.JanusGraphTest;
import org.janusgraph.graphdb.configuration.GraphDatabaseConfiguration;
import org.junit.jupiter.api.Disabled;
Expand All @@ -36,6 +39,7 @@

import java.time.Duration;
import java.time.temporal.ChronoUnit;
import java.util.concurrent.TimeUnit;

import static org.janusgraph.graphdb.configuration.GraphDatabaseConfiguration.ALLOW_SETTING_VERTEX_ID;
import static org.janusgraph.graphdb.configuration.GraphDatabaseConfiguration.ALLOW_CUSTOM_VERTEX_ID_TYPES;
Expand All @@ -49,6 +53,12 @@ public class BerkeleyGraphTest extends JanusGraphTest {
private static final Logger log =
LoggerFactory.getLogger(BerkeleyGraphTest.class);

public EnvironmentConfig getCurrentEnvironmentConfig() {
BerkeleyJEStoreManager storeManager = (BerkeleyJEStoreManager) ((OrderedKeyValueStoreManagerAdapter) graph.getBackend().getStoreManager()).getManager();
Environment environment = storeManager.getEnvironment();
return environment.getConfig();
}

@Override
public WriteConfiguration getConfiguration() {
ModifiableConfiguration modifiableConfiguration = BerkeleyStorageSetup.getBerkeleyJEConfiguration();
Expand Down Expand Up @@ -162,4 +172,19 @@ public void testCannotUseCustomStringId() {
() -> clopen(option(ALLOW_SETTING_VERTEX_ID), true, option(ALLOW_CUSTOM_VERTEX_ID_TYPES), true));
assertEquals("allow-custom-vid-types is not supported for OrderedKeyValueStore", ex.getMessage());
}

@Test
public void testExposedConfigurations(){
clopen(option(BerkeleyJEStoreManager.EXT_LOCK_TIMEOUT), "4321 ms");
assertEquals(4321, getCurrentEnvironmentConfig().getLockTimeout(TimeUnit.MILLISECONDS));
close();
WriteConfiguration configuration = getConfiguration();
configuration.set(BerkeleyJEStoreManager.BERKELEY_EXTRAS_NS.toStringWithoutRoot()+"."+EnvironmentConfig.LOCK_TIMEOUT, "12345 ms");
open(configuration);
assertEquals(12345, getCurrentEnvironmentConfig().getLockTimeout(TimeUnit.MILLISECONDS));
configuration.set(BerkeleyJEStoreManager.BERKELEY_EXTRAS_NS.toStringWithoutRoot()+"."+EnvironmentConfig.ENV_IS_TRANSACTIONAL, "true");
open(configuration);
assertTrue(getCurrentEnvironmentConfig().getTransactional());
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -275,5 +275,8 @@ public static Predicate<Long> positiveLong() {
return num -> num!=null && num>0;
}

public static Predicate<Long> nonnegativeLong() {
return num -> num!=null && num>=0;
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -155,4 +155,8 @@ public List<KeyRange> getLocalKeyPartition() throws BackendException {
public String getName() {
return manager.getName();
}

public OrderedKeyValueStoreManager getManager() {
return manager;

Check warning on line 160 in janusgraph-core/src/main/java/org/janusgraph/diskstorage/keycolumnvalue/keyvalue/OrderedKeyValueStoreManagerAdapter.java

View check run for this annotation

Codecov / codecov/patch

janusgraph-core/src/main/java/org/janusgraph/diskstorage/keycolumnvalue/keyvalue/OrderedKeyValueStoreManagerAdapter.java#L160

Added line #L160 was not covered by tests
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@

package org.janusgraph.util.system;

import com.google.common.base.Joiner;
import com.google.common.base.Preconditions;
import org.apache.commons.configuration2.BaseConfiguration;
import org.apache.commons.configuration2.Configuration;
Expand All @@ -24,11 +25,14 @@
import org.apache.commons.configuration2.builder.fluent.PropertiesBuilderParameters;
import org.apache.commons.configuration2.convert.DefaultListDelimiterHandler;
import org.apache.commons.configuration2.ex.ConfigurationException;
import org.janusgraph.diskstorage.configuration.ConfigNamespace;

import java.io.File;
import java.lang.reflect.Array;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
Expand Down Expand Up @@ -169,4 +173,34 @@ private static PropertiesConfiguration loadPropertiesConfig(PropertiesBuilderPar
}
return builder.configure(newParams).getConfiguration();
}

public static Map<String, String> getSettingsFromJanusGraphConf(org.janusgraph.diskstorage.configuration.Configuration config, ConfigNamespace namespace) {

final Map<String, String> settings = new HashMap<>();

final Map<String,Object> configSub = config.getSubset(namespace);
for (Map.Entry<String,Object> entry : configSub.entrySet()) {
String key = entry.getKey();
Object val = entry.getValue();
if (null == val) continue;
if (List.class.isAssignableFrom(val.getClass())) {
// Pretty print lists using comma-separated values and no surrounding square braces for ES
List l = (List) val;
settings.put(key, Joiner.on(",").join(l));

Check warning on line 189 in janusgraph-core/src/main/java/org/janusgraph/util/system/ConfigurationUtil.java

View check run for this annotation

Codecov / codecov/patch

janusgraph-core/src/main/java/org/janusgraph/util/system/ConfigurationUtil.java#L188-L189

Added lines #L188 - L189 were not covered by tests
} else if (val.getClass().isArray()) {
// As with Lists, but now for arrays
// The Object copy[] business lets us avoid repetitive primitive array type checking and casting
Object[] copy = new Object[Array.getLength(val)];

Check warning on line 193 in janusgraph-core/src/main/java/org/janusgraph/util/system/ConfigurationUtil.java

View check run for this annotation

Codecov / codecov/patch

janusgraph-core/src/main/java/org/janusgraph/util/system/ConfigurationUtil.java#L193

Added line #L193 was not covered by tests
for (int i= 0; i < copy.length; i++) {
copy[i] = Array.get(val, i);

Check warning on line 195 in janusgraph-core/src/main/java/org/janusgraph/util/system/ConfigurationUtil.java

View check run for this annotation

Codecov / codecov/patch

janusgraph-core/src/main/java/org/janusgraph/util/system/ConfigurationUtil.java#L195

Added line #L195 was not covered by tests
}
settings.put(key, Joiner.on(",").join(copy));
} else {

Check warning on line 198 in janusgraph-core/src/main/java/org/janusgraph/util/system/ConfigurationUtil.java

View check run for this annotation

Codecov / codecov/patch

janusgraph-core/src/main/java/org/janusgraph/util/system/ConfigurationUtil.java#L197-L198

Added lines #L197 - L198 were not covered by tests
// Copy anything else unmodified
settings.put(key, val.toString());
}
}

return settings;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -14,17 +14,15 @@

package org.janusgraph.diskstorage.es;

import com.google.common.base.Joiner;
import com.google.common.base.Preconditions;
import org.janusgraph.diskstorage.configuration.Configuration;
import org.janusgraph.diskstorage.es.rest.RestClientSetup;
import org.janusgraph.util.system.ConfigurationUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.lang.reflect.Array;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
Expand Down Expand Up @@ -53,36 +51,12 @@ public Connection connect(Configuration config) throws IOException {
};

static Map<String, Object> getSettingsFromJanusGraphConf(Configuration config) {

final Map<String, Object> settings = new HashMap<>();

int keysLoaded = 0;
final Map<String,Object> configSub = config.getSubset(ElasticSearchIndex.ES_CREATE_EXTRAS_NS);
for (Map.Entry<String,Object> entry : configSub.entrySet()) {
String key = entry.getKey();
Object val = entry.getValue();
if (null == val) continue;
if (List.class.isAssignableFrom(val.getClass())) {
// Pretty print lists using comma-separated values and no surrounding square braces for ES
List l = (List) val;
settings.put(key, Joiner.on(",").join(l));
} else if (val.getClass().isArray()) {
// As with Lists, but now for arrays
// The Object copy[] business lets us avoid repetitive primitive array type checking and casting
Object[] copy = new Object[Array.getLength(val)];
for (int i= 0; i < copy.length; i++) {
copy[i] = Array.get(val, i);
}
settings.put(key, Joiner.on(",").join(copy));
} else {
// Copy anything else unmodified
settings.put(key, val.toString());
}
log.debug("[ES ext.* cfg] Set {}: {}", key, val);
keysLoaded++;
final Map<String, String> settings = ConfigurationUtil.getSettingsFromJanusGraphConf(config, ElasticSearchIndex.ES_CREATE_EXTRAS_NS);
if(log.isDebugEnabled()){
settings.forEach((key, val) -> log.debug("[ES ext.* cfg] Set {}: {}", key, val));
log.debug("Loaded {} settings from the {} JanusGraph config namespace", settings.size(), ElasticSearchIndex.ES_CREATE_EXTRAS_NS);

Check warning on line 57 in janusgraph-es/src/main/java/org/janusgraph/diskstorage/es/ElasticSearchSetup.java

View check run for this annotation

Codecov / codecov/patch

janusgraph-es/src/main/java/org/janusgraph/diskstorage/es/ElasticSearchSetup.java#L56-L57

Added lines #L56 - L57 were not covered by tests
}
log.debug("Loaded {} settings from the {} JanusGraph config namespace", keysLoaded, ElasticSearchIndex.ES_CREATE_EXTRAS_NS);
return settings;
return new HashMap<>(settings);
}

private static final Logger log = LoggerFactory.getLogger(ElasticSearchSetup.class);
Expand Down

0 comments on commit 67ee2d2

Please sign in to comment.