2016-02-09 15:12:00 -08:00
|
|
|
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
|
2015-08-04 16:07:17 -07:00
|
|
|
// This source code is licensed under the BSD-style license found in the
|
|
|
|
// LICENSE file in the root directory of this source tree. An additional grant
|
|
|
|
// of patent rights can be found in the PATENTS file in the same directory.
|
|
|
|
//
|
|
|
|
// This file implements the "bridge" between Java and C++ and enables
|
|
|
|
// calling C++ rocksdb::BackupEngine methods from the Java side.
|
|
|
|
|
|
|
|
#include <jni.h>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "include/org_rocksdb_BackupEngine.h"
|
|
|
|
#include "rocksdb/utilities/backupable_db.h"
|
|
|
|
#include "rocksjni/portal.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class: org_rocksdb_BackupEngine
|
|
|
|
* Method: open
|
2016-01-20 17:05:41 +00:00
|
|
|
* Signature: (JJ)J
|
2015-08-04 16:07:17 -07:00
|
|
|
*/
|
2016-01-20 17:05:41 +00:00
|
|
|
jlong Java_org_rocksdb_BackupEngine_open(
|
|
|
|
JNIEnv* env, jclass jcls, jlong env_handle,
|
2015-08-04 16:07:17 -07:00
|
|
|
jlong backupable_db_options_handle) {
|
|
|
|
auto* rocks_env = reinterpret_cast<rocksdb::Env*>(env_handle);
|
|
|
|
auto* backupable_db_options =
|
|
|
|
reinterpret_cast<rocksdb::BackupableDBOptions*>(
|
|
|
|
backupable_db_options_handle);
|
|
|
|
rocksdb::BackupEngine* backup_engine;
|
|
|
|
auto status = rocksdb::BackupEngine::Open(rocks_env,
|
|
|
|
*backupable_db_options, &backup_engine);
|
|
|
|
|
|
|
|
if (status.ok()) {
|
2016-01-20 17:05:41 +00:00
|
|
|
return reinterpret_cast<jlong>(backup_engine);
|
|
|
|
} else {
|
|
|
|
rocksdb::RocksDBExceptionJni::ThrowNew(env, status);
|
|
|
|
return 0;
|
2015-08-04 16:07:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class: org_rocksdb_BackupEngine
|
|
|
|
* Method: createNewBackup
|
|
|
|
* Signature: (JJZ)V
|
|
|
|
*/
|
|
|
|
void Java_org_rocksdb_BackupEngine_createNewBackup(
|
|
|
|
JNIEnv* env, jobject jbe, jlong jbe_handle, jlong db_handle,
|
|
|
|
jboolean jflush_before_backup) {
|
|
|
|
auto* db = reinterpret_cast<rocksdb::DB*>(db_handle);
|
|
|
|
auto* backup_engine = reinterpret_cast<rocksdb::BackupEngine*>(jbe_handle);
|
|
|
|
auto status = backup_engine->CreateNewBackup(db,
|
|
|
|
static_cast<bool>(jflush_before_backup));
|
|
|
|
|
|
|
|
if (status.ok()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
rocksdb::RocksDBExceptionJni::ThrowNew(env, status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class: org_rocksdb_BackupEngine
|
|
|
|
* Method: getBackupInfo
|
|
|
|
* Signature: (J)Ljava/util/List;
|
|
|
|
*/
|
|
|
|
jobject Java_org_rocksdb_BackupEngine_getBackupInfo(
|
|
|
|
JNIEnv* env, jobject jbe, jlong jbe_handle) {
|
|
|
|
auto* backup_engine = reinterpret_cast<rocksdb::BackupEngine*>(jbe_handle);
|
|
|
|
std::vector<rocksdb::BackupInfo> backup_infos;
|
|
|
|
backup_engine->GetBackupInfo(&backup_infos);
|
|
|
|
return rocksdb::BackupInfoListJni::getBackupInfo(env, backup_infos);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class: org_rocksdb_BackupEngine
|
|
|
|
* Method: getCorruptedBackups
|
|
|
|
* Signature: (J)[I
|
|
|
|
*/
|
|
|
|
jintArray Java_org_rocksdb_BackupEngine_getCorruptedBackups(
|
|
|
|
JNIEnv* env, jobject jbe, jlong jbe_handle) {
|
|
|
|
auto* backup_engine = reinterpret_cast<rocksdb::BackupEngine*>(jbe_handle);
|
|
|
|
std::vector<rocksdb::BackupID> backup_ids;
|
|
|
|
backup_engine->GetCorruptedBackups(&backup_ids);
|
|
|
|
// store backupids in int array
|
2016-01-28 15:44:31 +01:00
|
|
|
std::vector<jint> int_backup_ids(backup_ids.begin(), backup_ids.end());
|
2017-02-27 16:26:12 -08:00
|
|
|
|
2015-08-04 16:07:17 -07:00
|
|
|
// Store ints in java array
|
|
|
|
// Its ok to loose precision here (64->32)
|
2016-01-28 15:44:31 +01:00
|
|
|
jsize ret_backup_ids_size = static_cast<jsize>(backup_ids.size());
|
2017-02-27 16:26:12 -08:00
|
|
|
jintArray ret_backup_ids = env->NewIntArray(ret_backup_ids_size);
|
|
|
|
if(ret_backup_ids == nullptr) {
|
|
|
|
// exception thrown: OutOfMemoryError
|
|
|
|
return nullptr;
|
|
|
|
}
|
2015-08-04 16:07:17 -07:00
|
|
|
env->SetIntArrayRegion(ret_backup_ids, 0, ret_backup_ids_size,
|
2016-01-28 15:44:31 +01:00
|
|
|
int_backup_ids.data());
|
2015-08-04 16:07:17 -07:00
|
|
|
return ret_backup_ids;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class: org_rocksdb_BackupEngine
|
|
|
|
* Method: garbageCollect
|
|
|
|
* Signature: (J)V
|
|
|
|
*/
|
|
|
|
void Java_org_rocksdb_BackupEngine_garbageCollect(
|
|
|
|
JNIEnv* env, jobject jbe, jlong jbe_handle) {
|
|
|
|
auto* backup_engine = reinterpret_cast<rocksdb::BackupEngine*>(jbe_handle);
|
|
|
|
auto status = backup_engine->GarbageCollect();
|
|
|
|
|
|
|
|
if (status.ok()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
rocksdb::RocksDBExceptionJni::ThrowNew(env, status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class: org_rocksdb_BackupEngine
|
|
|
|
* Method: purgeOldBackups
|
|
|
|
* Signature: (JI)V
|
|
|
|
*/
|
|
|
|
void Java_org_rocksdb_BackupEngine_purgeOldBackups(
|
|
|
|
JNIEnv* env, jobject jbe, jlong jbe_handle, jint jnum_backups_to_keep) {
|
|
|
|
auto* backup_engine = reinterpret_cast<rocksdb::BackupEngine*>(jbe_handle);
|
|
|
|
auto status =
|
|
|
|
backup_engine->
|
|
|
|
PurgeOldBackups(static_cast<uint32_t>(jnum_backups_to_keep));
|
|
|
|
|
|
|
|
if (status.ok()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
rocksdb::RocksDBExceptionJni::ThrowNew(env, status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class: org_rocksdb_BackupEngine
|
|
|
|
* Method: deleteBackup
|
|
|
|
* Signature: (JI)V
|
|
|
|
*/
|
|
|
|
void Java_org_rocksdb_BackupEngine_deleteBackup(
|
|
|
|
JNIEnv* env, jobject jbe, jlong jbe_handle, jint jbackup_id) {
|
|
|
|
auto* backup_engine = reinterpret_cast<rocksdb::BackupEngine*>(jbe_handle);
|
|
|
|
auto status =
|
|
|
|
backup_engine->DeleteBackup(static_cast<rocksdb::BackupID>(jbackup_id));
|
|
|
|
|
|
|
|
if (status.ok()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
rocksdb::RocksDBExceptionJni::ThrowNew(env, status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class: org_rocksdb_BackupEngine
|
|
|
|
* Method: restoreDbFromBackup
|
|
|
|
* Signature: (JILjava/lang/String;Ljava/lang/String;J)V
|
|
|
|
*/
|
|
|
|
void Java_org_rocksdb_BackupEngine_restoreDbFromBackup(
|
|
|
|
JNIEnv* env, jobject jbe, jlong jbe_handle, jint jbackup_id,
|
|
|
|
jstring jdb_dir, jstring jwal_dir, jlong jrestore_options_handle) {
|
|
|
|
auto* backup_engine = reinterpret_cast<rocksdb::BackupEngine*>(jbe_handle);
|
2017-02-27 16:26:12 -08:00
|
|
|
const char* db_dir = env->GetStringUTFChars(jdb_dir, nullptr);
|
|
|
|
if(db_dir == nullptr) {
|
|
|
|
// exception thrown: OutOfMemoryError
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const char* wal_dir = env->GetStringUTFChars(jwal_dir, nullptr);
|
|
|
|
if(wal_dir == nullptr) {
|
|
|
|
// exception thrown: OutOfMemoryError
|
|
|
|
env->ReleaseStringUTFChars(jdb_dir, db_dir);
|
|
|
|
return;
|
|
|
|
}
|
2015-08-04 16:07:17 -07:00
|
|
|
auto* restore_options =
|
|
|
|
reinterpret_cast<rocksdb::RestoreOptions*>(jrestore_options_handle);
|
|
|
|
auto status =
|
|
|
|
backup_engine->RestoreDBFromBackup(
|
|
|
|
static_cast<rocksdb::BackupID>(jbackup_id), db_dir, wal_dir,
|
|
|
|
*restore_options);
|
2017-02-27 16:26:12 -08:00
|
|
|
|
2015-08-04 16:07:17 -07:00
|
|
|
env->ReleaseStringUTFChars(jwal_dir, wal_dir);
|
|
|
|
env->ReleaseStringUTFChars(jdb_dir, db_dir);
|
|
|
|
|
|
|
|
if (status.ok()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
rocksdb::RocksDBExceptionJni::ThrowNew(env, status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class: org_rocksdb_BackupEngine
|
|
|
|
* Method: restoreDbFromLatestBackup
|
|
|
|
* Signature: (JLjava/lang/String;Ljava/lang/String;J)V
|
|
|
|
*/
|
|
|
|
void Java_org_rocksdb_BackupEngine_restoreDbFromLatestBackup(
|
|
|
|
JNIEnv* env, jobject jbe, jlong jbe_handle, jstring jdb_dir,
|
|
|
|
jstring jwal_dir, jlong jrestore_options_handle) {
|
|
|
|
auto* backup_engine = reinterpret_cast<rocksdb::BackupEngine*>(jbe_handle);
|
2017-02-27 16:26:12 -08:00
|
|
|
const char* db_dir = env->GetStringUTFChars(jdb_dir, nullptr);
|
|
|
|
if(db_dir == nullptr) {
|
|
|
|
// exception thrown: OutOfMemoryError
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const char* wal_dir = env->GetStringUTFChars(jwal_dir, nullptr);
|
|
|
|
if(wal_dir == nullptr) {
|
|
|
|
// exception thrown: OutOfMemoryError
|
|
|
|
env->ReleaseStringUTFChars(jdb_dir, db_dir);
|
|
|
|
return;
|
|
|
|
}
|
2015-08-04 16:07:17 -07:00
|
|
|
auto* restore_options =
|
|
|
|
reinterpret_cast<rocksdb::RestoreOptions*>(jrestore_options_handle);
|
|
|
|
auto status =
|
|
|
|
backup_engine->RestoreDBFromLatestBackup(db_dir, wal_dir,
|
|
|
|
*restore_options);
|
2017-02-27 16:26:12 -08:00
|
|
|
|
2015-08-04 16:07:17 -07:00
|
|
|
env->ReleaseStringUTFChars(jwal_dir, wal_dir);
|
|
|
|
env->ReleaseStringUTFChars(jdb_dir, db_dir);
|
|
|
|
|
|
|
|
if (status.ok()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
rocksdb::RocksDBExceptionJni::ThrowNew(env, status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Class: org_rocksdb_BackupEngine
|
|
|
|
* Method: disposeInternal
|
|
|
|
* Signature: (J)V
|
|
|
|
*/
|
|
|
|
void Java_org_rocksdb_BackupEngine_disposeInternal(
|
|
|
|
JNIEnv* env, jobject jbe, jlong jbe_handle) {
|
2017-02-27 16:26:12 -08:00
|
|
|
auto* be = reinterpret_cast<rocksdb::BackupEngine*>(jbe_handle);
|
|
|
|
assert(be != nullptr);
|
|
|
|
delete be;
|
2015-08-04 16:07:17 -07:00
|
|
|
}
|