<div align="center">
<h1>hostaddr</h1>
</div>
<div align="center">
Extended types (`Domain`, `Host`, `HostAddr`) for working with net in Rust.
[<img alt="github" src="https://img.shields.io/badge/github-al8n/hostaddr-8da0cb?style=for-the-badge&logo=Github" height="22">][Github-url]
<img alt="LoC" src="https://img.shields.io/endpoint?url=https%3A%2F%2Fgist.githubusercontent.com%2Fal8n%2F327b2a8aef9003246e45c6e47fe63937%2Fraw%2Fhostaddr" height="22">
[<img alt="Build" src="https://img.shields.io/github/actions/workflow/status/al8n/hostaddr/ci.yml?logo=Github-Actions&style=for-the-badge" height="22">][CI-url]
[<img alt="codecov" src="https://img.shields.io/codecov/c/gh/al8n/hostaddr?style=for-the-badge&token=6R3QFWRWHL&logo=codecov" height="22">][codecov-url]
[<img alt="docs.rs" src="https://img.shields.io/badge/docs.rs-hostaddr-66c2a5?style=for-the-badge&labelColor=555555&logo=" height="20">][doc-url]
[<img alt="crates.io" src="https://img.shields.io/crates/v/hostaddr?style=for-the-badge&logo=" height="22">][crates-url]
[<img alt="crates.io" src="https://img.shields.io/crates/d/hostaddr?color=critical&logo=&style=for-the-badge" height="22">][crates-url]
<img alt="license" src="https://img.shields.io/badge/License-Apache%202.0/MIT-blue.svg?style=for-the-badge&fontColor=white&logoColor=f5c076&logo=" height="22">
English | [简体中文][zh-cn-url]
</div>
## Features
- **`no_std` and `no-alloc` compatible**: Use the `Buffer` type for stack-allocated domains
- **Generic storage**: Works with `String`, `Arc<str>`, `Box<str>`, `Vec<u8>`, and many more
- **IDNA/Punycode support**: Automatic conversion of international domain names (requires `alloc` or `std`)
- **Type-safe parsing**: Compile-time guarantees for valid domains and hosts
- **Percent-encoding**: Automatic decoding of percent-encoded domains
## Installation
```toml
[dependencies]
hostaddr = "0.2"
```
### Cargo Features
- **`std`** (default): Enables standard library support
- **`alloc`**: Enables allocation support without `std`
- **`serde`**: Serialize/deserialize support
- **`arbitrary`**: Support for fuzzing with `arbitrary`
- **`cheap-clone`**: Implement `CheapClone` for smart pointer types
- **String types**: `smol_str`, `triomphe`, `bytes`, `tinyvec`, `smallvec`
## API Overview
This library provides three main types:
- **`Domain<S>`**: A validated DNS domain name (e.g., `example.com`, `測試.中國`)
- **`Host<S>`**: Either a domain name or an IP address
- **`HostAddr<S>`**: A host with an optional port number
## Examples
### Basic Usage
```rust,ignore
use hostaddr::HostAddr;
// Parse domain with String storage
let addr: HostAddr<String> = "example.com".parse().unwrap();
assert!(addr.is_domain());
assert_eq!(addr.to_string(), "example.com");
// Parse domain with port
let addr: HostAddr<String> = "example.com:8080".parse().unwrap();
assert_eq!(addr.port(), Some(8080));
// Parse IP address with port
let addr: HostAddr<String> = "127.0.0.1:8080".parse().unwrap();
assert!(addr.is_ip());
assert_eq!(addr.port(), Some(8080));
// Parse IPv6 address with port
let addr: HostAddr<String> = "[::1]:8080".parse().unwrap();
assert!(addr.is_ipv6());
```
### Using Different Storage Types
```rust,ignore
use hostaddr::HostAddr;
use std::sync::Arc;
// Using Arc<str> for shared ownership
let addr: HostAddr<Arc<str>> = "example.com:8080".parse().unwrap();
// Using Box<str> for heap allocation
let addr: HostAddr<Box<str>> = "example.com:8080".parse().unwrap();
// Using Vec<u8> for byte storage
let addr: HostAddr<Vec<u8>> = "example.com".parse().unwrap();
```
### Using Buffer for `no_std`/`no-alloc`
```rust,ignore
use hostaddr::{HostAddr, Buffer, Domain};
// Stack-allocated domain (no heap allocation)
let domain: Domain<Buffer> = Domain::try_from("example.com").unwrap();
// Works in no_std environments
let addr: HostAddr<Buffer> = HostAddr::try_from("example.com:443").unwrap();
```
### Working with Domains
```rust,ignore
use hostaddr::Domain;
// International domain names are automatically converted to punycode
let domain: Domain<String> = "測試.中國".parse().unwrap();
assert_eq!(domain.as_inner().as_str(), "xn--g6w251d.xn--fiqz9s");
// Parse ASCII-only domains without punycode conversion
let domain = Domain::try_from_ascii_str("example.com").unwrap();
assert_eq!(domain.as_ref().into_inner(), "example.com");
// Fully-qualified domain names (FQDN) are supported
let domain: Domain<String> = "example.com.".parse().unwrap();
assert_eq!(domain.as_inner().as_str(), "example.com.");
```
### Working with Hosts
```rust,ignore
use hostaddr::Host;
use std::net::IpAddr;
// Parse a host (domain or IP)
let host: Host<String> = "example.com".parse().unwrap();
assert!(host.is_domain());
let host: Host<String> = "127.0.0.1".parse().unwrap();
assert!(host.is_ip());
// Convert between types
let ip: IpAddr = "127.0.0.1".parse().unwrap();
let host: Host<String> = Host::from(ip);
```
### Manipulating HostAddr
```rust,ignore
use hostaddr::HostAddr;
// Create and modify
let mut addr: HostAddr<String> = "example.com".parse().unwrap();
addr.set_port(8080);
assert_eq!(addr.port(), Some(8080));
// Builder-style API
let addr = HostAddr::<String>::from_domain("api.example.com".parse().unwrap())
.with_port(443);
assert_eq!(addr.to_string(), "api.example.com:443");
// Check host type
if addr.is_domain() {
assert_eq!(addr.port(), Some(443));
}
```
### Percent Encoding
```rust,ignore
use hostaddr::Domain;
// Percent-encoded domains are automatically decoded
let domain: Domain<String> = "example%2Ecom".parse().unwrap();
assert_eq!(domain.as_inner().as_str(), "example.com");
// Works with international domains too
let domain: Domain<String> = "測試%2E中國".parse().unwrap();
assert_eq!(domain.as_inner().as_str(), "xn--g6w251d.xn--fiqz9s");
```
### Verification Functions
```rust,ignore
use hostaddr::{verify_domain, verify_ascii_domain};
// Verify any domain (including international)
assert!(verify_domain(b"example.com").is_ok());
assert!(verify_domain("測試.中國".as_bytes()).is_ok());
// Verify ASCII-only domains
assert!(verify_ascii_domain(b"example.com").is_ok());
assert!(verify_ascii_domain("測試.中國".as_bytes()).is_err());
```
#### License
`hostaddr` is under the terms of both the MIT license and the
Apache License (Version 2.0).
See [LICENSE-APACHE](LICENSE-APACHE), [LICENSE-MIT](LICENSE-MIT) for details.
Copyright (c) 2025 Al Liu.
[Github-url]: https://github.com/al8n/hostaddr/
[CI-url]: https://github.com/al8n/hostaddr/actions/workflows/ci.yml
[doc-url]: https://docs.rs/hostaddr
[crates-url]: https://crates.io/crates/hostaddr
[codecov-url]: https://app.codecov.io/gh/al8n/hostaddr/
[zh-cn-url]: https://github.com/al8n/hostaddr/tree/main/README-zh_CN.md