archivist_lib/logs/
socket.rsuse super::stats::LogStreamStats;
use crate::logs::stored_message::StoredMessage;
use fuchsia_async as fasync;
use futures::Stream;
use std::marker::PhantomData;
use std::pin::Pin;
use std::sync::Arc;
use std::task::{Context, Poll};
pub trait Encoding {
fn wrap_bytes(bytes: Vec<u8>, stats: &Arc<LogStreamStats>) -> Option<StoredMessage>;
}
#[derive(Clone, Debug)]
pub struct LegacyEncoding;
#[derive(Clone, Debug)]
pub struct StructuredEncoding;
impl Encoding for LegacyEncoding {
fn wrap_bytes(buf: Vec<u8>, stats: &Arc<LogStreamStats>) -> Option<StoredMessage> {
StoredMessage::from_legacy(buf.into_boxed_slice(), stats)
}
}
impl Encoding for StructuredEncoding {
fn wrap_bytes(buf: Vec<u8>, stats: &Arc<LogStreamStats>) -> Option<StoredMessage> {
StoredMessage::new(buf.into_boxed_slice(), stats)
}
}
#[must_use = "don't drop logs on the floor please!"]
pub struct LogMessageSocket<E> {
buffer: Vec<u8>,
stats: Arc<LogStreamStats>,
socket: fasync::Socket,
_encoder: PhantomData<E>,
}
impl LogMessageSocket<LegacyEncoding> {
pub fn new(socket: fasync::Socket, stats: Arc<LogStreamStats>) -> Self {
stats.open_socket();
Self { socket, stats, _encoder: PhantomData, buffer: Vec::new() }
}
}
impl LogMessageSocket<StructuredEncoding> {
pub fn new_structured(socket: fasync::Socket, stats: Arc<LogStreamStats>) -> Self {
stats.open_socket();
Self { socket, stats, _encoder: PhantomData, buffer: Vec::new() }
}
}
impl<E> Stream for LogMessageSocket<E>
where
E: Encoding + Unpin,
{
type Item = Result<StoredMessage, zx::Status>;
fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
let this = self.get_mut();
loop {
match this.socket.poll_datagram(cx, &mut this.buffer) {
Poll::Pending => return Poll::Pending,
Poll::Ready(Err(zx::Status::PEER_CLOSED)) => return Poll::Ready(None),
Poll::Ready(Err(status)) => return Poll::Ready(Some(Err(status))),
Poll::Ready(Ok(0)) => continue,
Poll::Ready(Ok(_len)) => {
let buf = std::mem::take(&mut this.buffer);
let Some(msg) = E::wrap_bytes(buf, &this.stats) else {
continue;
};
return Poll::Ready(Some(Ok(msg)));
}
}
}
}
}
impl<E> Drop for LogMessageSocket<E> {
fn drop(&mut self) {
self.stats.close_socket();
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::testing::TEST_IDENTITY;
use diagnostics_data::{LogsField, Severity};
use diagnostics_log_encoding::encode::{Encoder, EncoderOpts};
use diagnostics_log_encoding::{Argument, Record};
use diagnostics_message::fx_log_packet_t;
use futures::StreamExt;
use std::io::Cursor;
#[fasync::run_until_stalled(test)]
async fn logger_stream_test() {
let (sin, sout) = zx::Socket::create_datagram();
let mut packet: fx_log_packet_t = Default::default();
packet.metadata.pid = 1;
packet.metadata.severity = 0x30; packet.data[0] = 5;
packet.fill_data(1..6, b'A' as _);
packet.fill_data(7..12, b'B' as _);
let socket = fasync::Socket::from_socket(sout);
let mut ls = LogMessageSocket::new(socket, Default::default());
sin.write(packet.as_bytes()).unwrap();
let expected_p = diagnostics_data::LogsDataBuilder::new(diagnostics_data::BuilderArgs {
timestamp: zx::BootInstant::from_nanos(packet.metadata.time),
component_url: Some(TEST_IDENTITY.url.clone()),
moniker: TEST_IDENTITY.moniker.clone(),
severity: Severity::Info,
})
.set_pid(packet.metadata.pid)
.set_tid(packet.metadata.tid)
.add_tag("AAAAA")
.set_message("BBBBB".to_string())
.build();
let bytes = ls.next().await.unwrap().unwrap();
let result_message = bytes.parse(&TEST_IDENTITY).unwrap();
assert_eq!(result_message, expected_p);
sin.write(packet.as_bytes()).unwrap();
let result_message = ls.next().await.unwrap().unwrap().parse(&TEST_IDENTITY).unwrap();
assert_eq!(result_message, expected_p);
}
#[fasync::run_until_stalled(test)]
async fn structured_logger_stream_test() {
let (sin, sout) = zx::Socket::create_datagram();
let timestamp = zx::BootInstant::from_nanos(107);
let record = Record {
timestamp,
severity: Severity::Fatal as u8,
arguments: vec![Argument::new("key", "value"), Argument::tag("tag-a")],
};
let mut buffer = Cursor::new(vec![0u8; 1024]);
let mut encoder = Encoder::new(&mut buffer, EncoderOpts::default());
encoder.write_record(record).unwrap();
let encoded = &buffer.get_ref()[..buffer.position() as usize];
let expected_p = diagnostics_data::LogsDataBuilder::new(diagnostics_data::BuilderArgs {
timestamp,
component_url: Some(TEST_IDENTITY.url.clone()),
moniker: TEST_IDENTITY.moniker.clone(),
severity: Severity::Fatal,
})
.add_tag("tag-a")
.add_key(diagnostics_data::LogsProperty::String(
LogsField::Other("key".to_string()),
"value".to_string(),
))
.build();
let socket = fasync::Socket::from_socket(sout);
let mut stream = LogMessageSocket::new_structured(socket, Default::default());
sin.write(encoded).unwrap();
let bytes = stream.next().await.unwrap().unwrap();
let result_message = bytes.parse(&TEST_IDENTITY).unwrap();
assert_eq!(bytes.size(), encoded.len());
assert_eq!(result_message, expected_p);
sin.write(encoded).unwrap();
let result_message = stream.next().await.unwrap().unwrap().parse(&TEST_IDENTITY).unwrap();
assert_eq!(result_message, expected_p);
}
}