Skip to content

Commit

Permalink
factor erange handling into its own function
Browse files Browse the repository at this point in the history
  • Loading branch information
leifwalsh authored and morisja committed Oct 12, 2024
1 parent 2fc55e8 commit 4ccdfad
Showing 1 changed file with 55 additions and 53 deletions.
108 changes: 55 additions & 53 deletions src/handlers.rs
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,24 @@ use nixish::{Netgroup, Service};

const ERANGE: i32 = 34;

fn call_with_erange_handling<F>(buf: &mut Vec<c_char>, mut f: F) -> i32
where
F: FnMut(&mut Vec<c_char>) -> i32,
{
loop {
let ret = f(buf);
if ret == ERANGE {
if buf.len() > 10 << 20 {
// Let's not let this get much bigger than 10MB
return ret;
}
buf.resize(buf.len() * 2, 0 as c_char);
} else {
return ret;
}
}
}

// these functions are not available in the nix::libc crate
extern "C" {
fn setnetgrent(netgroup: *const c_char) -> i32;
Expand Down Expand Up @@ -137,31 +155,26 @@ impl ServiceWithName {
let mut buffer: Vec<c_char> = vec![0; 1024];
let mut result: *mut servent = ptr::null_mut();

loop {
let ret = unsafe {
getservbyname_r(
service_name.as_ptr(),
proto.as_ref().map_or(ptr::null(), |p| p.as_ptr()),
&mut result_buf,
buffer.as_mut_ptr(),
buffer.len(),
&mut result,
)
};
// lookup was successful
if ret == 0 {
if !result.is_null() {
let service: Service = unsafe { *result }.try_into()?;
return Ok(Some(service));
} else {
return Ok(None);
};
} else if ret == ERANGE {
buffer.resize(buffer.len() * 2, 0 as c_char);
continue;
let ret = call_with_erange_handling(&mut buffer, |buffer| unsafe {
getservbyname_r(
service_name.as_ptr(),
proto.as_ref().map_or(ptr::null(), |p| p.as_ptr()),
&mut result_buf,
buffer.as_mut_ptr(),
buffer.len(),
&mut result,
)
});
// lookup was successful
if ret == 0 {
if !result.is_null() {
let service: Service = unsafe { *result }.try_into()?;
Ok(Some(service))
} else {
anyhow::bail!("Error: getservbyname_r failed with code {}", ret);
Ok(None)
}
} else {
anyhow::bail!("Error: getservbyname_r failed with code {}", ret);
}
}
}
Expand Down Expand Up @@ -200,30 +213,25 @@ impl ServiceWithPort {
let mut buffer: Vec<c_char> = vec![0; 1024];
let mut result: *mut servent = ptr::null_mut();

loop {
let ret = unsafe {
getservbyport_r(
self.port as c_int,
proto.as_ref().map_or(ptr::null(), |p| p.as_ptr()),
&mut result_buf,
buffer.as_mut_ptr(),
buffer.len(),
&mut result,
)
};
if ret == 0 {
if !result.is_null() {
let service: Service = unsafe { *result }.try_into()?;
return Ok(Some(service));
} else {
return Ok(None);
};
} else if ret == ERANGE {
buffer.resize(buffer.len() * 2, 0 as c_char);
continue;
let ret = call_with_erange_handling(&mut buffer, |buffer| unsafe {
getservbyport_r(
self.port as c_int,
proto.as_ref().map_or(ptr::null(), |p| p.as_ptr()),
&mut result_buf,
buffer.as_mut_ptr(),
buffer.len(),
&mut result,
)
});
if ret == 0 {
if !result.is_null() {
let service: Service = unsafe { *result }.try_into()?;
Ok(Some(service))
} else {
anyhow::bail!("Error: getservbyport_r failed with code {}", ret);
Ok(None)
}
} else {
anyhow::bail!("Error: getservbyport_r failed with code {}", ret);
}
}
}
Expand Down Expand Up @@ -343,16 +351,15 @@ impl NetgroupWithName {
let mut domain: *mut c_char = std::ptr::null_mut();

loop {
let ret = unsafe {
let ret = call_with_erange_handling(&mut buffer, |buffer| unsafe {
getnetgrent_r(
&mut host,
&mut user,
&mut domain,
buffer.as_mut_ptr(),
buffer.len() as size_t,
)
};

});
if ret == 1 {
let host_str = if !host.is_null() {
Some(unsafe { CStr::from_ptr(host) }.to_owned())
Expand Down Expand Up @@ -380,11 +387,6 @@ impl NetgroupWithName {
} else if ret == 0 {
unsafe { endnetgrent() };
break;
} else if ret == ERANGE {
//TODO - include error number libs
// Double the buffer size and retry
buffer.resize(buffer.len() * 2, 0 as c_char);
continue;
} else {
// Handle other errors
anyhow::bail!("Error: getnetgrent_r failed with code {}", ret);
Expand Down

0 comments on commit 4ccdfad

Please sign in to comment.