1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
//!
//! Subscribable data types.
//!
//! A subscription is a request for a callback on a subset of network traffic specified by a filter.
//! Each callback function requires one or more *subscribable data types* as parameter(s), which it
//! immutably borrows.
//!
//! Each subscribable datatype must:
//!
//! - Be defined as a [DataType](retina_core::filter::DataType), with appropriate parameters and [retina_core::filter::Level].
//! - Implement one of the traits defined in this module (Tracked, FromSession, etc.)
//! - Be added to the [DATATYPES](`crate::typedefs::DATATYPES`) map (note: we are actively working on an approach that eliminates this requirement).
//!
//!

pub mod conn_fts;
pub mod typedefs;
pub use conn_fts::*;
pub mod connection;
pub use connection::ConnRecord;
pub mod http_transaction;
pub use http_transaction::HttpTransaction;
pub mod dns_transaction;
pub use dns_transaction::DnsTransaction;
pub mod tls_handshake;
pub use tls_handshake::TlsHandshake;
pub mod quic_stream;
pub use quic_stream::QuicStream;
pub mod ssh_handshake;
pub use ssh_handshake::SshHandshake;
pub mod packet;
pub use packet::{Payload, ZcFrame};
pub mod static_type;
pub use static_type::*;
pub mod packet_list;
pub use packet_list::*;
pub use typedefs::*;

use retina_core::conntrack::pdu::L4Pdu;
use retina_core::filter::SubscriptionSpec;
use retina_core::protocols::stream::Session;
use retina_core::Mbuf;

/// Trait implemented by datatypes that require inline tracking.
/// This is typically required for subscribable types that require
/// calculating metrics throughout a connection, e.g. QoS metrics.
pub trait Tracked {
    /// Initialize internal data; called once per connection.
    /// Note `first_pkt` will also be delivered to `update`.
    fn new(first_pkt: &L4Pdu) -> Self;
    /// New packet in connection received (or reassembled, if reassembled=true)
    /// Note this may be invoked both pre- and post-reassembly; types
    /// should check `reassembled` to avoid double-counting.
    fn update(&mut self, pdu: &L4Pdu, reassembled: bool);
    /// The stream protocols (lower-case) required for this datatype.
    /// See `IMPLEMENTED_PROTOCOLS` in retina_core for list of supported protocols.
    fn stream_protocols() -> Vec<&'static str>;
    /// Clear internal data; called if connection no longer matches filter
    /// that requires the Tracked type.
    fn clear(&mut self);
}

/// Trait implemented by datatypes that are built from session data.
/// This is used when subscribing to specific parsed application-layer data.
pub trait FromSession {
    /// The stream protocols (lower-case) required for this datatype.
    /// See `IMPLEMENTED_PROTOCOLS` in retina_core for list of supported protocols.
    fn stream_protocols() -> Vec<&'static str>;
    /// Build Self from a parsed session, or return None if impossible.
    /// Invoked when the session is fully matched, parsed, and ready to
    /// be delivered to a callback.
    fn from_session(session: &Session) -> Option<&Self>;
    /// Build Self from a *list* of sessions, or return None if impossible.
    /// Invoked when the connection has terminated and a FromSession datatype
    /// must be delivered to a callback.
    fn from_sessionlist(sessionlist: &SessionList) -> Option<&Self>;
}

/// Trait implemented by datatypes that are built from a packet (Mbuf).
/// This is used when subscribing to packet-level data.
pub trait FromMbuf {
    fn from_mbuf(mbuf: &Mbuf) -> Option<&Self>;
}

/// Trait implemented by datatypes that are constant throughout
/// a connection and inferrable at first packet.
pub trait StaticData {
    fn new(first_pkt: &L4Pdu) -> Self;
}

/// Trait for a datatype that is built from a subscription specification.
/// [retina-filtergen](../filtergen) assumes that FilterStr is the only use-case for this.
#[doc(hidden)]
pub trait FromSubscription {
    /// Output the literal tokenstream (e.g., string literal) representing
    /// the constant value (e.g., matched filter string).
    fn from_subscription(spec: &SubscriptionSpec) -> proc_macro2::TokenStream;
}

/// Trait for a datatype that is built from a list of raw packets.
pub trait PacketList {
    /// Initialize internal data; called once per connection.
    /// Note `first_pkt` will also be delivered to `update`.
    fn new(first_pkt: &L4Pdu) -> Self;
    /// New packet in connection received (or reassembled, if reassembled=true)
    /// Note this may be invoked both pre- and post-reassembly; types
    /// should check `reassembled` to avoid double-counting.
    fn track_packet(&mut self, pdu: &L4Pdu, reassembled: bool);
    /// Clear internal data; called if connection no longer matches filter
    /// that requires the Tracked type.
    fn clear(&mut self);
}