use alloc::{
boxed::Box,
string::{String, ToString},
vec::Vec,
};
use core::fmt;
#[cfg(feature = "serde")]
use serde::{Deserialize, Serialize};
use crate::{
error::ProtoResult,
rr::{RData, RecordData, RecordDataDecodable, RecordType},
serialize::{binary::*, txt::ParseError},
};
#[cfg_attr(feature = "serde", derive(Deserialize, Serialize))]
#[derive(Debug, PartialEq, Eq, Hash, Clone)]
#[non_exhaustive]
pub struct TXT {
pub txt_data: Box<[Box<[u8]>]>,
}
impl TXT {
pub fn new(txt_data: Vec<String>) -> Self {
Self {
txt_data: txt_data
.into_iter()
.map(|s| s.into_bytes().into_boxed_slice())
.collect::<Vec<_>>()
.into_boxed_slice(),
}
}
pub fn from_bytes(txt_data: Vec<&[u8]>) -> Self {
Self {
txt_data: txt_data
.into_iter()
.map(|s| s.to_vec().into_boxed_slice())
.collect::<Vec<_>>()
.into_boxed_slice(),
}
}
#[allow(clippy::unnecessary_wraps)]
pub(crate) fn from_tokens<'i, I: Iterator<Item = &'i str>>(
tokens: I,
) -> Result<Self, ParseError> {
let txt_data = tokens.map(ToString::to_string).collect::<Vec<_>>();
Ok(Self::new(txt_data))
}
}
impl BinEncodable for TXT {
fn emit(&self, encoder: &mut BinEncoder<'_>) -> ProtoResult<()> {
for s in &self.txt_data {
encoder.emit_character_data(s)?;
}
Ok(())
}
}
impl RecordDataDecodable<'_> for TXT {
fn read_data(
decoder: &mut BinDecoder<'_>,
rdata_length: Restrict<u16>,
) -> Result<Self, DecodeError> {
let data_len = decoder.len();
let mut strings = Vec::with_capacity(1);
let rdata_length =
rdata_length.map(|u| u as usize).unverified();
while data_len - decoder.len() < rdata_length {
let string = decoder.read_character_data()?.unverified();
strings.push(string.to_vec().into_boxed_slice());
}
Ok(Self {
txt_data: strings.into_boxed_slice(),
})
}
}
impl RecordData for TXT {
fn try_borrow(data: &RData) -> Option<&Self> {
match data {
RData::TXT(data) => Some(data),
_ => None,
}
}
fn record_type(&self) -> RecordType {
RecordType::TXT
}
fn into_rdata(self) -> RData {
RData::TXT(self)
}
}
impl fmt::Display for TXT {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
for txt in self.txt_data.iter() {
f.write_str(&String::from_utf8_lossy(txt))?;
}
Ok(())
}
}
#[cfg(test)]
mod tests {
#![allow(clippy::dbg_macro, clippy::print_stdout)]
use alloc::string::ToString;
#[cfg(feature = "std")]
use std::println;
use super::*;
#[test]
fn test() {
let rdata = TXT::new(vec!["Test me some".to_string(), "more please".to_string()]);
let mut bytes = Vec::new();
let mut encoder: BinEncoder<'_> = BinEncoder::new(&mut bytes);
assert!(rdata.emit(&mut encoder).is_ok());
let bytes = encoder.into_bytes();
#[cfg(feature = "std")]
println!("bytes: {bytes:?}");
let mut decoder: BinDecoder<'_> = BinDecoder::new(bytes);
let restrict = Restrict::new(bytes.len() as u16);
let read_rdata = TXT::read_data(&mut decoder, restrict).expect("Decoding error");
assert_eq!(rdata, read_rdata);
}
#[test]
fn publish_binary_txt_record() {
let bin_data = vec![0, 1, 2, 3, 4, 5, 6, 7, 8];
let rdata = TXT::from_bytes(vec![b"Test me some", &bin_data]);
let mut bytes = Vec::new();
let mut encoder: BinEncoder<'_> = BinEncoder::new(&mut bytes);
assert!(rdata.emit(&mut encoder).is_ok());
let bytes = encoder.into_bytes();
#[cfg(feature = "std")]
println!("bytes: {bytes:?}");
let mut decoder: BinDecoder<'_> = BinDecoder::new(bytes);
let restrict = Restrict::new(bytes.len() as u16);
let read_rdata = TXT::read_data(&mut decoder, restrict).expect("Decoding error");
assert_eq!(rdata, read_rdata);
}
}