Cleanup, polish and version bumping

This commit is contained in:
Imbus 2021-07-20 18:21:21 +02:00
parent 44e63e679e
commit d70e57ea90
8 changed files with 86 additions and 389 deletions

197
Cargo.lock generated
View file

@ -22,12 +22,6 @@ dependencies = [
"winapi", "winapi",
] ]
[[package]]
name = "autocfg"
version = "0.1.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1d49d90015b3c36167a20fe2810c5cd875ad504b39cff3d4eae7977e6b7c1cb2"
[[package]] [[package]]
name = "bitflags" name = "bitflags"
version = "1.2.1" version = "1.2.1"
@ -55,15 +49,6 @@ dependencies = [
"vec_map", "vec_map",
] ]
[[package]]
name = "cloudabi"
version = "0.0.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ddfc5b9aa5d4507acaf872de71051dfd0e309860e88966e1051e462a077aac4f"
dependencies = [
"bitflags",
]
[[package]] [[package]]
name = "dirs" name = "dirs"
version = "3.0.2" version = "3.0.2"
@ -84,21 +69,6 @@ dependencies = [
"winapi", "winapi",
] ]
[[package]]
name = "eff-wordlist"
version = "1.0.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "226cc2588a6b769f444d06dedd29a366ea92469edea1cef7873fbd76e5b34c24"
dependencies = [
"rand 0.6.5",
]
[[package]]
name = "fuchsia-cprng"
version = "0.1.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a06f77d526c1a601b7c4cdd98f54b5eaabffc14d5f2f0296febdc7f357c6d3ba"
[[package]] [[package]]
name = "getrandom" name = "getrandom"
version = "0.2.3" version = "0.2.3"
@ -133,15 +103,9 @@ checksum = "dd25036021b0de88a0aff6b850051563c6516d0bf53f8638938edbb9de732736"
[[package]] [[package]]
name = "libc" name = "libc"
version = "0.2.97" version = "0.2.98"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "12b8adadd720df158f4d70dfe7ccc6adb0472d7c55ca83445f6a5ab3e36f8fb6" checksum = "320cfe77175da3a483efed4bc0adc1968ca050b098ce4f2f1c13a56626128790"
[[package]]
name = "ppv-lite86"
version = "0.2.10"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ac74c624d6b2d21f425f752262f42188365d7b8ff1aff74c82e45136510a4857"
[[package]] [[package]]
name = "proc-macro2" name = "proc-macro2"
@ -161,161 +125,6 @@ dependencies = [
"proc-macro2", "proc-macro2",
] ]
[[package]]
name = "rand"
version = "0.6.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "6d71dacdc3c88c1fde3885a3be3fbab9f35724e6ce99467f7d9c5026132184ca"
dependencies = [
"autocfg",
"libc",
"rand_chacha 0.1.1",
"rand_core 0.4.2",
"rand_hc 0.1.0",
"rand_isaac",
"rand_jitter",
"rand_os",
"rand_pcg",
"rand_xorshift",
"winapi",
]
[[package]]
name = "rand"
version = "0.8.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2e7573632e6454cf6b99d7aac4ccca54be06da05aca2ef7423d22d27d4d4bcd8"
dependencies = [
"libc",
"rand_chacha 0.3.1",
"rand_core 0.6.3",
"rand_hc 0.3.1",
]
[[package]]
name = "rand_chacha"
version = "0.1.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "556d3a1ca6600bfcbab7c7c91ccb085ac7fbbcd70e008a98742e7847f4f7bcef"
dependencies = [
"autocfg",
"rand_core 0.3.1",
]
[[package]]
name = "rand_chacha"
version = "0.3.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
dependencies = [
"ppv-lite86",
"rand_core 0.6.3",
]
[[package]]
name = "rand_core"
version = "0.3.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7a6fdeb83b075e8266dcc8762c22776f6877a63111121f5f8c7411e5be7eed4b"
dependencies = [
"rand_core 0.4.2",
]
[[package]]
name = "rand_core"
version = "0.4.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9c33a3c44ca05fa6f1807d8e6743f3824e8509beca625669633be0acbdf509dc"
[[package]]
name = "rand_core"
version = "0.6.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d34f1408f55294453790c48b2f1ebbb1c5b4b7563eb1f418bcfcfdbb06ebb4e7"
dependencies = [
"getrandom",
]
[[package]]
name = "rand_hc"
version = "0.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7b40677c7be09ae76218dc623efbf7b18e34bced3f38883af07bb75630a21bc4"
dependencies = [
"rand_core 0.3.1",
]
[[package]]
name = "rand_hc"
version = "0.3.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d51e9f596de227fda2ea6c84607f5558e196eeaf43c986b724ba4fb8fdf497e7"
dependencies = [
"rand_core 0.6.3",
]
[[package]]
name = "rand_isaac"
version = "0.1.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ded997c9d5f13925be2a6fd7e66bf1872597f759fd9dd93513dd7e92e5a5ee08"
dependencies = [
"rand_core 0.3.1",
]
[[package]]
name = "rand_jitter"
version = "0.1.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1166d5c91dc97b88d1decc3285bb0a99ed84b05cfd0bc2341bdf2d43fc41e39b"
dependencies = [
"libc",
"rand_core 0.4.2",
"winapi",
]
[[package]]
name = "rand_os"
version = "0.1.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7b75f676a1e053fc562eafbb47838d67c84801e38fc1ba459e8f180deabd5071"
dependencies = [
"cloudabi",
"fuchsia-cprng",
"libc",
"rand_core 0.4.2",
"rdrand",
"winapi",
]
[[package]]
name = "rand_pcg"
version = "0.1.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "abf9b09b01790cfe0364f52bf32995ea3c39f4d2dd011eac241d2914146d0b44"
dependencies = [
"autocfg",
"rand_core 0.4.2",
]
[[package]]
name = "rand_xorshift"
version = "0.1.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "cbf7e9e623549b0e21f6e97cf8ecf247c1a8fd2e8a992ae265314300b2455d5c"
dependencies = [
"rand_core 0.3.1",
]
[[package]]
name = "rdrand"
version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "678054eb77286b51581ba43620cc911abf02758c91f93f479767aed0f90458b2"
dependencies = [
"rand_core 0.3.1",
]
[[package]] [[package]]
name = "redox_syscall" name = "redox_syscall"
version = "0.2.9" version = "0.2.9"
@ -422,9 +231,7 @@ version = "0.1.0"
dependencies = [ dependencies = [
"clap", "clap",
"dirs", "dirs",
"eff-wordlist",
"hex", "hex",
"rand 0.8.4",
"serde", "serde",
"serde_json", "serde_json",
] ]

View file

@ -11,5 +11,5 @@ serde = { version = "*", features = ["derive"] }
serde_json = "*" serde_json = "*"
dirs = "*" dirs = "*"
clap = "*" clap = "*"
eff-wordlist = "*" # eff-wordlist = "*"
rand = "*" # rand = "*"

View file

@ -1,61 +1,36 @@
TODO: Review parsing methods # Waker
Add confirm for wake all **An intelligent wake on lan utility to keep track of your machines**
Hosts should be pinged on listing The output from program help flag should provide a clear idea of its use:
```
Waker 0.1.0
Imbus64
Utility for sending magic packets to configured machines.
Structs defined: USAGE:
Host waker [FLAGS] [OPTIONS] [MAC ADDRESSES]...
- Implement methods to retrieve as macbytes/ipv4addr
- Field containing MagicPacket?
Machines
- WakeAll method
- PingAll method
MagicPacket
- Finalze parsers
The format of a Wake-on-LAN (WOL) magic packet is defined FLAGS:
as a byte array with 6 bytes of value 255 (0xFF) and -a, --add Add a new host
16 repetitions of the target machines 48-bit (6-byte) MAC address. --all Wake all configured hosts
-e, --edit Enter edit mode
-h, --help Prints help information
-l, --list List all configured entries
-p, --print-config Print contents of configuration file to stdout
-V, --version Prints version information
102 bytes OPTIONS:
--backup <File> Backup configuration file
several macs per name ARGS:
<MAC ADDRESSES>...
edit, delete, add ```
This project is currently in beta. Many features are implemented, but some may not work as expected.
wakemode::all ## Future plans:
- Further testing and polish in general.
wake(bytes) method - [ ] Include pinging functionality, so the user gets feedback on what machines are already awake (Top priority)
async ping method(ip)? - [ ] Enable users to import an existing config, appending selected hosts to current config. (Second priority)
- [ ] Perhaps wrap run_mode in an Option, with None being the default when the program is invoked without CLI parameters
const MAGIC_BYTES_HEADER: [u8; 6] = [0xFF; 6]; - [ ] Rewrite all input/blocking related code into a struct of some sort
vec<u8> bytes = FFFFFFFFFFFF .reserve(enough) 102?
match run_mode {
Some() => Do according to some
None => Figure out what to do(set run_mode)
}
add 8 char string -> to bytes ask for name and add
add 11 char string -> if mac[2,5,8] is same char -> split by it and ask for name -> add
store json in ~/.config/waker.conf
dump-json method
export/import? append?
struct mechine {
vec of all macs
ip of machine? so we can ping it?
name
}
struct mac_list {
Vec
}
ask what do:
1. Wake
2. Wake all
3. Add
4. Edit

View file

@ -1,7 +1,7 @@
use std::io; use std::io;
use std::io::prelude::*; use std::io::prelude::*;
/// Python like input method with prompt message /// Python like input function with prompt message
pub fn input(prompt: &str) -> String { pub fn input(prompt: &str) -> String {
print!("{}", prompt); print!("{}", prompt);
let _ = io::stdout().flush(); let _ = io::stdout().flush();
@ -15,11 +15,14 @@ pub fn input(prompt: &str) -> String {
input.trim().to_string() input.trim().to_string()
} }
/// Simple confirm dialogue /// Simple confirm dialogue. Appends " [y/N]: " to your message, and prints feedback on your
/// choice.
pub fn confirm(message: &str) -> bool { pub fn confirm(message: &str) -> bool {
let answer = input(format!("{} [y/N]: ", message).as_str()).to_uppercase(); let answer = input(format!("{} [y/N]: ", message).as_str()).to_uppercase();
if answer == "YES" || answer == "Y" { if answer == "YES" || answer == "Y" {
println!("Yes");
return true; return true;
} }
println!("No");
return false; return false;
} }

View file

@ -56,7 +56,9 @@ impl Machines {
Ok(machines) Ok(machines)
} }
fn create_skeleton_config(file: &PathBuf) -> Result<(), Box<dyn Error>> { /// Creates file and dumps a skeleton config into it
pub fn create_skeleton_config(file: &PathBuf) -> Result<(), Box<dyn Error>> {
std::fs::File::create(&file)?;
let skel_machines = Machines::new(); let skel_machines = Machines::new();
skel_machines.dump(file)?; skel_machines.dump(file)?;
return Ok(()); return Ok(());

View file

@ -16,7 +16,6 @@ mod host; // The actual Host struct
mod input; // Gives us a python-like input function, as well as a simple confirm function mod input; // Gives us a python-like input function, as well as a simple confirm function
mod machines; // Struct that holds a vec of Hosts, as well as operations on those mod machines; // Struct that holds a vec of Hosts, as well as operations on those
mod packet; // The actual magic packet struct, with wake methods e.t.c. mod packet; // The actual magic packet struct, with wake methods e.t.c.
// mod random_machine; // Exposes a function that generates a random Host, with random mac, ip and name // The actual host struct.
mod sanitizers; // Functions that sanitizes MAC and IP addresses mod sanitizers; // Functions that sanitizes MAC and IP addresses
// use crate::packet::*; // use crate::packet::*;
@ -76,20 +75,17 @@ fn prompt_file_creation(config_path: &PathBuf) -> Option<File> {
config_path.to_str().unwrap() config_path.to_str().unwrap()
); );
if input::confirm(&msg) { if input::confirm(&msg) {
let newfile = std::fs::File::create(&config_path).expect(&format!( let newfile = std::fs::File::create(&config_path).expect(&format!(
"Could not create file: {}", "Could not create file: {}",
config_path.to_str().unwrap() config_path.to_str().unwrap()
)); ));
// Deserialize an empty Machines struct into the new config file. // Deserialize an empty Machines struct into the new config file.
// The parser is not equipped to handle empty or malformatted files... match Machines::create_skeleton_config(config_path) {
// This could potentially be moved into a static method in Machines, like Ok(()) => { return Some(newfile); }
// Machines::init_config_file(), which does this for us. Err(_) => return None,
// }
// For now, this will do.
let skeleton_machines = Machines::new();
skeleton_machines.dump(&config_path).unwrap();
return Some(newfile);
} else { } else {
return None; return None;
} }
@ -169,7 +165,7 @@ fn edit_host(host: &mut Host, editmode: HostEditMode) {
let newip = input("New IP: "); let newip = input("New IP: ");
match sanitizers::sanitize(&newip, sanitizers::AddrType::IPv4) { match sanitizers::sanitize(&newip, sanitizers::AddrType::IPv4) {
Some(ip) => { Some(ip) => {
host.ips[index as usize] = newip; host.ips[index as usize] = ip;
} }
None => { None => {
println!("Could not parse IP"); println!("Could not parse IP");
@ -226,7 +222,7 @@ fn edit_host(host: &mut Host, editmode: HostEditMode) {
let newmac = input("New MAC: "); let newmac = input("New MAC: ");
match sanitizers::sanitize(&newmac, sanitizers::AddrType::MAC) { match sanitizers::sanitize(&newmac, sanitizers::AddrType::MAC) {
Some(mac_addr) => { Some(mac_addr) => {
host.macs[index as usize] = newmac; host.macs[index as usize] = mac_addr;
} }
None => { None => {
println!("Could not parse MAC"); println!("Could not parse MAC");
@ -345,8 +341,11 @@ fn main() -> Result<(), Box<dyn Error>> {
let file = prompt_file_creation(&config_path); let file = prompt_file_creation(&config_path);
if file.is_none() { if file.is_none() {
println!("Exiting..."); println!("Exiting...");
return Ok(());
} }
else {
println!("Config file created.");
}
return Ok(());
} }
// TODO: More sophisticated error checking and logging // TODO: More sophisticated error checking and logging
@ -362,12 +361,15 @@ fn main() -> Result<(), Box<dyn Error>> {
RunMode::Wake(wake_mode) => { RunMode::Wake(wake_mode) => {
match wake_mode { match wake_mode {
WakeMode::WakeAll => { WakeMode::WakeAll => {
if confirm("You are about to wake all configured machines.\nContinue?") {
machines.wakeall(); machines.wakeall();
for host in &machines.list { for host in &machines.list {
println!("Woke {}", host.name) println!("Woke {}", host.name)
} }
} }
}
WakeMode::WakeSome => { WakeMode::WakeSome => {
if ! machines.list.is_empty() {
println!("{}", machines); println!("{}", machines);
let indexes = which_indexes( let indexes = which_indexes(
"Select which hosts to wake up (Comma separated integers): ", "Select which hosts to wake up (Comma separated integers): ",
@ -380,6 +382,10 @@ fn main() -> Result<(), Box<dyn Error>> {
println!("Woke {}", host.name) println!("Woke {}", host.name)
} }
} }
else {
println!("No machines configured yet... Try \"waker --help\" for information about usage");
}
}
_ => println!("undefined"), _ => println!("undefined"),
} }
} }

View file

@ -1,7 +1,5 @@
use std::{ use std::{convert::TryInto, error::Error, net::{Ipv4Addr, ToSocketAddrs, UdpSocket}};
error::Error, use crate::sanitizers::{self, sanitize};
net::{Ipv4Addr, ToSocketAddrs, UdpSocket},
};
// The format of a Wake-on-LAN (WOL) magic packet is defined // The format of a Wake-on-LAN (WOL) magic packet is defined
// as a byte array with 6 bytes of value 255 (0xFF) followed by // as a byte array with 6 bytes of value 255 (0xFF) followed by
@ -34,73 +32,22 @@ impl MagicPacket {
return MagicPacket { bytes: magic_bytes }; return MagicPacket { bytes: magic_bytes };
} }
// This is horrible code
/// Parse a MAC-string into a packet. /// Parse a MAC-string into a packet.
/// Takes both separated and unseparated /// The MAC-string should be 17 characters long, separated by colons (i.e. XX:XX:XX:XX:XX:XX)
pub fn from_str(mac_str: &str) -> Result<MagicPacket, Box<dyn Error>> { pub fn from_str(mac_str: &str) -> Result<MagicPacket, Box<dyn Error>> {
use hex::FromHex; let mac_bytes = MagicPacket::parse_macstr(mac_str, ':')?;
let mut mac_string = mac_str.trim().to_string(); return Ok(MagicPacket::new(&mac_bytes));
if mac_string.len() == 17 {
for i in 0..5 {
mac_string.remove(14 - (i * 3)); // Deal with it
}
} }
let mut mac_arr: [u8; 6]; // This method is a bit allocation heavy.
let hex_vec = Vec::from_hex(mac_string)?; pub fn parse_macstr<S: AsRef<str>>(mac_str: S, sep: char) -> Result<Box<[u8; 6]>, Box<dyn Error>> {
let sanitized_macstr = sanitize(mac_str.as_ref(), sanitizers::AddrType::MAC).unwrap();
unsafe { let bytes_split: Vec<u8> = sanitized_macstr.split(sep)
mac_arr = std::mem::MaybeUninit::uninit().assume_init();
let src: *const u8 = &hex_vec[0];
let dst: *mut u8 = &mut mac_arr[0];
dst.copy_from_nonoverlapping(src, 6);
}
return Ok(MagicPacket::new(&mac_arr));
}
pub fn from_str2(mac_str: &str) -> Result<MagicPacket, Box<dyn Error>> {
return Ok(MagicPacket::new(&MagicPacket::parse(mac_str).unwrap()));
}
// Stolen
fn mac_to_byte(data: &str, sep: char) -> Vec<u8> {
data.split(sep)
.flat_map(|x| hex::decode(x).expect("Invalid mac!")) .flat_map(|x| hex::decode(x).expect("Invalid mac!"))
.collect()
}
// Parses string by position if string is 12+5 characters long (delimited by : for example)
pub fn parse<S: AsRef<str>>(mac_str: S) -> Result<Box<[u8; 6]>, Box<dyn Error>> {
use hex::FromHex;
let mut mstr: &str = mac_str.as_ref();
mstr = mstr.trim();
let mut bytes: [u8; 6] = [0, 0, 0, 0, 0, 0];
for (index, byte) in bytes.iter_mut().enumerate() {
// 0,3,6,9...
let substr = &mstr[3 * index..3 * index + 2];
*byte = <[u8; 1]>::from_hex(substr).unwrap()[0];
}
return Ok(Box::new(bytes));
}
// Loops the string and parses any valid hex
pub fn parse_harder<S: Into<String>>(mac_str: S) -> Result<Box<[u8; 6]>, Box<dyn Error>> {
let mstr: String = mac_str.into();
let mut hexdigits: String = mstr
.chars()
.filter(|c| char::is_ascii_hexdigit(c))
.collect(); .collect();
let mut bytes: [u8; 6] = [0; 6];
if hexdigits.len() >= 12 { let arr: [u8; 6] = bytes_split.try_into().unwrap();
hexdigits.truncate(12); // May not be needed, since bytes is only 6 bytes, and from_hex might be smart enough to realize... Ok(Box::new(arr))
//let bytes: [u8; 6] = hex::FromHex::from_hex(hexdigits)?;
bytes = hex::FromHex::from_hex(hexdigits)?;
}
return Ok(Box::new(bytes));
} }
/// Send packet to/from specific address/interface /// Send packet to/from specific address/interface
@ -118,11 +65,6 @@ impl MagicPacket {
(Ipv4Addr::new(0x0, 0x0, 0x0, 0x0), 0), (Ipv4Addr::new(0x0, 0x0, 0x0, 0x0), 0),
) )
} }
/// Return raw bytes, ready to be broadcasted
pub fn get_bytes(&self) -> &[u8; 102] {
&self.bytes
}
} }
#[cfg(test)] #[cfg(test)]
@ -142,7 +84,7 @@ mod tests {
#[test] #[test]
fn magic_packet_from_new() { fn magic_packet_from_new() {
let packet3 = MagicPacket::from_str2("10:10:10:10:10:10").unwrap(); let packet3 = MagicPacket::from_str("10:10:10:10:10:10").unwrap();
//let vec: [u8; 102] = [0xFF; 102]; //let vec: [u8; 102] = [0xFF; 102];
let slice = &packet3.bytes[packet3.bytes.len() - 6..]; let slice = &packet3.bytes[packet3.bytes.len() - 6..];
for a in slice { for a in slice {
@ -155,15 +97,9 @@ mod tests {
#[test] #[test]
fn test_parse() { fn test_parse() {
let mp = MagicPacket::parse("ff:ff:ff:ff:ff:ff").unwrap(); let mp = MagicPacket::parse_macstr("ff:ff:ff:ff:ff:ff", ':').unwrap();
assert_eq!([0xFF; 6], *mp); assert_eq!([0xFF; 6], *mp);
let mp2 = MagicPacket::parse("10:10:10:10:10:10").unwrap(); let mp2 = MagicPacket::parse_macstr("10:10:10:10:10:10", ':').unwrap();
assert_eq!([0x10; 6], *mp2); assert_eq!([0x10; 6], *mp2);
} }
#[test]
fn parse_harder_test() {
let mac = MagicPacket::parse_harder("10:10:10:10:10:10").unwrap();
assert_eq!([0x10; 6], *mac);
}
} }

View file

@ -1,32 +0,0 @@
{
"list": [
{
"name": "Name",
"macs": [
"E9:A3:29:73:FE:D2"
],
"ips": []
},
{
"name": "i",
"macs": [
"7D:E1:AC:94:29:79"
],
"ips": []
},
{
"name": "PartyMaskinen",
"macs": [
"96:F8:FF:30:4C:EE"
],
"ips": []
},
{
"name": "Kalle",
"macs": [
"FF:FF:FF:FF:FF:FF"
],
"ips": []
}
]
}