A PHP extension for interacting with RocksDB, built with Rust. This extension provides a simple interface for RocksDB operations such as putting, getting, deleting, and managing column families.
php-rocksdb-rc
is a PHP extension that allows you to use RocksDB, a high-performance embedded database for key-value data, directly from your PHP applications. This extension is written in Rust using the ext-php-rs
and rust-rocksdb
crates to provide a seamless integration between PHP and RocksDB.
- Basic CRUD operations on RocksDB
- Support for column families
- TTL support for key-value pairs
- Advanced options like flushing and repairing the database
- Backup and restore functionality
- Write batch operations
- Snapshot support
- Transaction support
You can download pre-built binaries from the releases page. Download the appropriate binary for your system and PHP version.
-
Download the
.so
file for your PHP version and architecture. -
Place the file in your PHP extensions directory (usually
/usr/lib/php/extensions
). -
Add the following line to your
php.ini
file:extension=librocksdb.so
-
Restart your web server or PHP-FPM to load the extension.
-
Download the
.dylib
file for your PHP version and architecture. -
Place the file in your PHP extensions directory (usually
/usr/local/lib/php/extensions
). -
Add the following line to your
php.ini
file:extension=librocksdb.dylib
-
Restart your web server or PHP-FPM to load the extension.
To build the extension from source, you will need Rust and Cargo installed. Follow the steps below:
-
On Linux and macOS:
The recommended way to install Rust is via
rustup
, a toolchain installer for Rust.curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh
Follow the on-screen instructions to complete the installation. After installation, you may need to restart your terminal or source your profile:
source $HOME/.cargo/env
-
On Windows:
Download and run the
rustup-init.exe
installer from the official Rust website. Follow the on-screen instructions to complete the installation.After installation, open a new command prompt or PowerShell window and ensure that
cargo
andrustc
are in your PATH by running:rustc --version cargo --version
-
Clone the repository:
git clone https://github.com/yourusername/php-rocksdb-rc.git cd php-rocksdb-rc
-
Build the extension:
cargo build --release
-
Copy the built library to your PHP extensions directory:
-
On Linux:
cp target/release/librocksdb.so /usr/lib/php/extensions/
-
On macOS:
cp target/release/librocksdb.dylib /usr/local/lib/php/extensions/
-
On Windows:
copy target\release\librocksdb.dll C:\path\to\php\ext\
-
-
Add the following line to your
php.ini
file:-
On Linux and macOS:
extension=librocksdb.so
-
On Windows:
extension=librocksdb.dll
-
-
Restart your web server or PHP-FPM to load the extension.
- Ensure that the path to the PHP extensions directory is correct. You can find the extension directory by running
php -i | grep extension_dir
. - If you encounter any issues during the installation of Rust, refer to the official Rust installation guide for troubleshooting tips.
- Make sure your PHP installation is compatible with the extension. You can check your PHP version by running
php -v
.
By following these steps, you should be able to build and install the PHP extension from source successfully.
Here are some examples of how to use the php-rocksdb-rc
extension:
<?php
$db = new RocksDB("/path/to/db", 3600); // 3600 seconds TTL
// Put a value
$db->put("key1", "value1");
// Get a value
$value = $db->get("key1");
echo $value; // Outputs: value1
// Delete a value
$db->delete("key1");
// Create a column family
$db->createColumnFamily("new_cf");
// Put a value in a column family
$db->put("key2", "value2", "new_cf");
// Get a value from a column family
$value = $db->get("key2", "new_cf");
echo $value; // Outputs: value2
// List column families
$column_families = RocksDB::listColumnFamilies("/path/to/db");
print_r($column_families);
// Drop a column family
$db->dropColumnFamily("new_cf");
// Flush the database
$db->flush();
// Repair the database
RocksDB::repair("/path/to/db");
// Close the database
$db->close();
?>
This example demonstrates basic put and get operations in RocksDB.
<?php
$dbPath = __DIR__ . "/temp/testdb";
$db = new RocksDB($dbPath, 3600); // 3600 seconds TTL
$db->put("key1", "value1");
$db = null; // Free the connection
$db = new RocksDB($dbPath, 3600);
$value = $db->get("key1");
var_dump($value); // Outputs: string(6) "value1"
$db = null; // Free the connection
?>
This example demonstrates how to delete a key-value pair from RocksDB.
<?php
$dbPath = __DIR__ . "/temp/testdb";
$db = new RocksDB($dbPath, 3600); // 3600 seconds TTL
$db->put("key1", "value1");
$db->delete("key1");
$db = null; // Free the connection
$db = new RocksDB($dbPath, 3600);
$value = $db->get("key1");
var_dump($value); // Outputs: NULL
$db = null; // Free the connection
?>
This example demonstrates how to use an iterator to traverse key-value pairs in RocksDB.
<?php
$dbPath = __DIR__ . "/temp/testdb_iter";
$db = new RocksDB($dbPath, 3600); // 3600 seconds TTL
$db->flush();
$db->put("key_vvv", "value_a");
$db->put("key_ggg", "value_b");
$db->put("key_hhh", "value_c");
$db->seekToFirst();
$result = [];
while ($db->valid()) {
$res = $db->next();
$key = $res['key'];
$value = $res['value'];
$result[$key] = $value;
}
var_dump($result); // Outputs an array with all key-value pairs
$db = null; // Free the connection
?>
This example demonstrates how to create a backup of the RocksDB database and retrieve information about the backups.
<?php
$dbPath = __DIR__ . "/temp/testdb_backup";
$backupPath = __DIR__ . "/temp/testdb_backup_files";
// Create and use RocksDB instance
$db = new RocksDB($dbPath, 3600); // 3600 seconds TTL
$db->put("key1", "value1");
// Initialize backup engine and create a backup
$backup = new \RocksDBBackup($dbPath, 3600); // 3600 seconds TTL
$backup->init($backupPath);
$backup->create();
// Get backup info
$info = $backup->info();
print_r($info); // Outputs information about the backups
// Restore from backup
$restorePath = __DIR__ . "/temp/testdb_restore";
$backup->restore(1, $restorePath); // Restore the first backup
// Verify the restored data
$restoredDb = new RocksDB($restorePath, 3600);
$value = $restoredDb->get("key1");
var_dump($value); // Outputs: string(6) "value1"
// Cleanup
$db = null;
$restoredDb = null;
?>
This example demonstrates how to use write batch operations to perform multiple writes in a single batch.
<?php
$dbPath = __DIR__ . "/temp/testdb_write_batch";
$db = new RocksDB($dbPath, 3600); // 3600 seconds TTL
// Initialize write batch
$writeBatch = new \RocksDBWriteBatch($dbPath, 3600); // 3600 seconds TTL
$writeBatch->start();
$writeBatch->put("key1", "value1");
$writeBatch->put("key2", "value2");
$writeBatch->delete("key1");
$writeBatch->write(); // Write the batch to the database
// Verify the data
$value1 = $db->get("key1");
$value2 = $db->get("key2");
var_dump($value1); // Outputs: NULL (since key1 was deleted)
var_dump($value2); // Outputs: string(6) "value2"
// Cleanup
$db = null;
?>
This example demonstrates how to use transactions to ensure atomicity of multiple operations.
<?php
$dbPath = __DIR__ . "/temp/testdb_transaction";
$db = new RocksDB($dbPath, 3600); // 3600 seconds TTL
// Initialize transaction
$transaction = new \RocksDBTransaction($dbPath, 3600); // 3600 seconds TTL
$transaction->start();
$transaction->put("key1", "value1");
$transaction->put("key2", "value2");
$transaction->delete("key1");
$transaction->commit(); // Commit the transaction
// Verify the data
$value1 = $db->get("key1");
$value2 = $db->get("key2");
var_dump($value1); // Outputs: NULL (since key1 was deleted)
var_dump($value2); // Outputs: string(6) "value2"
// Cleanup
$db = null;
?>
This example demonstrates how to use snapshots to capture the state of the database at a specific point in time.
<?php
$dbPath = __DIR__ . "/temp/testdb_snapshot";
$db = new RocksDB($dbPath, 3600); // 3600 seconds TTL
// Put some initial data
$db->put("key1", "value1");
$db->put("key2", "value2");
// Create a snapshot
$snapshot = new \RocksDBSnapshot($dbPath, 3600); // 3600 seconds TTL
$snapshot->create();
// Modify the database after taking the snapshot
$db->put("key1", "new_value1");
$db->delete("key2");
// Verify the data in the snapshot
$snapshotValue1 = $snapshot->get("key1");
$snapshotValue2 = $snapshot->get("key2");
var_dump($snapshotValue1); // Outputs: string(6) "value1" (original value)
var_dump($snapshotValue2); // Outputs: string(6) "value2" (original value)
// Verify the current data in the database
$currentValue1 = $db->get("key1");
$currentValue2 = $db->get("key2");
var_dump($currentValue1); // Outputs: string(10) "new_value1"
var_dump($currentValue2); // Outputs: NULL (since key2 was deleted)
// Cleanup
$db = null;
$snapshot = null;
?>
Creates a new RocksDB instance with the specified path and TTL.
<?php
$db = new RocksDB("/path/to/db", 3600); // 3600 seconds TTL
?>
Inserts a key-value pair into the database.
<?php
$db->put("key1", "value1");
$db->put("key2", "value2", "new_cf"); // Using column family
?>
Retrieves the value associated with the given key.
<?php
$value = $db->get("key1");
echo $value; // Outputs: value1
$value = $db->get("key2", "new_cf"); // From column family
echo $value; // Outputs: value2
?>
Merges a value into the database using JSON Patch.
<?php
$db->merge("json_obj_key", '[ { "op": "replace", "path": "/employees/1/first_name", "value": "lucy" } ]');
$db->merge("json_obj_key", '[ { "op": "replace", "path": "/employees/0/last_name", "value": "dow" } ]');
?>
This method uses JSON Patch to update the JSON object in the database. For more details on JSON Patch, refer to RFC 6902.
Deletes the key-value pair associated with the given key.
<?php
$db->delete("key1");
$db->delete("key2", "new_cf"); // From column family
?>
Lists all column families in the database.
<?php
$column_families = RocksDB::listColumnFamilies("/path/to/db");
print_r($column_families);
?>
Creates a new column family with the specified name.
<?php
$db->createColumnFamily("new_cf");
?>
Drops the column family with the specified name.
<?php
$db->dropColumnFamily("new_cf");
?>
Retrieves a database property.
<?php
$property = $db->getProperty("rocksdb.stats");
echo $property;
$property = $db->getProperty("rocksdb.stats", "new_cf"); // From column family
echo $property;
?>
Flushes all memtable data to SST files.
<?php
$db->flush();
$db->flush("new_cf"); // Flush column family
?>
Repairs a RocksDB database at the specified path.
<?php
RocksDB::repair("/path/to/db");
?>
Closes the RocksDB instance.
<?php
$db->close();
?>
Returns all key-value pairs in the database or column family.
<?php
$data = $db->all();
print_r($data);
$data = $db->all("new_cf"); // From column family
print_r($data);
?>
Returns all keys in the database or column family.
<?php
$keys = $db->keys();
print_r($keys);
$keys = $db->keys("new_cf"); // From column family
print_r($keys);
?>
Seeks to the first key in the database or column family.
<?php
$db->seekToFirst();
?>
Seeks to the last key in the database or column family.
<?php
$db->seekToLast();
?>
Seeks to the specified key in the database or column family.
<?php
$db->seek("key1");
?>
Seeks to the specified key or previous key in the database or column family.
<?php
$db->seekForPrev("key1");
?>
Checks if the current iterator position is valid.
<?php
$isValid = $db->valid();
echo $isValid ? 'true' : 'false';
?>
Moves to the next key-value pair in the database or column family.
<?php
$kv = $db->next();
print_r($kv);
?>
Moves to the previous key-value pair in the database or column family.
<?php
$kv = $db->prev();
print_r($kv);
?>
Compacts the key-value pairs in the specified range within the database or column family.
<?php
$db->compact_range("key_start", "key_end");
$db->compact_range("key_start", "key_end", "new_cf"); // In column family
?>
Returns the names of the live SST files in the database.
<?php
$live_files = $db->get_live_files();
print_r($live_files);
?>
Sets the database options.
<?php
$options = [
"write_buffer_size" => "4194304",
"max_write_buffer_number" => "3",
];
$db->set_options($options);
$db->set_options($options, "new_cf"); // For column family
?>
Sets the compression type for the database or column family.
<?php
$db->set_compression("snappy");
$db->set_compression("zlib", "new_cf"); // For column family
?>
This method supports the following compression types: "none", "snappy", "zlib", "bzip2", "lz4", "lz4hc", "zstd".
Sets the size of the write buffer for the database or column family.
<?php
$db->set_write_buffer_size(4194304);
$db->set_write_buffer_size(4194304, "new_cf"); // For column family
?>
This method sets the amount of data to build up in memory (backed by an unsorted log on disk) before converting to a sorted on-disk file.
Sets the size of the block cache for the database or column family.
<?php
$db->set_cache_size(8388608);
$db->set_cache_size(8388608, "new_cf"); // For column family
?>
This method sets the amount of memory to use for the block cache, which is used to accelerate the read operations by caching the data blocks in memory.
Creates a new RocksDBBackup instance.
<?php
$backup = new \RocksDBBackup("/path/to/db", 3600); // 3600 seconds TTL
?>
Initializes the backup engine with the specified path.
<?php
$backup->init("/path/to/backup");
?>
Creates a backup of the database.
<?php
$backup->init("/path/to/backup");
$backup->create();
?>
Returns information about the backups.
<?php
$backup->init("/path/to/backup");
$info = $backup->info();
print_r($info);
?>
Purges old backups, keeping the specified number of backups.
<?php
$backup->init("/path/to/backup");
$backup->purgeOld(2);
?>
Restores the database from a backup.
<?php
$backup->init("/path/to/backup");
$backup->restore(1, "/path/to/restore");
?>
Creates a new RocksDBWriteBatch instance.
<?php
$write_batch = new \RocksDBWriteBatch("/path/to/db", 3600); // 3600 seconds TTL
?>
Starts a new write batch.
<?php
$write_batch->start();
?>
Puts a key-value pair into the current write batch.
<?php
$write_batch->start();
$write_batch->put("key1", "value1");
$write_batch->put("key2", "value2", "new_cf"); // Using column family
?>
Merges a value into the current write batch.
<?php
$write_batch->start();
$write_batch->merge("json_obj_key", "employees[1].first_name = lucy");
$write_batch->merge("json_obj_key", "employees[0].last_name = dow", "new_cf"); // Using column family
?>
Deletes a key-value pair from the current write batch.
<?php
$write_batch->start();
$write_batch->delete("key1");
$write_batch->delete("key2", "new_cf"); // From column family
?>
Writes the current write batch to the database.
<?php
$write_batch->start();
$write_batch->write();
?>
Clears the current write batch.
<?php
$write_batch->start();
$write_batch->clear();
?>
Destroys the current write batch.
<?php
$write_batch->start();
$write_batch->destroy();
?>
Creates a new RocksDBTransaction instance.
<?php
$transaction = new \RocksDBTransaction("/path/to/db", 3600); // 3600 seconds TTL
?>
Starts a new transaction.
<?php
$transaction->start();
?>
Commits the current transaction.
<?php
$transaction->commit();
?>
Rolls back the current transaction.
<?php
$transaction->rollback();
?>
Sets a savepoint within the current transaction.
<?php
$transaction->setSavepoint();
?>
Rolls back the transaction to the last savepoint.
<?php
$transaction->rollbackToSavepoint();
?>
Puts a key-value pair into the current transaction.
<?php
$transaction->put("key1", "value1");
$transaction->put("key2", "value2", "new_cf"); // Using column family
?>
Gets the value associated with the given key within the current transaction.
<?php
$value = $transaction->get("key1");
echo $value; // Outputs: value1
$value = $transaction->get("key2", "new_cf"); // From column family
echo $value; // Outputs: value2
?>
Deletes a key-value pair within the current transaction.
<?php
$transaction->delete("key1");
$transaction->delete("key2", "new_cf"); // From column family
?>
Merges a value within the current transaction.
<?php
$transaction->merge("json_obj_key", "employees[1].first_name = lucy");
$transaction->merge("json_obj_key", "employees[0].last_name = dow", "new_cf"); // Using column family
?>
Before creating a new instance of any class (e.g., RocksDB
, RocksDBBackup
, RocksDBWriteBatch
, RocksDBTransaction
), ensure to destroy the previous instance to free up the database connection.
<?php
$db = new \RocksDB($dbPath);
$db = null; // Free the connection
$iterator = new \RocksDBIterator($dbPath); // Now you can create a new instance
?>
To enable autocompletion for the librocksdb
extension in PhpStorm, follow these steps:
-
Download the Stub File
Download the
.php_rocksdb_rc.php
file from here. -
Place the Stub File in Your Project
Save the
.php_rocksdb_rc.php
file in your project directory.
After completing these steps, you should have autocompletion support for the librocksdb
extension in PhpStorm.
Contributions are welcome! Please submit pull requests or issues on the GitHub repository.
This project is licensed under the MIT License. See the LICENSE file for details.
This project uses the following open-source libraries:
The php-rocksdb-rc
extension provides a powerful and efficient way to interact with RocksDB from PHP. With support for basic CRUD operations, column families, TTL, backups, write batches, transactions, and snapshots, it offers a comprehensive set of features for managing key-value data in PHP applications. The provided examples demonstrate how to use these features effectively. Happy coding!