# Rust ClamAV Client
A simple ClamAV client to send files, in-memory data, and byte streams to `clamd` for antivirus scanning. Supports Tokio and async-std.
Please note: The functions `ping_socket`, `scan_file_socket`, `scan_buffer_socket`, and `scan_stream_socket` are only available on Unix platforms.
Check out the [examples](#examples) below, the [integration tests](tests/clamav_client.rs), or the [documentation on docs.rs](https://docs.rs/clamav-client/latest/clamav_client/) for more information on how to use this library.

## Installation
Add this to your `Cargo.toml`:
```toml
[dependencies]
clamav-client = "0.4.4"
```
To use the `async` functions in `clamav_client::tokio`, add this to your `Cargo.toml`:
```toml
[dependencies]
clamav-client = { version = "0.4.4", features = ["tokio"] }
```
To scan Tokio streams, enable the `tokio-stream` feature instead and add this to your `Cargo.toml`:
```toml
[dependencies]
clamav-client = { version = "0.4.4", features = ["tokio-stream"] }
```
Support for `async-std` is also available by enabling the `async-std` feature:
```toml
[dependencies]
clamav-client = { version = "0.4.4", features = ["async-std"] }
```
## Examples
### Usage
```rust
let clamd_host_address = "localhost:3310";
// Ping clamd to make sure the server is available and accepting TCP connections
let clamd_available = match clamav_client::ping_tcp(clamd_host_address) {
Ok(ping_response) => ping_response == b"PONG\0",
Err(_) => false,
};
if !clamd_available {
println!("Cannot ping clamd at {}", clamd_host_address);
return;
}
assert!(clamd_available);
// Scan file for viruses
let file_path = "tests/data/eicar.txt";
let scan_file_response = clamav_client::scan_file_tcp(file_path, clamd_host_address, None).unwrap();
let file_clean = clamav_client::clean(&scan_file_response).unwrap();
if file_clean {
println!("No virus found in {}", file_path);
} else {
println!("The file {} is infected!", file_path);
}
assert!(!file_clean);
// Scan in-memory data for viruses
let buffer = br#"X5O!P%@AP[4\PZX54(P^)7CC)7}$EICAR-STANDARD-ANTIVIRUS-TEST-FILE!$H+H*"#;
let scan_buffer_response = clamav_client::scan_buffer_tcp(buffer, clamd_host_address, None).unwrap();
let data_clean = clamav_client::clean(&scan_buffer_response).unwrap();
if data_clean {
println!("No virus found");
} else {
println!("The data is infected!");
}
assert!(!data_clean);
```
### Usage - Async with Tokio
```rust
#[cfg(feature = "tokio")]
tokio::runtime::Builder::new_current_thread().enable_all().build().unwrap().block_on(async {
let clamd_host_address = "localhost:3310";
// Ping clamd asynchronously and await the result
let clamd_available = match clamav_client::tokio::ping_tcp(clamd_host_address).await {
Ok(ping_response) => ping_response == b"PONG\0",
Err(_) => false,
};
if !clamd_available {
println!("Cannot ping clamd at {}", clamd_host_address);
return;
}
assert!(clamd_available);
let file_path = "tests/data/eicar.txt";
let buffer = br#"X5O!P%@AP[4\PZX54(P^)7CC)7}$EICAR-STANDARD-ANTIVIRUS-TEST-FILE!$H+H*"#;
// Concurrently scan a file and a data buffer for viruses
let (scan_file_result, scan_buffer_result) = tokio::join!(
clamav_client::tokio::scan_file_tcp(file_path, clamd_host_address, None),
clamav_client::tokio::scan_buffer_tcp(buffer, clamd_host_address, None)
);
let scan_file_response = scan_file_result.unwrap();
let file_clean = clamav_client::clean(&scan_file_response).unwrap();
if file_clean {
println!("No virus found in {}", file_path);
} else {
println!("The file {} is infected!", file_path);
}
assert!(!file_clean);
let scan_buffer_response = scan_buffer_result.unwrap();
let data_clean = clamav_client::clean(&scan_buffer_response).unwrap();
if data_clean {
println!("No virus found");
} else {
println!("The data is infected!");
}
assert!(!data_clean);
})
```
More examples can be found in the [tests](tests/clamav_client.rs).
## Links
- [API documentation on docs.rs](https://docs.rs/clamav-client/latest/clamav_client/)
- [clamav-client on crates.io](https://crates.io/crates/clamav-client/)
## Development
### Testing locally
For the tests to pass, you should start `clamd` as follows:
`clamd -F --config-file=clamd/clamd.conf --datadir=clamd/database`
and then run `cargo test --all-features` to cover all tests.
It doesn't really matter how you start `clamd`, as long as the options from [clamd.conf](clamd/clamd.conf) are the same for your configuration.
## Contributors
- [Christopher Prohm](https://github.com/chmp)
- [Paul Makles](https://github.com/insertish)
- [Sean Clarke](https://github.com/SeanEClarke)
- [Kanji Tanaka](https://github.com/kaicoh)