use rlua::prelude::*;use std::net::Ipv4Addr;use crate::bindings::net::ipv6::LuaIpv6;pub struct LuaIpv4(pub Ipv4Addr);impl LuaUserData for LuaIpv4 { fn add_methods<'lua, M: LuaUserDataMethods<'lua, Self>>(methods: &mut M) { methods.add_method("octets", |_, this: &LuaIpv4, _: ()| { Ok(this.0.octets().to_vec()) }); methods.add_method("is_unspecified", |_, this: &LuaIpv4, _: ()| { Ok(this.0.is_unspecified()) }); methods.add_method("is_loopback", |_, this: &LuaIpv4, _: ()| { Ok(this.0.is_loopback()) }); methods.add_method("is_private", |_, this: &LuaIpv4, _: ()| { Ok(this.0.is_private()) }); methods.add_method("is_link_local", |_, this: &LuaIpv4, _: ()| { Ok(this.0.is_link_local()) }); methods.add_method("is_multicast", |_, this: &LuaIpv4, _: ()| { Ok(this.0.is_multicast()) }); methods.add_method("is_broadcast", |_, this: &LuaIpv4, _: ()| { Ok(this.0.is_broadcast()) }); methods.add_method("is_documentation", |_, this: &LuaIpv4, _: ()| { Ok(this.0.is_documentation()) }); methods.add_method("to_ipv6_compatible", |_, this: &LuaIpv4, _: ()| { Ok(LuaIpv6(this.0.to_ipv6_compatible())) }); methods.add_method("to_ipv6_mapped", |_, this: &LuaIpv4, _: ()| { Ok(LuaIpv6(this.0.to_ipv6_mapped())) }); methods.add_meta_method(LuaMetaMethod::ToString, |_, this, _: ()| { Ok(this.0.to_string()) }); }}pub fn init(lua: &Lua) -> crate::Result<()> { let module = lua.create_table()?; module.set("new", lua.create_function( |_, ip: String| { ip.parse().map(LuaIpv4).map_err(LuaError::external) })? )?; module.set("from_table", lua.create_function( |_, ip: Vec<u8>| { if ip.len() != 4 { return Ok(None); } let (a, b, c, d) = ( ip.get(0).unwrap_or(&0), ip.get(1).unwrap_or(&0), ip.get(2).unwrap_or(&0), ip.get(3).unwrap_or(&0)); Ok(Some(LuaIpv4(Ipv4Addr::new(*a, *b, *c, *d)))) })? )?; lua.globals().set("ipv4", module)?; Ok(())}