mirror of
https://github.com/zaphar/icmp-socket.git
synced 2025-07-21 19:29:47 -04:00
Plug new packet parser into socket and echo apis
This commit is contained in:
parent
3ac3c8ed4b
commit
89f866f4c0
70
src/echo.rs
70
src/echo.rs
@ -11,21 +11,43 @@
|
|||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
use std::convert::From;
|
use std::convert::{TryFrom, From, TryInto};
|
||||||
use std::net::{Ipv4Addr, Ipv6Addr};
|
use std::net::{Ipv4Addr, Ipv6Addr};
|
||||||
|
|
||||||
use packet::{Builder, Packet as P};
|
use packet::{Builder, Packet as P};
|
||||||
use packet::icmp::echo::Packet;
|
use packet::icmp::echo::Packet;
|
||||||
|
|
||||||
|
use crate::packet::{Icmpv6Packet, Icmpv6Message::{EchoReply, EchoRequest}};
|
||||||
|
|
||||||
// TODO(jwall): It turns out that the ICMPv6 packets are sufficiently
|
// TODO(jwall): It turns out that the ICMPv6 packets are sufficiently
|
||||||
// different from the ICMPv4 packets. In order to handle them appropriately
|
// different from the ICMPv4 packets. In order to handle them appropriately
|
||||||
// It is going to take some consideration.
|
// It is going to take some consideration.
|
||||||
use crate::{IcmpSocket4, IcmpSocket6};
|
use crate::{IcmpSocket4, IcmpSocket6};
|
||||||
|
|
||||||
pub struct EchoResponse {
|
pub struct EchoResponse {
|
||||||
identifier: u16,
|
pub identifier: u16,
|
||||||
sequence: u16,
|
pub sequence: u16,
|
||||||
payload: Vec<u8>,
|
pub payload: Vec<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TryFrom<Icmpv6Packet> for EchoResponse {
|
||||||
|
type Error = std::io::Error;
|
||||||
|
|
||||||
|
fn try_from(pkt: Icmpv6Packet) -> Result<Self, Self::Error> {
|
||||||
|
if let EchoReply{
|
||||||
|
identifier,
|
||||||
|
sequence,
|
||||||
|
payload,
|
||||||
|
} = pkt.message {
|
||||||
|
Ok(EchoResponse{
|
||||||
|
identifier,
|
||||||
|
sequence,
|
||||||
|
payload,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
Err(std::io::Error::new(std::io::ErrorKind::Other, "Incorrect icmpv6 message"))
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct EchoSocket4 {
|
pub struct EchoSocket4 {
|
||||||
@ -71,4 +93,42 @@ impl From<IcmpSocket4> for EchoSocket4 {
|
|||||||
fn from(sock: IcmpSocket4) -> Self {
|
fn from(sock: IcmpSocket4) -> Self {
|
||||||
EchoSocket4::new(sock)
|
EchoSocket4::new(sock)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub struct EchoSocket6 {
|
||||||
|
sequence: u16,
|
||||||
|
buf: Vec<u8>,
|
||||||
|
inner: IcmpSocket6,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl EchoSocket6 {
|
||||||
|
|
||||||
|
pub fn new(sock: IcmpSocket6) -> Self {
|
||||||
|
EchoSocket6{inner:sock, sequence: 0, buf: Vec::with_capacity(512)}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_max_hops(&mut self, hops: u32) {
|
||||||
|
self.inner.set_max_hops(hops);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn send_ping(&mut self, dest: Ipv6Addr, identifier: u16, payload: &[u8]) -> std::io::Result<()> {
|
||||||
|
let packet = Icmpv6Packet::with_echo_request(identifier, self.sequence, payload.to_owned())?;
|
||||||
|
self.sequence += 1;
|
||||||
|
self.inner.send_to(dest, packet)?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn recv_ping(&mut self) -> std::io::Result<EchoResponse> {
|
||||||
|
let bytes_read = self.inner.rcv_from(&mut self.buf)?;
|
||||||
|
match Icmpv6Packet::parse(&self.buf[0..bytes_read]) {
|
||||||
|
Ok(p) => return Ok(p.try_into()?),
|
||||||
|
Err(e) => return Err(std::io::Error::new(std::io::ErrorKind::Other, format!("Malformed ICMP Response: {:?}", e))),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<IcmpSocket6> for EchoSocket6 {
|
||||||
|
fn from(sock: IcmpSocket6) -> Self {
|
||||||
|
EchoSocket6::new(sock)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -326,4 +326,19 @@ impl Icmpv6Packet {
|
|||||||
|
|
||||||
pub enum Icmpv6PacketBuildError {
|
pub enum Icmpv6PacketBuildError {
|
||||||
InvalidCode(u8),
|
InvalidCode(u8),
|
||||||
|
}
|
||||||
|
use Icmpv6PacketBuildError::InvalidCode;
|
||||||
|
|
||||||
|
impl std::fmt::Display for Icmpv6PacketBuildError {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
write!(f, "{}", match self {
|
||||||
|
InvalidCode(c) => format!("Invalid Code: {}", c),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Icmpv6PacketBuildError> for std::io::Error {
|
||||||
|
fn from(err: Icmpv6PacketBuildError) -> Self {
|
||||||
|
std::io::Error::new(std::io::ErrorKind::Other, format!("{}", err))
|
||||||
|
}
|
||||||
}
|
}
|
@ -95,7 +95,6 @@ impl IcmpSocket6 {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO(jwall): This should take an actual packet not the payload.
|
|
||||||
pub fn send_to(&mut self, dest: Ipv6Addr, mut packet: Icmpv6Packet) -> std::io::Result<()> {
|
pub fn send_to(&mut self, dest: Ipv6Addr, mut packet: Icmpv6Packet) -> std::io::Result<()> {
|
||||||
let source = match self.bound_to {
|
let source = match self.bound_to {
|
||||||
Some(ref addr) => addr,
|
Some(ref addr) => addr,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user