mirror of
https://github.com/zerotier/ZeroTierOne.git
synced 2025-01-10 23:12:39 +00:00
382 lines
12 KiB
Markdown
382 lines
12 KiB
Markdown
|
winreg
|
||
|
[![Winreg on Appveyor][appveyor-image]][appveyor]
|
||
|
[![Winreg on crates.io][cratesio-image]][cratesio]
|
||
|
[![Winreg on docs.rs][docsrs-image]][docsrs]
|
||
|
======
|
||
|
|
||
|
[appveyor-image]: https://ci.appveyor.com/api/projects/status/f3lwrt67ghrf5omd?svg=true
|
||
|
[appveyor]: https://ci.appveyor.com/project/gentoo90/winreg-rs
|
||
|
[cratesio-image]: https://img.shields.io/crates/v/winreg.svg
|
||
|
[cratesio]: https://crates.io/crates/winreg
|
||
|
[docsrs-image]: https://docs.rs/winreg/badge.svg
|
||
|
[docsrs]: https://docs.rs/winreg
|
||
|
|
||
|
Rust bindings to MS Windows Registry API. Work in progress.
|
||
|
|
||
|
Current features:
|
||
|
* Basic registry operations:
|
||
|
* open/create/delete keys
|
||
|
* load application hive from a file
|
||
|
* read and write values
|
||
|
* seamless conversion between `REG_*` types and rust primitives
|
||
|
* `String` and `OsString` <= `REG_SZ`, `REG_EXPAND_SZ` or `REG_MULTI_SZ`
|
||
|
* `String`, `&str`, `OsString`, `&OsStr` => `REG_SZ`
|
||
|
* `Vec<String>`, `Vec<OsString>` <= `REG_MULTI_SZ`
|
||
|
* `Vec<String>`, `Vec<&str>`, `Vec<OsString>`, `Vec<&OsStr>` => `REG_MULTI_SZ`
|
||
|
* `u32` <=> `REG_DWORD`
|
||
|
* `u64` <=> `REG_QWORD`
|
||
|
* Iteration through key names and through values
|
||
|
* Transactions
|
||
|
* Transacted serialization of rust types into/from registry (only primitives, structures and maps for now)
|
||
|
|
||
|
## Usage
|
||
|
|
||
|
### Basic usage
|
||
|
|
||
|
```toml
|
||
|
# Cargo.toml
|
||
|
[dependencies]
|
||
|
winreg = "0.10"
|
||
|
```
|
||
|
|
||
|
```rust
|
||
|
extern crate winreg;
|
||
|
use std::io;
|
||
|
use std::path::Path;
|
||
|
use winreg::enums::*;
|
||
|
use winreg::RegKey;
|
||
|
|
||
|
fn main() -> io::Result<()> {
|
||
|
println!("Reading some system info...");
|
||
|
let hklm = RegKey::predef(HKEY_LOCAL_MACHINE);
|
||
|
let cur_ver = hklm.open_subkey("SOFTWARE\\Microsoft\\Windows\\CurrentVersion")?;
|
||
|
let pf: String = cur_ver.get_value("ProgramFilesDir")?;
|
||
|
let dp: String = cur_ver.get_value("DevicePath")?;
|
||
|
println!("ProgramFiles = {}\nDevicePath = {}", pf, dp);
|
||
|
let info = cur_ver.query_info()?;
|
||
|
println!("info = {:?}", info);
|
||
|
let mt = info.get_last_write_time_system();
|
||
|
println!(
|
||
|
"last_write_time as winapi::um::minwinbase::SYSTEMTIME = {}-{:02}-{:02} {:02}:{:02}:{:02}",
|
||
|
mt.wYear, mt.wMonth, mt.wDay, mt.wHour, mt.wMinute, mt.wSecond
|
||
|
);
|
||
|
|
||
|
// enable `chrono` feature on `winreg` to make this work
|
||
|
// println!(
|
||
|
// "last_write_time as chrono::NaiveDateTime = {}",
|
||
|
// info.get_last_write_time_chrono()
|
||
|
// );
|
||
|
|
||
|
println!("And now lets write something...");
|
||
|
let hkcu = RegKey::predef(HKEY_CURRENT_USER);
|
||
|
let path = Path::new("Software").join("WinregRsExample1");
|
||
|
let (key, disp) = hkcu.create_subkey(&path)?;
|
||
|
|
||
|
match disp {
|
||
|
REG_CREATED_NEW_KEY => println!("A new key has been created"),
|
||
|
REG_OPENED_EXISTING_KEY => println!("An existing key has been opened"),
|
||
|
}
|
||
|
|
||
|
key.set_value("TestSZ", &"written by Rust")?;
|
||
|
let sz_val: String = key.get_value("TestSZ")?;
|
||
|
key.delete_value("TestSZ")?;
|
||
|
println!("TestSZ = {}", sz_val);
|
||
|
|
||
|
key.set_value("TestMultiSZ", &vec!["written", "by", "Rust"])?;
|
||
|
let multi_sz_val: Vec<String> = key.get_value("TestMultiSZ")?;
|
||
|
key.delete_value("TestMultiSZ")?;
|
||
|
println!("TestMultiSZ = {:?}", multi_sz_val);
|
||
|
|
||
|
key.set_value("TestDWORD", &1234567890u32)?;
|
||
|
let dword_val: u32 = key.get_value("TestDWORD")?;
|
||
|
println!("TestDWORD = {}", dword_val);
|
||
|
|
||
|
key.set_value("TestQWORD", &1234567891011121314u64)?;
|
||
|
let qword_val: u64 = key.get_value("TestQWORD")?;
|
||
|
println!("TestQWORD = {}", qword_val);
|
||
|
|
||
|
key.create_subkey("sub\\key")?;
|
||
|
hkcu.delete_subkey_all(&path)?;
|
||
|
|
||
|
println!("Trying to open nonexistent key...");
|
||
|
hkcu.open_subkey(&path).unwrap_or_else(|e| match e.kind() {
|
||
|
io::ErrorKind::NotFound => panic!("Key doesn't exist"),
|
||
|
io::ErrorKind::PermissionDenied => panic!("Access denied"),
|
||
|
_ => panic!("{:?}", e),
|
||
|
});
|
||
|
Ok(())
|
||
|
}
|
||
|
```
|
||
|
|
||
|
### Iterators
|
||
|
|
||
|
```rust
|
||
|
extern crate winreg;
|
||
|
use std::io;
|
||
|
use winreg::RegKey;
|
||
|
use winreg::enums::*;
|
||
|
|
||
|
fn main() -> io::Result<()> {
|
||
|
println!("File extensions, registered in system:");
|
||
|
for i in RegKey::predef(HKEY_CLASSES_ROOT)
|
||
|
.enum_keys().map(|x| x.unwrap())
|
||
|
.filter(|x| x.starts_with("."))
|
||
|
{
|
||
|
println!("{}", i);
|
||
|
}
|
||
|
|
||
|
let system = RegKey::predef(HKEY_LOCAL_MACHINE)
|
||
|
.open_subkey("HARDWARE\\DESCRIPTION\\System")?;
|
||
|
for (name, value) in system.enum_values().map(|x| x.unwrap()) {
|
||
|
println!("{} = {:?}", name, value);
|
||
|
}
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
```
|
||
|
|
||
|
### Transactions
|
||
|
|
||
|
```toml
|
||
|
# Cargo.toml
|
||
|
[dependencies]
|
||
|
winreg = { version = "0.10", features = ["transactions"] }
|
||
|
```
|
||
|
|
||
|
```rust
|
||
|
extern crate winreg;
|
||
|
use std::io;
|
||
|
use winreg::RegKey;
|
||
|
use winreg::enums::*;
|
||
|
use winreg::transaction::Transaction;
|
||
|
|
||
|
fn main() -> io::Result<()> {
|
||
|
let t = Transaction::new()?;
|
||
|
let hkcu = RegKey::predef(HKEY_CURRENT_USER);
|
||
|
let (key, _disp) = hkcu.create_subkey_transacted("Software\\RustTransaction", &t)?;
|
||
|
key.set_value("TestQWORD", &1234567891011121314u64)?;
|
||
|
key.set_value("TestDWORD", &1234567890u32)?;
|
||
|
|
||
|
println!("Commit transaction? [y/N]:");
|
||
|
let mut input = String::new();
|
||
|
io::stdin().read_line(&mut input)?;
|
||
|
input = input.trim_right().to_owned();
|
||
|
if input == "y" || input == "Y" {
|
||
|
t.commit()?;
|
||
|
println!("Transaction committed.");
|
||
|
}
|
||
|
else {
|
||
|
// this is optional, if transaction wasn't committed,
|
||
|
// it will be rolled back on disposal
|
||
|
t.rollback()?;
|
||
|
|
||
|
println!("Transaction wasn't committed, it will be rolled back.");
|
||
|
}
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
```
|
||
|
|
||
|
### Serialization
|
||
|
|
||
|
```toml
|
||
|
# Cargo.toml
|
||
|
[dependencies]
|
||
|
winreg = { version = "0.10", features = ["serialization-serde"] }
|
||
|
serde = "1"
|
||
|
serde_derive = "1"
|
||
|
```
|
||
|
|
||
|
```rust
|
||
|
#[macro_use]
|
||
|
extern crate serde_derive;
|
||
|
extern crate winreg;
|
||
|
use std::collections::HashMap;
|
||
|
use std::error::Error;
|
||
|
use winreg::enums::*;
|
||
|
|
||
|
#[derive(Debug, Serialize, Deserialize, PartialEq)]
|
||
|
struct Coords {
|
||
|
x: u32,
|
||
|
y: u32,
|
||
|
}
|
||
|
|
||
|
#[derive(Debug, Serialize, Deserialize, PartialEq)]
|
||
|
struct Size {
|
||
|
w: u32,
|
||
|
h: u32,
|
||
|
}
|
||
|
|
||
|
#[derive(Debug, Serialize, Deserialize, PartialEq)]
|
||
|
struct Rectangle {
|
||
|
coords: Coords,
|
||
|
size: Size,
|
||
|
}
|
||
|
|
||
|
#[derive(Debug, Serialize, Deserialize, PartialEq)]
|
||
|
struct Test {
|
||
|
t_bool: bool,
|
||
|
t_u8: u8,
|
||
|
t_u16: u16,
|
||
|
t_u32: u32,
|
||
|
t_u64: u64,
|
||
|
t_usize: usize,
|
||
|
t_struct: Rectangle,
|
||
|
t_map: HashMap<String, u32>,
|
||
|
t_string: String,
|
||
|
#[serde(rename = "")] // empty name becomes the (Default) value in the registry
|
||
|
t_char: char,
|
||
|
t_i8: i8,
|
||
|
t_i16: i16,
|
||
|
t_i32: i32,
|
||
|
t_i64: i64,
|
||
|
t_isize: isize,
|
||
|
t_f64: f64,
|
||
|
t_f32: f32,
|
||
|
}
|
||
|
|
||
|
fn main() -> Result<(), Box<dyn Error>> {
|
||
|
let hkcu = winreg::RegKey::predef(HKEY_CURRENT_USER);
|
||
|
let (key, _disp) = hkcu.create_subkey("Software\\RustEncode")?;
|
||
|
|
||
|
let mut map = HashMap::new();
|
||
|
map.insert("".to_owned(), 0); // empty name becomes the (Default) value in the registry
|
||
|
map.insert("v1".to_owned(), 1);
|
||
|
map.insert("v2".to_owned(), 2);
|
||
|
map.insert("v3".to_owned(), 3);
|
||
|
|
||
|
let v1 = Test {
|
||
|
t_bool: false,
|
||
|
t_u8: 127,
|
||
|
t_u16: 32768,
|
||
|
t_u32: 123_456_789,
|
||
|
t_u64: 123_456_789_101_112,
|
||
|
t_usize: 1_234_567_891,
|
||
|
t_struct: Rectangle {
|
||
|
coords: Coords { x: 55, y: 77 },
|
||
|
size: Size { w: 500, h: 300 },
|
||
|
},
|
||
|
t_map: map,
|
||
|
t_string: "test 123!".to_owned(),
|
||
|
t_char: 'a',
|
||
|
t_i8: -123,
|
||
|
t_i16: -2049,
|
||
|
t_i32: 20100,
|
||
|
t_i64: -12_345_678_910,
|
||
|
t_isize: -1_234_567_890,
|
||
|
t_f64: -0.01,
|
||
|
t_f32: 3.15,
|
||
|
};
|
||
|
|
||
|
key.encode(&v1)?;
|
||
|
|
||
|
let v2: Test = key.decode()?;
|
||
|
println!("Decoded {:?}", v2);
|
||
|
|
||
|
println!("Equal to encoded: {:?}", v1 == v2);
|
||
|
Ok(())
|
||
|
}
|
||
|
```
|
||
|
|
||
|
## Changelog
|
||
|
|
||
|
### 0.10.1
|
||
|
|
||
|
* Bump minimal required version of `winapi` to `0.3.9` (required for `load_app_key`)
|
||
|
* Reexport `REG_PROCESS_APPKEY` and use it in the `load_app_key` example
|
||
|
|
||
|
### 0.10.0
|
||
|
|
||
|
* Add `RegKey::load_app_key()` and `RegKey::load_app_key_with_flags()` ([#30](https://github.com/gentoo90/winreg-rs/issues/30))
|
||
|
* Update dev dependency `rand` to `0.8`
|
||
|
* Add Github actions
|
||
|
* Fix some clippy warnings
|
||
|
|
||
|
### 0.9.0
|
||
|
|
||
|
* Breaking change: `OsStr` and `OsString` registry values are not `NULL`-terminated any more ([#34](https://github.com/gentoo90/winreg-rs/issues/34), [#42](https://github.com/gentoo90/winreg-rs/issues/42))
|
||
|
* Refactoring: use macros for `ToRegValue` impls and tests for string values
|
||
|
* Fix `bare_trait_objects` warning in the doctests
|
||
|
* Add `impl ToRegValue for OsString`
|
||
|
* Add conversion between `REG_MULTI_SZ` and vectors of strings ([#16](https://github.com/gentoo90/winreg-rs/issues/16))
|
||
|
* Fix: set minimal `winapi` version to 0.3.7 (earlier versions don't have `impl-default` and `impl-debug` features which we use)
|
||
|
* Appveyor now checks the crate against `rust-1.31.1` too
|
||
|
|
||
|
### 0.8.0
|
||
|
|
||
|
* Implement serialization of `char` and maps
|
||
|
* Implement `std::fmt::Display` for `RegValue`
|
||
|
* Make `RegKey::{predef,raw_handle,enum_keys,enum_values}` functions `const`
|
||
|
* Give a better error message when compiling on platforms other than Windows ([#38](https://github.com/gentoo90/winreg-rs/pull/38))
|
||
|
* Tests are moved from `src/lib.rs` to `tests/reg_key.rs`
|
||
|
|
||
|
### 0.7.0
|
||
|
|
||
|
* Breaking change: remove deprecated `Error::description` ([#28](https://github.com/gentoo90/winreg-rs/pull/28))
|
||
|
* Optimize `Iterator::nth()` for the `Enum*` iterators ([#29](https://github.com/gentoo90/winreg-rs/pull/29))
|
||
|
|
||
|
### 0.6.2
|
||
|
|
||
|
* Add `RegKey::delete_subkey_with_flags()` ([#27](https://github.com/gentoo90/winreg-rs/pull/27))
|
||
|
|
||
|
### 0.6.1
|
||
|
|
||
|
* Add `last_write_time` field to `RegKeyMetadata` (returned by `RegKey::query_info()`) ([#25](https://github.com/gentoo90/winreg-rs/pull/25)).
|
||
|
* Add `get_last_write_time_system()` and `get_last_write_time_chrono()` (under `chrono` feature) methods to `RegKeyMetadata`.
|
||
|
|
||
|
### 0.6.0
|
||
|
|
||
|
* Breaking change: `create_subkey`, `create_subkey_with_flags`, `create_subkey_transacted` and
|
||
|
`create_subkey_transacted_with_flags` now return a tuple which contains the subkey and its disposition
|
||
|
which can be `REG_CREATED_NEW_KEY` or `REG_OPENED_EXISTING_KEY` ([#21](https://github.com/gentoo90/winreg-rs/issues/21)).
|
||
|
* Examples fixed to not use `unwrap` according to [Rust API guidelines](https://rust-lang-nursery.github.io/api-guidelines/documentation.html#examples-use--not-try-not-unwrap-c-question-mark).
|
||
|
|
||
|
### 0.5.1
|
||
|
|
||
|
* Reexport `HKEY` ([#15](https://github.com/gentoo90/winreg-rs/issues/15)).
|
||
|
* Add `raw_handle` method ([#18](https://github.com/gentoo90/winreg-rs/pull/18)).
|
||
|
|
||
|
### 0.5.0
|
||
|
|
||
|
* Breaking change: `open_subkey` now opens a key with readonly permissions.
|
||
|
Use `create_subkey` or `open_subkey_with_flags` to open with read-write permissins.
|
||
|
* Breaking change: features `transactions` and `serialization-serde` are now disabled by default.
|
||
|
* Breaking change: serialization now uses `serde` instead of `rustc-serialize`.
|
||
|
* `winapi` updated to `0.3`.
|
||
|
* Documentation fixes ([#14](https://github.com/gentoo90/winreg-rs/pull/14))
|
||
|
|
||
|
### 0.4.0
|
||
|
|
||
|
* Make transactions and serialization otional features
|
||
|
* Update dependensies + minor fixes ([#12](https://github.com/gentoo90/winreg-rs/pull/12))
|
||
|
|
||
|
### 0.3.5
|
||
|
|
||
|
* Implement `FromRegValue` for `OsString` and `ToRegValue` for `OsStr` ([#8](https://github.com/gentoo90/winreg-rs/issues/8))
|
||
|
* Minor fixes
|
||
|
|
||
|
### 0.3.4
|
||
|
|
||
|
* Add `copy_tree` method to `RegKey`
|
||
|
* Now checked with [rust-clippy](https://github.com/Manishearth/rust-clippy)
|
||
|
* no more `unwrap`s
|
||
|
* replaced `to_string` with `to_owned`
|
||
|
* Fix: reading strings longer than 2048 characters ([#6](https://github.com/gentoo90/winreg-rs/pull/6))
|
||
|
|
||
|
### 0.3.3
|
||
|
|
||
|
* Fix: now able to read values longer than 2048 bytes ([#3](https://github.com/gentoo90/winreg-rs/pull/3))
|
||
|
|
||
|
### 0.3.2
|
||
|
|
||
|
* Fix: `FromRegValue` trait now requires `Sized` (fixes build with rust 1.4)
|
||
|
|
||
|
### 0.3.1
|
||
|
|
||
|
* Fix: bump `winapi` version to fix build
|
||
|
|
||
|
### 0.3.0
|
||
|
|
||
|
* Add transactions support and make serialization transacted
|
||
|
* Breaking change: use `std::io::{Error,Result}` instead of own `RegError` and `RegResult`
|