Conflicts: db/db_impl.cc db/db_test.cc db/internal_stats.cc db/internal_stats.h db/version_edit.cc db/version_edit.h db/version_set.cc include/rocksdb/options.h util/options.ccmain
						commit
						ddbd1ece88
					
				| @ -0,0 +1,103 @@ | ||||
| //  Copyright (c) 2013, Facebook, Inc.  All rights reserved.
 | ||||
| //  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.
 | ||||
| //
 | ||||
| 
 | ||||
| #pragma once | ||||
| #include <string> | ||||
| #include <vector> | ||||
| 
 | ||||
| #include "utilities/stackable_db.h" | ||||
| #include "rocksdb/status.h" | ||||
| 
 | ||||
| namespace rocksdb { | ||||
| 
 | ||||
| //
 | ||||
| // Configurable options needed for setting up a Geo database
 | ||||
| //
 | ||||
| struct GeoDBOptions { | ||||
|   // Backup info and error messages will be written to info_log
 | ||||
|   // if non-nullptr.
 | ||||
|   // Default: nullptr
 | ||||
|   Logger* info_log; | ||||
| 
 | ||||
|   explicit GeoDBOptions(Logger* _info_log = nullptr):info_log(_info_log) { } | ||||
| }; | ||||
| 
 | ||||
| //
 | ||||
| // A position in the earth's geoid
 | ||||
| //
 | ||||
| class GeoPosition { | ||||
|  public: | ||||
|   double latitude; | ||||
|   double longitude; | ||||
| 
 | ||||
|   explicit GeoPosition(double la = 0, double lo = 0) : | ||||
|     latitude(la), longitude(lo) { | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| //
 | ||||
| // Description of an object on the Geoid. It is located by a GPS location,
 | ||||
| // and is identified by the id. The value associated with this object is
 | ||||
| // an opaque string 'value'. Different objects identified by unique id's
 | ||||
| // can have the same gps-location associated with them.
 | ||||
| //
 | ||||
| class GeoObject { | ||||
|  public: | ||||
|   GeoPosition position; | ||||
|   std::string id; | ||||
|   std::string value; | ||||
| 
 | ||||
|   GeoObject() {} | ||||
| 
 | ||||
|   GeoObject(const GeoPosition& pos, const std::string& i, | ||||
|             const std::string& val) : | ||||
|     position(pos), id(i), value(val) { | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| //
 | ||||
| // Stack your DB with GeoDB to be able to get geo-spatial support
 | ||||
| //
 | ||||
| class GeoDB : public StackableDB { | ||||
|  public: | ||||
|   // GeoDBOptions have to be the same as the ones used in a previous
 | ||||
|   // incarnation of the DB
 | ||||
|   //
 | ||||
|   // GeoDB owns the pointer `DB* db` now. You should not delete it or
 | ||||
|   // use it after the invocation of GeoDB
 | ||||
|   // GeoDB(DB* db, const GeoDBOptions& options) : StackableDB(db) {}
 | ||||
|   GeoDB(DB* db, const GeoDBOptions& options) : StackableDB(db) {} | ||||
|   virtual ~GeoDB() {} | ||||
| 
 | ||||
|   // Insert a new object into the location database. The object is
 | ||||
|   // uniquely identified by the id. If an object with the same id already
 | ||||
|   // exists in the db, then the old one is overwritten by the new
 | ||||
|   // object being inserted here.
 | ||||
|   virtual Status Insert(const GeoObject& object) = 0; | ||||
| 
 | ||||
|   // Retrieve the value of the object located at the specified GPS
 | ||||
|   // location and is identified by the 'id'.
 | ||||
|   virtual Status GetByPosition(const GeoPosition& pos, | ||||
|                                const Slice& id, std::string* value) = 0; | ||||
| 
 | ||||
|   // Retrieve the value of the object identified by the 'id'. This method
 | ||||
|   // could be potentially slower than GetByPosition
 | ||||
|   virtual Status GetById(const Slice& id, GeoObject*  object) = 0; | ||||
| 
 | ||||
|   // Delete the specified object
 | ||||
|   virtual Status Remove(const Slice& id) = 0; | ||||
| 
 | ||||
|   // Returns a list of all items within a circular radius from the
 | ||||
|   // specified gps location. If 'number_of_values' is specified,
 | ||||
|   // then this call returns at most that many number of objects.
 | ||||
|   // The radius is specified in 'meters'.
 | ||||
|   virtual Status SearchRadial(const GeoPosition& pos, | ||||
|                               double radius, | ||||
|                               std::vector<GeoObject>* values, | ||||
|                               int number_of_values = INT_MAX) = 0; | ||||
| }; | ||||
| 
 | ||||
| }  // namespace rocksdb
 | ||||
| @ -0,0 +1,17 @@ | ||||
| NATIVE_JAVA_CLASSES = org.rocksdb.RocksDB
 | ||||
| NATIVE_INCLUDE = ./include
 | ||||
| ROCKSDB_JAR = rocksdbjni.jar
 | ||||
| 
 | ||||
| clean: | ||||
| 	-find . -name "*.class" -exec rm {} \;
 | ||||
| 	-find . -name "hs*.log" -exec rm {} \;
 | ||||
| 	rm -f $(ROCKSDB_JAR)
 | ||||
| 
 | ||||
| java: | ||||
| 	javac org/rocksdb/*.java
 | ||||
| 	jar -cf $(ROCKSDB_JAR) org/rocksdb/*.class
 | ||||
| 	javah -d $(NATIVE_INCLUDE) -jni $(NATIVE_JAVA_CLASSES)
 | ||||
| 
 | ||||
| sample: | ||||
| 	javac -cp $(ROCKSDB_JAR) RocksDBSample.java
 | ||||
| 	java -ea -Djava.library.path=.:../ -cp ".:./*" RocksDBSample /tmp/rocksdbjni/
 | ||||
| @ -0,0 +1,79 @@ | ||||
| // Copyright (c) 2014, Facebook, Inc.  All rights reserved.
 | ||||
| // 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.
 | ||||
| 
 | ||||
| import java.util.*; | ||||
| import java.lang.*; | ||||
| import org.rocksdb.*; | ||||
| import java.io.IOException; | ||||
| 
 | ||||
| public class RocksDBSample { | ||||
|   static { | ||||
|     System.loadLibrary("rocksdbjni"); | ||||
|   } | ||||
| 
 | ||||
|   public static void main(String[] args) { | ||||
|     if (args.length < 1) { | ||||
|       System.out.println("usage: RocksDBSample db_path"); | ||||
|       return; | ||||
|     } | ||||
|     String db_path = args[0]; | ||||
| 
 | ||||
|     System.out.println("RocksDBSample"); | ||||
| 
 | ||||
|     try { | ||||
|       RocksDB db = RocksDB.open(db_path); | ||||
|       db.put("hello".getBytes(), "world".getBytes()); | ||||
|       byte[] value = db.get("hello".getBytes()); | ||||
|       System.out.format("Get('hello') = %s\n", | ||||
|           new String(value)); | ||||
| 
 | ||||
|       for (int i = 1; i <= 9; ++i) { | ||||
|         for (int j = 1; j <= 9; ++j) { | ||||
|           db.put(String.format("%dx%d", i, j).getBytes(), | ||||
|                  String.format("%d", i * j).getBytes()); | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       for (int i = 1; i <= 9; ++i) { | ||||
|         for (int j = 1; j <= 9; ++j) { | ||||
|           System.out.format("%s ", new String(db.get( | ||||
|               String.format("%dx%d", i, j).getBytes()))); | ||||
|         } | ||||
|         System.out.println(""); | ||||
|       } | ||||
| 
 | ||||
|       value = db.get("1x1".getBytes()); | ||||
|       assert(value != null); | ||||
|       value = db.get("world".getBytes()); | ||||
|       assert(value == null); | ||||
| 
 | ||||
|       byte[] testKey = "asdf".getBytes(); | ||||
|       byte[] testValue = | ||||
|           "asdfghjkl;'?><MNBVCXZQWERTYUIOP{+_)(*&^%$#@".getBytes(); | ||||
|       db.put(testKey, testValue); | ||||
|       byte[] testResult = db.get(testKey); | ||||
|       assert(testResult != null); | ||||
|       assert(Arrays.equals(testValue, testResult)); | ||||
|       assert(new String(testValue).equals(new String(testResult))); | ||||
| 
 | ||||
|       byte[] insufficientArray = new byte[10]; | ||||
|       byte[] enoughArray = new byte[50]; | ||||
|       int len; | ||||
|       len = db.get(testKey, insufficientArray); | ||||
|       assert(len > insufficientArray.length); | ||||
|       len = db.get("asdfjkl;".getBytes(), enoughArray); | ||||
|       assert(len == RocksDB.NOT_FOUND); | ||||
|       len = db.get(testKey, enoughArray); | ||||
|       assert(len == testValue.length); | ||||
|       try { | ||||
|         db.close(); | ||||
|       } catch (IOException e) { | ||||
|         System.err.println(e); | ||||
|       } | ||||
|     } catch (RocksDBException e) { | ||||
|       System.err.println(e); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @ -0,0 +1,103 @@ | ||||
| // Copyright (c) 2013, Facebook, Inc.  All rights reserved.
 | ||||
| // 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.
 | ||||
| 
 | ||||
| package org.rocksdb; | ||||
| 
 | ||||
| import java.lang.*; | ||||
| import java.util.*; | ||||
| import java.io.Closeable; | ||||
| import java.io.IOException; | ||||
| 
 | ||||
| /** | ||||
|  * A RocksDB is a persistent ordered map from keys to values.  It is safe for | ||||
|  * concurrent access from multiple threads without any external synchronization. | ||||
|  * All methods of this class could potentially throw RocksDBException, which | ||||
|  * indicates sth wrong at the rocksdb library side and the call failed. | ||||
|  */ | ||||
| public class RocksDB implements Closeable { | ||||
|   public static final int NOT_FOUND = -1; | ||||
|   /** | ||||
|    * The factory constructor of RocksDB that opens a RocksDB instance given | ||||
|    * the path to the database. | ||||
|    * | ||||
|    * @param path the path to the rocksdb. | ||||
|    * @param status an out value indicating the status of the Open(). | ||||
|    * @return a rocksdb instance on success, null if the specified rocksdb can | ||||
|    *     not be opened. | ||||
|    */ | ||||
|   public static RocksDB open(String path) throws RocksDBException { | ||||
|     RocksDB db = new RocksDB(); | ||||
|     db.open0(path); | ||||
|     return db; | ||||
|   } | ||||
| 
 | ||||
|   @Override public void close() throws IOException { | ||||
|     if (nativeHandle != 0) { | ||||
|       close0(); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Set the database entry for "key" to "value". | ||||
|    * | ||||
|    * @param key the specified key to be inserted. | ||||
|    * @param value the value associated with the specified key. | ||||
|    */ | ||||
|   public void put(byte[] key, byte[] value) throws RocksDBException { | ||||
|     put(key, key.length, value, value.length); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get the value associated with the specified key. | ||||
|    * | ||||
|    * @param key the key to retrieve the value. | ||||
|    * @param value the out-value to receive the retrieved value. | ||||
|    * @return The size of the actual value that matches the specified | ||||
|    *     {@code key} in byte.  If the return value is greater than the | ||||
|    *     length of {@code value}, then it indicates that the size of the | ||||
|    *     input buffer {@code value} is insufficient and partial result will | ||||
|    *     be returned.  RocksDB.NOT_FOUND will be returned if the value not | ||||
|    *     found. | ||||
|    */ | ||||
|   public int get(byte[] key, byte[] value) throws RocksDBException { | ||||
|     return get(key, key.length, value, value.length); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * The simplified version of get which returns a new byte array storing | ||||
|    * the value associated with the specified input key if any.  null will be | ||||
|    * returned if the specified key is not found. | ||||
|    * | ||||
|    * @param key the key retrieve the value. | ||||
|    * @return a byte array storing the value associated with the input key if | ||||
|    *     any.  null if it does not find the specified key. | ||||
|    * | ||||
|    * @see RocksDBException | ||||
|    */ | ||||
|   public byte[] get(byte[] key) throws RocksDBException { | ||||
|     return get(key, key.length); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Private constructor. | ||||
|    */ | ||||
|   private RocksDB() { | ||||
|     nativeHandle = -1; | ||||
|   } | ||||
| 
 | ||||
|   // native methods
 | ||||
|   private native void open0(String path) throws RocksDBException; | ||||
|   private native void put( | ||||
|       byte[] key, int keyLen, | ||||
|       byte[] value, int valueLen) throws RocksDBException; | ||||
|   private native int get( | ||||
|       byte[] key, int keyLen, | ||||
|       byte[] value, int valueLen) throws RocksDBException; | ||||
|   private native byte[] get( | ||||
|       byte[] key, int keyLen) throws RocksDBException; | ||||
|   private native void close0(); | ||||
| 
 | ||||
|   private long nativeHandle; | ||||
| } | ||||
| @ -0,0 +1,24 @@ | ||||
| // Copyright (c) 2013, Facebook, Inc.  All rights reserved.
 | ||||
| // 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.
 | ||||
| 
 | ||||
| package org.rocksdb; | ||||
| 
 | ||||
| import java.lang.*; | ||||
| import java.util.*; | ||||
| 
 | ||||
| /** | ||||
|  * A RocksDBException encapsulates the error of an operation.  This exception | ||||
|  * type is used to describe an internal error from the c++ rocksdb library. | ||||
|  */ | ||||
| public class RocksDBException extends Exception { | ||||
|   /** | ||||
|    * The private construct used by a set of public static factory method. | ||||
|    * | ||||
|    * @param msg the specified error message. | ||||
|    */ | ||||
|   public RocksDBException(String msg) { | ||||
|     super(msg); | ||||
|   } | ||||
| } | ||||
| @ -0,0 +1,81 @@ | ||||
| // Copyright (c) 2014, Facebook, Inc.  All rights reserved.
 | ||||
| // 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 is designed for caching those frequently used IDs and provide
 | ||||
| // efficient portal (i.e, a set of static functions) to access java code
 | ||||
| // from c++.
 | ||||
| 
 | ||||
| #ifndef JAVA_ROCKSJNI_PORTAL_H_ | ||||
| #define JAVA_ROCKSJNI_PORTAL_H_ | ||||
| 
 | ||||
| #include <jni.h> | ||||
| #include "rocksdb/db.h" | ||||
| 
 | ||||
| namespace rocksdb { | ||||
| 
 | ||||
| // The portal class for org.rocksdb.RocksDB
 | ||||
| class RocksDBJni { | ||||
|  public: | ||||
|   // Get the java class id of org.rocksdb.RocksDB.
 | ||||
|   static jclass getJClass(JNIEnv* env) { | ||||
|     static jclass jclazz = env->FindClass("org/rocksdb/RocksDB"); | ||||
|     assert(jclazz != nullptr); | ||||
|     return jclazz; | ||||
|   } | ||||
| 
 | ||||
|   // Get the field id of the member variable of org.rocksdb.RocksDB
 | ||||
|   // that stores the pointer to rocksdb::DB.
 | ||||
|   static jfieldID getHandleFieldID(JNIEnv* env) { | ||||
|     static jfieldID fid = env->GetFieldID( | ||||
|         getJClass(env), "nativeHandle", "J"); | ||||
|     assert(fid != nullptr); | ||||
|     return fid; | ||||
|   } | ||||
| 
 | ||||
|   // Get the pointer to rocksdb::DB of the specified org.rocksdb.RocksDB.
 | ||||
|   static rocksdb::DB* getHandle(JNIEnv* env, jobject jdb) { | ||||
|     return reinterpret_cast<rocksdb::DB*>( | ||||
|         env->GetLongField(jdb, getHandleFieldID(env))); | ||||
|   } | ||||
| 
 | ||||
|   // Pass the rocksdb::DB pointer to the java side.
 | ||||
|   static void setHandle(JNIEnv* env, jobject jdb, rocksdb::DB* db) { | ||||
|     env->SetLongField( | ||||
|         jdb, getHandleFieldID(env), | ||||
|         reinterpret_cast<jlong>(db)); | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| // The portal class for org.rocksdb.RocksDBException
 | ||||
| class RocksDBExceptionJni { | ||||
|  public: | ||||
|   // Get the jclass of org.rocksdb.RocksDBException
 | ||||
|   static jclass getJClass(JNIEnv* env) { | ||||
|     static jclass jclazz = env->FindClass("org/rocksdb/RocksDBException"); | ||||
|     assert(jclazz != nullptr); | ||||
|     return jclazz; | ||||
|   } | ||||
| 
 | ||||
|   // Create and throw a java exception by converting the input
 | ||||
|   // Status to an RocksDBException.
 | ||||
|   //
 | ||||
|   // In case s.ok() is true, then this function will not throw any
 | ||||
|   // exception.
 | ||||
|   static void ThrowNew(JNIEnv* env, Status s) { | ||||
|     if (s.ok()) { | ||||
|       return; | ||||
|     } | ||||
|     jstring msg = env->NewStringUTF(s.ToString().c_str()); | ||||
|     // get the constructor id of org.rocksdb.RocksDBException
 | ||||
|     static jmethodID mid = env->GetMethodID( | ||||
|         getJClass(env), "<init>", "(Ljava/lang/String;)V"); | ||||
|     assert(mid != nullptr); | ||||
| 
 | ||||
|     env->Throw((jthrowable)env->NewObject(getJClass(env), mid, msg)); | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| }  // namespace rocksdb
 | ||||
| #endif  // JAVA_ROCKSJNI_PORTAL_H_
 | ||||
| @ -0,0 +1,185 @@ | ||||
| // Copyright (c) 2014, Facebook, Inc.  All rights reserved.
 | ||||
| // 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::DB methods from Java side.
 | ||||
| 
 | ||||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| #include <jni.h> | ||||
| #include <string> | ||||
| 
 | ||||
| #include "include/org_rocksdb_RocksDB.h" | ||||
| #include "rocksjni/portal.h" | ||||
| #include "rocksdb/db.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * Class:     org_rocksdb_RocksDB | ||||
|  * Method:    open0 | ||||
|  * Signature: (Ljava/lang/String;)V | ||||
|  */ | ||||
| void Java_org_rocksdb_RocksDB_open0( | ||||
|     JNIEnv* env, jobject java_db, jstring jdb_path) { | ||||
|   rocksdb::DB* db; | ||||
|   rocksdb::Options options; | ||||
|   options.create_if_missing = true; | ||||
| 
 | ||||
|   jboolean isCopy = false; | ||||
|   const char* db_path = env->GetStringUTFChars(jdb_path, &isCopy); | ||||
|   rocksdb::Status s = rocksdb::DB::Open(options, db_path, &db); | ||||
|   env->ReleaseStringUTFChars(jdb_path, db_path); | ||||
| 
 | ||||
|   if (s.ok()) { | ||||
|     rocksdb::RocksDBJni::setHandle(env, java_db, db); | ||||
|     return; | ||||
|   } | ||||
|   rocksdb::RocksDBExceptionJni::ThrowNew(env, s); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Class:     org_rocksdb_RocksDB | ||||
|  * Method:    put | ||||
|  * Signature: ([BI[BI)V | ||||
|  */ | ||||
| void Java_org_rocksdb_RocksDB_put( | ||||
|     JNIEnv* env, jobject jdb, | ||||
|     jbyteArray jkey, jint jkey_len, | ||||
|     jbyteArray jvalue, jint jvalue_len) { | ||||
|   rocksdb::DB* db = rocksdb::RocksDBJni::getHandle(env, jdb); | ||||
| 
 | ||||
|   jboolean isCopy; | ||||
|   jbyte* key = env->GetByteArrayElements(jkey, &isCopy); | ||||
|   jbyte* value = env->GetByteArrayElements(jvalue, &isCopy); | ||||
|   rocksdb::Slice key_slice( | ||||
|       reinterpret_cast<char*>(key), jkey_len); | ||||
|   rocksdb::Slice value_slice( | ||||
|       reinterpret_cast<char*>(value), jvalue_len); | ||||
| 
 | ||||
|   rocksdb::Status s = db->Put( | ||||
|       rocksdb::WriteOptions(), key_slice, value_slice); | ||||
| 
 | ||||
|   // trigger java unref on key and value.
 | ||||
|   // by passing JNI_ABORT, it will simply release the reference without
 | ||||
|   // copying the result back to the java byte array.
 | ||||
|   env->ReleaseByteArrayElements(jkey, key, JNI_ABORT); | ||||
|   env->ReleaseByteArrayElements(jvalue, value, JNI_ABORT); | ||||
| 
 | ||||
|   if (s.ok()) { | ||||
|     return; | ||||
|   } | ||||
|   rocksdb::RocksDBExceptionJni::ThrowNew(env, s); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Class:     org_rocksdb_RocksDB | ||||
|  * Method:    get | ||||
|  * Signature: ([BI)[B | ||||
|  */ | ||||
| jbyteArray Java_org_rocksdb_RocksDB_get___3BI( | ||||
|     JNIEnv* env, jobject jdb, jbyteArray jkey, jint jkey_len) { | ||||
|   rocksdb::DB* db = rocksdb::RocksDBJni::getHandle(env, jdb); | ||||
| 
 | ||||
|   jboolean isCopy; | ||||
|   jbyte* key = env->GetByteArrayElements(jkey, &isCopy); | ||||
|   rocksdb::Slice key_slice( | ||||
|       reinterpret_cast<char*>(key), jkey_len); | ||||
| 
 | ||||
|   std::string value; | ||||
|   rocksdb::Status s = db->Get( | ||||
|       rocksdb::ReadOptions(), | ||||
|       key_slice, &value); | ||||
| 
 | ||||
|   // trigger java unref on key.
 | ||||
|   // by passing JNI_ABORT, it will simply release the reference without
 | ||||
|   // copying the result back to the java byte array.
 | ||||
|   env->ReleaseByteArrayElements(jkey, key, JNI_ABORT); | ||||
| 
 | ||||
|   if (s.IsNotFound()) { | ||||
|     return nullptr; | ||||
|   } | ||||
| 
 | ||||
|   if (s.ok()) { | ||||
|     jbyteArray jvalue = env->NewByteArray(value.size()); | ||||
|     env->SetByteArrayRegion( | ||||
|         jvalue, 0, value.size(), | ||||
|         reinterpret_cast<const jbyte*>(value.c_str())); | ||||
|     return jvalue; | ||||
|   } | ||||
|   rocksdb::RocksDBExceptionJni::ThrowNew(env, s); | ||||
| 
 | ||||
|   return nullptr; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Class:     org_rocksdb_RocksDB | ||||
|  * Method:    get | ||||
|  * Signature: ([BI[BI)I | ||||
|  */ | ||||
| jint Java_org_rocksdb_RocksDB_get___3BI_3BI( | ||||
|     JNIEnv* env, jobject jdb, | ||||
|     jbyteArray jkey, jint jkey_len, | ||||
|     jbyteArray jvalue, jint jvalue_len) { | ||||
|   static const int kNotFound = -1; | ||||
|   static const int kStatusError = -2; | ||||
| 
 | ||||
|   rocksdb::DB* db = rocksdb::RocksDBJni::getHandle(env, jdb); | ||||
| 
 | ||||
|   jboolean isCopy; | ||||
|   jbyte* key = env->GetByteArrayElements(jkey, &isCopy); | ||||
|   jbyte* value = env->GetByteArrayElements(jvalue, &isCopy); | ||||
|   rocksdb::Slice key_slice( | ||||
|       reinterpret_cast<char*>(key), jkey_len); | ||||
| 
 | ||||
|   // TODO(yhchiang): we might save one memory allocation here by adding
 | ||||
|   // a DB::Get() function which takes preallocated jbyte* as input.
 | ||||
|   std::string cvalue; | ||||
|   rocksdb::Status s = db->Get( | ||||
|       rocksdb::ReadOptions(), key_slice, &cvalue); | ||||
| 
 | ||||
|   // trigger java unref on key.
 | ||||
|   // by passing JNI_ABORT, it will simply release the reference without
 | ||||
|   // copying the result back to the java byte array.
 | ||||
|   env->ReleaseByteArrayElements(jkey, key, JNI_ABORT); | ||||
| 
 | ||||
|   if (s.IsNotFound()) { | ||||
|     env->ReleaseByteArrayElements(jvalue, value, JNI_ABORT); | ||||
|     return kNotFound; | ||||
|   } else if (!s.ok()) { | ||||
|     // Here since we are throwing a Java exception from c++ side.
 | ||||
|     // As a result, c++ does not know calling this function will in fact
 | ||||
|     // throwing an exception.  As a result, the execution flow will
 | ||||
|     // not stop here, and codes after this throw will still be
 | ||||
|     // executed.
 | ||||
|     rocksdb::RocksDBExceptionJni::ThrowNew(env, s); | ||||
| 
 | ||||
|     // Return a dummy const value to avoid compilation error, although
 | ||||
|     // java side might not have a chance to get the return value :)
 | ||||
|     return kStatusError; | ||||
|   } | ||||
| 
 | ||||
|   int cvalue_len = static_cast<int>(cvalue.size()); | ||||
|   int length = std::min(jvalue_len, cvalue_len); | ||||
| 
 | ||||
|   memcpy(value, cvalue.c_str(), length); | ||||
|   env->ReleaseByteArrayElements(jvalue, value, JNI_COMMIT); | ||||
|   if (cvalue_len > length) { | ||||
|     return static_cast<jint>(cvalue_len); | ||||
|   } | ||||
|   return length; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Class:     org_rocksdb_RocksDB | ||||
|  * Method:    close0 | ||||
|  * Signature: ()V | ||||
|  */ | ||||
| void Java_org_rocksdb_RocksDB_close0( | ||||
|     JNIEnv* env, jobject java_db) { | ||||
|   rocksdb::DB* db = rocksdb::RocksDBJni::getHandle(env, java_db); | ||||
|   delete db; | ||||
|   db = nullptr; | ||||
| 
 | ||||
|   rocksdb::RocksDBJni::setHandle(env, java_db, db); | ||||
| } | ||||
| @ -0,0 +1,427 @@ | ||||
| //  Copyright (c) 2013, Facebook, Inc.  All rights reserved.
 | ||||
| //  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.
 | ||||
| //
 | ||||
| #include "utilities/geodb/geodb_impl.h" | ||||
| 
 | ||||
| #define __STDC_FORMAT_MACROS | ||||
| 
 | ||||
| #include <vector> | ||||
| #include <map> | ||||
| #include <string> | ||||
| #include <limits> | ||||
| #include "db/filename.h" | ||||
| #include "util/coding.h" | ||||
| 
 | ||||
| //
 | ||||
| // There are two types of keys. The first type of key-values
 | ||||
| // maps a geo location to the set of object ids and their values.
 | ||||
| // Table 1
 | ||||
| //   key     : p + : + $quadkey + : + $id +
 | ||||
| //             : + $latitude + : + $longitude
 | ||||
| //   value  :  value of the object
 | ||||
| // This table can be used to find all objects that reside near
 | ||||
| // a specified geolocation.
 | ||||
| //
 | ||||
| // Table 2
 | ||||
| //   key  : 'k' + : + $id
 | ||||
| //   value:  $quadkey
 | ||||
| 
 | ||||
| namespace rocksdb { | ||||
| 
 | ||||
| GeoDBImpl::GeoDBImpl(DB* db, const GeoDBOptions& options) : | ||||
|   GeoDB(db, options), db_(db), options_(options) { | ||||
| } | ||||
| 
 | ||||
| GeoDBImpl::~GeoDBImpl() { | ||||
| } | ||||
| 
 | ||||
| Status GeoDBImpl::Insert(const GeoObject& obj) { | ||||
|   WriteBatch batch; | ||||
| 
 | ||||
|   // It is possible that this id is already associated with
 | ||||
|   // with a different position. We first have to remove that
 | ||||
|   // association before we can insert the new one.
 | ||||
| 
 | ||||
|   // remove existing object, if it exists
 | ||||
|   GeoObject old; | ||||
|   Status status = GetById(obj.id, &old); | ||||
|   if (status.ok()) { | ||||
|     assert(obj.id.compare(old.id) == 0); | ||||
|     std::string quadkey = PositionToQuad(old.position, Detail); | ||||
|     std::string key1 = MakeKey1(old.position, old.id, quadkey); | ||||
|     std::string key2 = MakeKey2(old.id); | ||||
|     batch.Delete(Slice(key1)); | ||||
|     batch.Delete(Slice(key2)); | ||||
|   } else if (status.IsNotFound()) { | ||||
|     // What if another thread is trying to insert the same ID concurrently?
 | ||||
|   } else { | ||||
|     return status; | ||||
|   } | ||||
| 
 | ||||
|   // insert new object
 | ||||
|   std::string quadkey = PositionToQuad(obj.position, Detail); | ||||
|   std::string key1 = MakeKey1(obj.position, obj.id, quadkey); | ||||
|   std::string key2 = MakeKey2(obj.id); | ||||
|   batch.Put(Slice(key1), Slice(obj.value)); | ||||
|   batch.Put(Slice(key2), Slice(quadkey)); | ||||
|   return db_->Write(woptions_, &batch); | ||||
| } | ||||
| 
 | ||||
| Status GeoDBImpl::GetByPosition(const GeoPosition& pos, | ||||
|                                 const Slice& id, | ||||
|                                 std::string* value) { | ||||
|   std::string quadkey = PositionToQuad(pos, Detail); | ||||
|   std::string key1 = MakeKey1(pos, id, quadkey); | ||||
|   return db_->Get(roptions_, Slice(key1), value); | ||||
| } | ||||
| 
 | ||||
| Status GeoDBImpl::GetById(const Slice& id, GeoObject* object) { | ||||
|   Status status; | ||||
|   Slice quadkey; | ||||
| 
 | ||||
|   // create an iterator so that we can get a consistent picture
 | ||||
|   // of the database.
 | ||||
|   Iterator* iter = db_->NewIterator(roptions_); | ||||
| 
 | ||||
|   // create key for table2
 | ||||
|   std::string kt = MakeKey2(id); | ||||
|   Slice key2(kt); | ||||
| 
 | ||||
|   iter->Seek(key2); | ||||
|   if (iter->Valid() && iter->status().ok()) { | ||||
|     if (iter->key().compare(key2) == 0) { | ||||
|       quadkey = iter->value(); | ||||
|     } | ||||
|   } | ||||
|   if (quadkey.size() == 0) { | ||||
|     delete iter; | ||||
|     return Status::NotFound(key2); | ||||
|   } | ||||
| 
 | ||||
|   //
 | ||||
|   // Seek to the quadkey + id prefix
 | ||||
|   //
 | ||||
|   std::string prefix = MakeKey1Prefix(quadkey.ToString(), id); | ||||
|   iter->Seek(Slice(prefix)); | ||||
|   assert(iter->Valid()); | ||||
|   if (!iter->Valid() || !iter->status().ok()) { | ||||
|     delete iter; | ||||
|     return Status::NotFound(); | ||||
|   } | ||||
| 
 | ||||
|   // split the key into p + quadkey + id + lat + lon
 | ||||
|   std::vector<std::string> parts; | ||||
|   Slice key = iter->key(); | ||||
|   StringSplit(&parts, key.ToString(), ':'); | ||||
|   assert(parts.size() == 5); | ||||
|   assert(parts[0] == "p"); | ||||
|   assert(parts[1] == quadkey); | ||||
|   assert(parts[2] == id); | ||||
| 
 | ||||
|   // fill up output parameters
 | ||||
|   object->position.latitude = atof(parts[3].c_str()); | ||||
|   object->position.longitude = atof(parts[4].c_str()); | ||||
|   object->id = id.ToString();  // this is redundant
 | ||||
|   object->value = iter->value().ToString(); | ||||
|   delete iter; | ||||
|   return Status::OK(); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| Status GeoDBImpl::Remove(const Slice& id) { | ||||
|   // Read the object from the database
 | ||||
|   GeoObject obj; | ||||
|   Status status = GetById(id, &obj); | ||||
|   if (!status.ok()) { | ||||
|     return status; | ||||
|   } | ||||
| 
 | ||||
|   // remove the object by atomically deleting it from both tables
 | ||||
|   std::string quadkey = PositionToQuad(obj.position, Detail); | ||||
|   std::string key1 = MakeKey1(obj.position, obj.id, quadkey); | ||||
|   std::string key2 = MakeKey2(obj.id); | ||||
|   WriteBatch batch; | ||||
|   batch.Delete(Slice(key1)); | ||||
|   batch.Delete(Slice(key2)); | ||||
|   return db_->Write(woptions_, &batch); | ||||
| } | ||||
| 
 | ||||
| Status GeoDBImpl::SearchRadial(const GeoPosition& pos, | ||||
|   double radius, | ||||
|   std::vector<GeoObject>* values, | ||||
|   int number_of_values) { | ||||
|   // Gather all bounding quadkeys
 | ||||
|   std::vector<std::string> qids; | ||||
|   Status s = searchQuadIds(pos, radius, &qids); | ||||
|   if (!s.ok()) { | ||||
|     return s; | ||||
|   } | ||||
| 
 | ||||
|   // create an iterator
 | ||||
|   Iterator* iter = db_->NewIterator(ReadOptions()); | ||||
| 
 | ||||
|   // Process each prospective quadkey
 | ||||
|   for (std::string qid : qids) { | ||||
|     // The user is interested in only these many objects.
 | ||||
|     if (number_of_values == 0) { | ||||
|       break; | ||||
|     } | ||||
| 
 | ||||
|     // convert quadkey to db key prefix
 | ||||
|     std::string dbkey = MakeQuadKeyPrefix(qid); | ||||
| 
 | ||||
|     for (iter->Seek(dbkey); | ||||
|          number_of_values > 0 && iter->Valid() && iter->status().ok(); | ||||
|          iter->Next()) { | ||||
|       // split the key into p + quadkey + id + lat + lon
 | ||||
|       std::vector<std::string> parts; | ||||
|       Slice key = iter->key(); | ||||
|       StringSplit(&parts, key.ToString(), ':'); | ||||
|       assert(parts.size() == 5); | ||||
|       assert(parts[0] == "p"); | ||||
|       std::string* quadkey = &parts[1]; | ||||
| 
 | ||||
|       // If the key we are looking for is a prefix of the key
 | ||||
|       // we found from the database, then this is one of the keys
 | ||||
|       // we are looking for.
 | ||||
|       auto res = std::mismatch(qid.begin(), qid.end(), quadkey->begin()); | ||||
|       if (res.first == qid.end()) { | ||||
|         GeoPosition pos(atof(parts[3].c_str()), atof(parts[4].c_str())); | ||||
|         GeoObject obj(pos, parts[4], iter->value().ToString()); | ||||
|         values->push_back(obj); | ||||
|         number_of_values--; | ||||
|       } else { | ||||
|         break; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   delete iter; | ||||
|   return Status::OK(); | ||||
| } | ||||
| 
 | ||||
| std::string GeoDBImpl::MakeKey1(const GeoPosition& pos, Slice id, | ||||
|                                 std::string quadkey) { | ||||
|   std::string lat = std::to_string(pos.latitude); | ||||
|   std::string lon = std::to_string(pos.longitude); | ||||
|   std::string key = "p:"; | ||||
|   key.reserve(5 + quadkey.size() + id.size() + lat.size() + lon.size()); | ||||
|   key.append(quadkey); | ||||
|   key.append(":"); | ||||
|   key.append(id.ToString()); | ||||
|   key.append(":"); | ||||
|   key.append(lat); | ||||
|   key.append(":"); | ||||
|   key.append(lon); | ||||
|   return key; | ||||
| } | ||||
| 
 | ||||
| std::string GeoDBImpl::MakeKey2(Slice id) { | ||||
|   std::string key = "k:"; | ||||
|   key.append(id.ToString()); | ||||
|   return key; | ||||
| } | ||||
| 
 | ||||
| std::string GeoDBImpl::MakeKey1Prefix(std::string quadkey, | ||||
|                                       Slice id) { | ||||
|   std::string key = "p:"; | ||||
|   key.reserve(3 + quadkey.size() + id.size()); | ||||
|   key.append(quadkey); | ||||
|   key.append(":"); | ||||
|   key.append(id.ToString()); | ||||
|   return key; | ||||
| } | ||||
| 
 | ||||
| std::string GeoDBImpl::MakeQuadKeyPrefix(std::string quadkey) { | ||||
|   std::string key = "p:"; | ||||
|   key.append(quadkey); | ||||
|   return key; | ||||
| } | ||||
| 
 | ||||
| void GeoDBImpl::StringSplit(std::vector<std::string>* tokens, | ||||
|                             const std::string &text, char sep) { | ||||
|   std::size_t start = 0, end = 0; | ||||
|   while ((end = text.find(sep, start)) != std::string::npos) { | ||||
|     tokens->push_back(text.substr(start, end - start)); | ||||
|     start = end + 1; | ||||
|   } | ||||
|   tokens->push_back(text.substr(start)); | ||||
| } | ||||
| 
 | ||||
| // convert degrees to radians
 | ||||
| double GeoDBImpl::radians(double x) { | ||||
|   return (x * PI) / 180; | ||||
| } | ||||
| 
 | ||||
| // convert radians to degrees
 | ||||
| double GeoDBImpl::degrees(double x) { | ||||
|   return (x * 180) / PI; | ||||
| } | ||||
| 
 | ||||
| // convert a gps location to quad coordinate
 | ||||
| std::string GeoDBImpl::PositionToQuad(const GeoPosition& pos, | ||||
|                                       int levelOfDetail) { | ||||
|   Pixel p = PositionToPixel(pos, levelOfDetail); | ||||
|   Tile tile = PixelToTile(p); | ||||
|   return TileToQuadKey(tile, levelOfDetail); | ||||
| } | ||||
| 
 | ||||
| GeoPosition GeoDBImpl::displaceLatLon(double lat, double lon, | ||||
|                                       double deltay, double deltax) { | ||||
|   double dLat = deltay / EarthRadius; | ||||
|   double dLon = deltax / (EarthRadius * cos(radians(lat))); | ||||
|   return GeoPosition(lat + degrees(dLat), | ||||
|                      lon + degrees(dLon)); | ||||
| } | ||||
| 
 | ||||
| //
 | ||||
| // Return the distance between two positions on the earth
 | ||||
| //
 | ||||
| double GeoDBImpl::distance(double lat1, double lon1, | ||||
|                            double lat2, double lon2) { | ||||
|   double lon = radians(lon2 - lon1); | ||||
|   double lat = radians(lat2 - lat1); | ||||
| 
 | ||||
|   double a = (sin(lat / 2) * sin(lat / 2)) + | ||||
|               cos(radians(lat1)) * cos(radians(lat2)) * | ||||
|               (sin(lon / 2) * sin(lon / 2)); | ||||
|   double angle = 2 * atan2(sqrt(a), sqrt(1 - a)); | ||||
|   return angle * EarthRadius; | ||||
| } | ||||
| 
 | ||||
| //
 | ||||
| // Returns all the quadkeys inside the search range
 | ||||
| //
 | ||||
| Status GeoDBImpl::searchQuadIds(const GeoPosition& position, | ||||
|                                 double radius, | ||||
|                                 std::vector<std::string>* quadKeys) { | ||||
|   // get the outline of the search square
 | ||||
|   GeoPosition topLeftPos = boundingTopLeft(position, radius); | ||||
|   GeoPosition bottomRightPos = boundingBottomRight(position, radius); | ||||
| 
 | ||||
|   Pixel topLeft =  PositionToPixel(topLeftPos, Detail); | ||||
|   Pixel bottomRight =  PositionToPixel(bottomRightPos, Detail); | ||||
| 
 | ||||
|   // how many level of details to look for
 | ||||
|   int numberOfTilesAtMaxDepth = floor((bottomRight.x - topLeft.x) / 256); | ||||
|   int zoomLevelsToRise = floor(log(numberOfTilesAtMaxDepth) / log(2)); | ||||
|   zoomLevelsToRise++; | ||||
|   int levels = std::max(0, Detail - zoomLevelsToRise); | ||||
| 
 | ||||
|   quadKeys->push_back(PositionToQuad(GeoPosition(topLeftPos.latitude, | ||||
|                                                  topLeftPos.longitude), | ||||
|                                      levels)); | ||||
|   quadKeys->push_back(PositionToQuad(GeoPosition(topLeftPos.latitude, | ||||
|                                                  bottomRightPos.longitude), | ||||
|                                      levels)); | ||||
|   quadKeys->push_back(PositionToQuad(GeoPosition(bottomRightPos.latitude, | ||||
|                                                  topLeftPos.longitude), | ||||
|                                      levels)); | ||||
|   quadKeys->push_back(PositionToQuad(GeoPosition(bottomRightPos.latitude, | ||||
|                                                  bottomRightPos.longitude), | ||||
|                                      levels)); | ||||
|   return Status::OK(); | ||||
| } | ||||
| 
 | ||||
| // Determines the ground resolution (in meters per pixel) at a specified
 | ||||
| // latitude and level of detail.
 | ||||
| // Latitude (in degrees) at which to measure the ground resolution.
 | ||||
| // Level of detail, from 1 (lowest detail) to 23 (highest detail).
 | ||||
| // Returns the ground resolution, in meters per pixel.
 | ||||
| double GeoDBImpl::GroundResolution(double latitude, int levelOfDetail) { | ||||
|   latitude = clip(latitude, MinLatitude, MaxLatitude); | ||||
|   return cos(latitude * PI / 180) * 2 * PI * EarthRadius / | ||||
|          MapSize(levelOfDetail); | ||||
| } | ||||
| 
 | ||||
| // Converts a point from latitude/longitude WGS-84 coordinates (in degrees)
 | ||||
| // into pixel XY coordinates at a specified level of detail.
 | ||||
| GeoDBImpl::Pixel GeoDBImpl::PositionToPixel(const GeoPosition& pos, | ||||
|                                             int levelOfDetail) { | ||||
|   double latitude = clip(pos.latitude, MinLatitude, MaxLatitude); | ||||
|   double x = (pos.longitude + 180) / 360; | ||||
|   double sinLatitude = sin(latitude * PI / 180); | ||||
|   double y = 0.5 - log((1 + sinLatitude) / (1 - sinLatitude)) / (4 * PI); | ||||
|   double mapSize = MapSize(levelOfDetail); | ||||
|   double X = floor(clip(x * mapSize + 0.5, 0, mapSize - 1)); | ||||
|   double Y = floor(clip(y * mapSize + 0.5, 0, mapSize - 1)); | ||||
|   return Pixel((unsigned int)X, (unsigned int)Y); | ||||
| } | ||||
| 
 | ||||
| GeoPosition GeoDBImpl::PixelToPosition(const Pixel& pixel, int levelOfDetail) { | ||||
|   double mapSize = MapSize(levelOfDetail); | ||||
|   double x = (clip(pixel.x, 0, mapSize - 1) / mapSize) - 0.5; | ||||
|   double y = 0.5 - (clip(pixel.y, 0, mapSize - 1) / mapSize); | ||||
|   double latitude = 90 - 360 * atan(exp(-y * 2 * PI)) / PI; | ||||
|   double longitude = 360 * x; | ||||
|   return GeoPosition(latitude, longitude); | ||||
| } | ||||
| 
 | ||||
| // Converts a Pixel to a Tile
 | ||||
| GeoDBImpl::Tile GeoDBImpl::PixelToTile(const Pixel& pixel) { | ||||
|   unsigned int tileX = floor(pixel.x / 256); | ||||
|   unsigned int tileY = floor(pixel.y / 256); | ||||
|   return Tile(tileX, tileY); | ||||
| } | ||||
| 
 | ||||
| GeoDBImpl::Pixel GeoDBImpl::TileToPixel(const Tile& tile) { | ||||
|   unsigned int pixelX = tile.x * 256; | ||||
|   unsigned int pixelY = tile.y * 256; | ||||
|   return Pixel(pixelX, pixelY); | ||||
| } | ||||
| 
 | ||||
| // Convert a Tile to a quadkey
 | ||||
| std::string GeoDBImpl::TileToQuadKey(const Tile& tile, int levelOfDetail) { | ||||
|   std::stringstream quadKey; | ||||
|   for (int i = levelOfDetail; i > 0; i--) { | ||||
|     char digit = '0'; | ||||
|     int mask = 1 << (i - 1); | ||||
|     if ((tile.x & mask) != 0) { | ||||
|       digit++; | ||||
|     } | ||||
|     if ((tile.y & mask) != 0) { | ||||
|       digit++; | ||||
|       digit++; | ||||
|     } | ||||
|     quadKey << digit; | ||||
|   } | ||||
|   return quadKey.str(); | ||||
| } | ||||
| 
 | ||||
| //
 | ||||
| // Convert a quadkey to a tile and its level of detail
 | ||||
| //
 | ||||
| void GeoDBImpl::QuadKeyToTile(std::string quadkey, Tile* tile, | ||||
|                                      int *levelOfDetail) { | ||||
|   tile->x = tile->y = 0; | ||||
|   *levelOfDetail = quadkey.size(); | ||||
|   const char* key = reinterpret_cast<const char *>(quadkey.c_str()); | ||||
|   for (int i = *levelOfDetail; i > 0; i--) { | ||||
|     int mask = 1 << (i - 1); | ||||
|     switch (key[*levelOfDetail - i]) { | ||||
|       case '0': | ||||
|         break; | ||||
| 
 | ||||
|       case '1': | ||||
|         tile->x |= mask; | ||||
|         break; | ||||
| 
 | ||||
|       case '2': | ||||
|         tile->y |= mask; | ||||
|         break; | ||||
| 
 | ||||
|       case '3': | ||||
|         tile->x |= mask; | ||||
|         tile->y |= mask; | ||||
|         break; | ||||
| 
 | ||||
|       default: | ||||
|         std::stringstream msg; | ||||
|         msg << quadkey; | ||||
|         msg << " Invalid QuadKey."; | ||||
|         throw std::runtime_error(msg.str()); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| }  // namespace rocksdb
 | ||||
| @ -0,0 +1,187 @@ | ||||
| //  Copyright (c) 2013, Facebook, Inc.  All rights reserved.
 | ||||
| //  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.
 | ||||
| //
 | ||||
| 
 | ||||
| #pragma once | ||||
| #include <algorithm> | ||||
| #include <cmath> | ||||
| #include <string> | ||||
| #include <sstream> | ||||
| #include <stdexcept> | ||||
| #include <vector> | ||||
| 
 | ||||
| #include "utilities/geo_db.h" | ||||
| #include "utilities/stackable_db.h" | ||||
| #include "rocksdb/env.h" | ||||
| #include "rocksdb/status.h" | ||||
| 
 | ||||
| namespace rocksdb { | ||||
| 
 | ||||
| // A specific implementation of GeoDB
 | ||||
| 
 | ||||
| class GeoDBImpl : public GeoDB { | ||||
|  public: | ||||
|   GeoDBImpl(DB* db, const GeoDBOptions& options); | ||||
|   ~GeoDBImpl(); | ||||
| 
 | ||||
|   // Associate the GPS location with the identified by 'id'. The value
 | ||||
|   // is a blob that is associated with this object.
 | ||||
|   virtual Status Insert(const GeoObject& object); | ||||
| 
 | ||||
|   // Retrieve the value of the object located at the specified GPS
 | ||||
|   // location and is identified by the 'id'.
 | ||||
|   virtual Status GetByPosition(const GeoPosition& pos, | ||||
|                                const Slice& id, | ||||
|                                std::string* value); | ||||
| 
 | ||||
|   // Retrieve the value of the object identified by the 'id'. This method
 | ||||
|   // could be potentially slower than GetByPosition
 | ||||
|   virtual Status GetById(const Slice& id, GeoObject* object); | ||||
| 
 | ||||
|   // Delete the specified object
 | ||||
|   virtual Status Remove(const Slice& id); | ||||
| 
 | ||||
|   // Returns a list of all items within a circular radius from the
 | ||||
|   // specified gps location
 | ||||
|   virtual Status SearchRadial(const GeoPosition& pos, | ||||
|                               double radius, | ||||
|                               std::vector<GeoObject>* values, | ||||
|                               int number_of_values); | ||||
| 
 | ||||
|  private: | ||||
|   DB* db_; | ||||
|   const GeoDBOptions options_; | ||||
|   const WriteOptions woptions_; | ||||
|   const ReadOptions roptions_; | ||||
| 
 | ||||
|   // The value of PI
 | ||||
|   static constexpr double PI = 3.141592653589793; | ||||
| 
 | ||||
|   // convert degrees to radians
 | ||||
|   static double radians(double x); | ||||
| 
 | ||||
|   // convert radians to degrees
 | ||||
|   static double degrees(double x); | ||||
| 
 | ||||
|   // A pixel class that captures X and Y coordinates
 | ||||
|   class Pixel { | ||||
|    public: | ||||
|     unsigned int x; | ||||
|     unsigned int y; | ||||
|     Pixel(unsigned int a, unsigned int b) : | ||||
|      x(a), y(b) { | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   // A Tile in the geoid
 | ||||
|   class Tile { | ||||
|    public: | ||||
|     unsigned int x; | ||||
|     unsigned int y; | ||||
|     Tile(unsigned int a, unsigned int b) : | ||||
|      x(a), y(b) { | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   // convert a gps location to quad coordinate
 | ||||
|   static std::string PositionToQuad(const GeoPosition& pos, int levelOfDetail); | ||||
| 
 | ||||
|   // arbitrary constant use for WGS84 via
 | ||||
|   // http://en.wikipedia.org/wiki/World_Geodetic_System
 | ||||
|   // http://mathforum.org/library/drmath/view/51832.html
 | ||||
|   // http://msdn.microsoft.com/en-us/library/bb259689.aspx
 | ||||
|   // http://www.tuicool.com/articles/NBrE73
 | ||||
|   //
 | ||||
|   const int Detail = 23; | ||||
|   static constexpr double EarthRadius = 6378137; | ||||
|   static constexpr double MinLatitude = -85.05112878; | ||||
|   static constexpr double MaxLatitude = 85.05112878; | ||||
|   static constexpr double MinLongitude = -180; | ||||
|   static constexpr double MaxLongitude = 180; | ||||
| 
 | ||||
|   // clips a number to the specified minimum and maximum values.
 | ||||
|   static double clip(double n, double minValue, double maxValue) { | ||||
|     return fmin(fmax(n, minValue), maxValue); | ||||
|   } | ||||
| 
 | ||||
|   // Determines the map width and height (in pixels) at a specified level
 | ||||
|   // of detail, from 1 (lowest detail) to 23 (highest detail).
 | ||||
|   // Returns the map width and height in pixels.
 | ||||
|   static unsigned int MapSize(int levelOfDetail) { | ||||
|     return (unsigned int)(256 << levelOfDetail); | ||||
|   } | ||||
| 
 | ||||
|   // Determines the ground resolution (in meters per pixel) at a specified
 | ||||
|   // latitude and level of detail.
 | ||||
|   // Latitude (in degrees) at which to measure the ground resolution.
 | ||||
|   // Level of detail, from 1 (lowest detail) to 23 (highest detail).
 | ||||
|   // Returns the ground resolution, in meters per pixel.
 | ||||
|   static double GroundResolution(double latitude, int levelOfDetail); | ||||
| 
 | ||||
|   // Converts a point from latitude/longitude WGS-84 coordinates (in degrees)
 | ||||
|   // into pixel XY coordinates at a specified level of detail.
 | ||||
|   static Pixel PositionToPixel(const GeoPosition& pos, int levelOfDetail); | ||||
| 
 | ||||
|   static GeoPosition PixelToPosition(const Pixel& pixel, int levelOfDetail); | ||||
| 
 | ||||
|   // Converts a Pixel to a Tile
 | ||||
|   static Tile PixelToTile(const Pixel& pixel); | ||||
| 
 | ||||
|   static Pixel TileToPixel(const Tile& tile); | ||||
| 
 | ||||
|   // Convert a Tile to a quadkey
 | ||||
|   static std::string TileToQuadKey(const Tile& tile, int levelOfDetail); | ||||
| 
 | ||||
|   // Convert a quadkey to a tile and its level of detail
 | ||||
|   static void QuadKeyToTile(std::string quadkey, Tile* tile, | ||||
|                             int *levelOfDetail); | ||||
| 
 | ||||
|   // Return the distance between two positions on the earth
 | ||||
|   static double distance(double lat1, double lon1, | ||||
|                          double lat2, double lon2); | ||||
|   static GeoPosition displaceLatLon(double lat, double lon, | ||||
|                                     double deltay, double deltax); | ||||
| 
 | ||||
|   //
 | ||||
|   // Returns the top left position after applying the delta to
 | ||||
|   // the specified position
 | ||||
|   //
 | ||||
|   static GeoPosition boundingTopLeft(const GeoPosition& in, double radius) { | ||||
|     return displaceLatLon(in.latitude, in.longitude, -radius, -radius); | ||||
|   } | ||||
| 
 | ||||
|   //
 | ||||
|   // Returns the bottom right position after applying the delta to
 | ||||
|   // the specified position
 | ||||
|   static GeoPosition boundingBottomRight(const GeoPosition& in, | ||||
|                                          double radius) { | ||||
|     return displaceLatLon(in.latitude, in.longitude, radius, radius); | ||||
|   } | ||||
| 
 | ||||
|   //
 | ||||
|   // Get all quadkeys within a radius of a specified position
 | ||||
|   //
 | ||||
|   Status searchQuadIds(const GeoPosition& position, | ||||
|                        double radius, | ||||
|                        std::vector<std::string>* quadKeys); | ||||
| 
 | ||||
|   // splits a string into its components
 | ||||
|   static void StringSplit(std::vector<std::string>* tokens, | ||||
|                           const std::string &text, | ||||
|                           char sep); | ||||
| 
 | ||||
|   //
 | ||||
|   // Create keys for accessing rocksdb table(s)
 | ||||
|   //
 | ||||
|   static std::string MakeKey1(const GeoPosition& pos, | ||||
|                               Slice id, | ||||
|                               std::string quadkey); | ||||
|   static std::string MakeKey2(Slice id); | ||||
|   static std::string MakeKey1Prefix(std::string quadkey, | ||||
|                                     Slice id); | ||||
|   static std::string MakeQuadKeyPrefix(std::string quadkey); | ||||
| }; | ||||
| 
 | ||||
| }  // namespace rocksdb
 | ||||
| @ -0,0 +1,123 @@ | ||||
| //  Copyright (c) 2013, Facebook, Inc.  All rights reserved.
 | ||||
| //  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.
 | ||||
| //
 | ||||
| //
 | ||||
| #include "utilities/geodb/geodb_impl.h" | ||||
| 
 | ||||
| #include <cctype> | ||||
| #include "util/testharness.h" | ||||
| 
 | ||||
| namespace rocksdb { | ||||
| 
 | ||||
| class GeoDBTest { | ||||
|  public: | ||||
|   static const std::string kDefaultDbName; | ||||
|   static Options options; | ||||
|   DB* db; | ||||
|   GeoDB* geodb; | ||||
| 
 | ||||
|   GeoDBTest() { | ||||
|     GeoDBOptions geodb_options; | ||||
|     ASSERT_OK(DestroyDB(kDefaultDbName, options)); | ||||
|     options.create_if_missing = true; | ||||
|     Status status = DB::Open(options, kDefaultDbName, &db); | ||||
|     geodb =  new GeoDBImpl(db, geodb_options); | ||||
|   } | ||||
| 
 | ||||
|   ~GeoDBTest() { | ||||
|     delete geodb; | ||||
|   } | ||||
| 
 | ||||
|   GeoDB* getdb() { | ||||
|     return geodb; | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| const std::string GeoDBTest::kDefaultDbName = "/tmp/geodefault/"; | ||||
| Options GeoDBTest::options = Options(); | ||||
| 
 | ||||
| // Insert, Get and Remove
 | ||||
| TEST(GeoDBTest, SimpleTest) { | ||||
|   GeoPosition pos1(100, 101); | ||||
|   std::string id1("id1"); | ||||
|   std::string value1("value1"); | ||||
| 
 | ||||
|   // insert first object into database
 | ||||
|   GeoObject obj1(pos1, id1, value1); | ||||
|   Status status = getdb()->Insert(obj1); | ||||
|   ASSERT_TRUE(status.ok()); | ||||
| 
 | ||||
|   // insert second object into database
 | ||||
|   GeoPosition pos2(200, 201); | ||||
|   std::string id2("id2"); | ||||
|   std::string value2 = "value2"; | ||||
|   GeoObject obj2(pos2, id2, value2); | ||||
|   status = getdb()->Insert(obj2); | ||||
|   ASSERT_TRUE(status.ok()); | ||||
| 
 | ||||
|   // retrieve first object using position
 | ||||
|   std::string value; | ||||
|   status = getdb()->GetByPosition(pos1, Slice(id1), &value); | ||||
|   ASSERT_TRUE(status.ok()); | ||||
|   ASSERT_EQ(value, value1); | ||||
| 
 | ||||
|   // retrieve first object using id
 | ||||
|   GeoObject obj; | ||||
|   status = getdb()->GetById(Slice(id1), &obj); | ||||
|   ASSERT_TRUE(status.ok()); | ||||
|   ASSERT_EQ(obj.position.latitude, 100); | ||||
|   ASSERT_EQ(obj.position.longitude, 101); | ||||
|   ASSERT_EQ(obj.id.compare(id1), 0); | ||||
|   ASSERT_EQ(obj.value, value1); | ||||
| 
 | ||||
|   // delete first object
 | ||||
|   status = getdb()->Remove(Slice(id1)); | ||||
|   ASSERT_TRUE(status.ok()); | ||||
|   status = getdb()->GetByPosition(pos1, Slice(id1), &value); | ||||
|   ASSERT_TRUE(status.IsNotFound()); | ||||
|   status = getdb()->GetById(id1, &obj); | ||||
|   ASSERT_TRUE(status.IsNotFound()); | ||||
| 
 | ||||
|   // check that we can still find second object
 | ||||
|   status = getdb()->GetByPosition(pos2, id2, &value); | ||||
|   ASSERT_TRUE(status.ok()); | ||||
|   ASSERT_EQ(value, value2); | ||||
|   status = getdb()->GetById(id2, &obj); | ||||
|   ASSERT_TRUE(status.ok()); | ||||
| } | ||||
| 
 | ||||
| // Search.
 | ||||
| // Verify distances via http://www.stevemorse.org/nearest/distance.php
 | ||||
| TEST(GeoDBTest, Search) { | ||||
|   GeoPosition pos1(45, 45); | ||||
|   std::string id1("mid1"); | ||||
|   std::string value1 = "midvalue1"; | ||||
| 
 | ||||
|   // insert object at 45 degree latitude
 | ||||
|   GeoObject obj1(pos1, id1, value1); | ||||
|   Status status = getdb()->Insert(obj1); | ||||
|   ASSERT_TRUE(status.ok()); | ||||
| 
 | ||||
|   // search all objects centered at 46 degree latitude with
 | ||||
|   // a radius of 200 kilometers. We should find the one object that
 | ||||
|   // we inserted earlier.
 | ||||
|   std::vector<GeoObject> values; | ||||
|   status = getdb()->SearchRadial(GeoPosition(46, 46), 200000, &values); | ||||
|   ASSERT_TRUE(status.ok()); | ||||
|   ASSERT_EQ(values.size(), 1); | ||||
| 
 | ||||
|   // search all objects centered at 46 degree latitude with
 | ||||
|   // a radius of 2 kilometers. There should be none.
 | ||||
|   values.clear(); | ||||
|   status = getdb()->SearchRadial(GeoPosition(46, 46), 2, &values); | ||||
|   ASSERT_TRUE(status.ok()); | ||||
|   ASSERT_EQ(values.size(), 0); | ||||
| } | ||||
| 
 | ||||
| }  // namespace rocksdb
 | ||||
| 
 | ||||
| int main(int argc, char* argv[]) { | ||||
|   return rocksdb::test::RunAllTests(); | ||||
| } | ||||
					Loading…
					
					
				
		Reference in new issue
	
	 Igor Canadi
						Igor Canadi