Compare commits
	
		
			4 Commits 
		
	
	
		
			319d8e6d29
			...
			5ad59b9347
		
	
	| Author | SHA1 | Date | 
|---|---|---|
|  _ | 5ad59b9347 | |
|  _ | 0286d854d6 | |
|  _ | 5aeb4c8d7a | |
|  _ | b3a576e13d | 
|  | @ -2,6 +2,12 @@ | ||||||
| # It is not intended for manual editing. | # It is not intended for manual editing. | ||||||
| version = 4 | version = 4 | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "anyhow" | ||||||
|  | version = "1.0.97" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "dcfed56ad506cb2c684a14971b8861fdc3baaaae314b9e5f9bb532cbe3ba7a4f" | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "autocfg" | name = "autocfg" | ||||||
| version = "1.1.0" | version = "1.1.0" | ||||||
|  | @ -20,6 +26,21 @@ version = "2.9.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "5c8214115b7bf84099f1309324e63141d4c5d7cc26862f97a0a857dbefe165bd" | checksum = "5c8214115b7bf84099f1309324e63141d4c5d7cc26862f97a0a857dbefe165bd" | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "camino" | ||||||
|  | version = "1.1.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8b96ec4966b5813e2c0507c1f86115c8c5abaadc3980879c3424042a02fd1ad3" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "cc" | ||||||
|  | version = "1.2.17" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "1fcb57c740ae1daf453ae85f16e37396f672b039e00d9d866e07ddb24e328e3a" | ||||||
|  | dependencies = [ | ||||||
|  |  "shlex", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "cfg-if" | name = "cfg-if" | ||||||
| version = "1.0.0" | version = "1.0.0" | ||||||
|  | @ -86,13 +107,16 @@ dependencies = [ | ||||||
| 
 | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "lookaround" | name = "lookaround" | ||||||
| version = "0.1.6" | version = "0.1.7" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  |  "anyhow", | ||||||
|  |  "camino", | ||||||
|  "configparser", |  "configparser", | ||||||
|  "directories", |  "directories", | ||||||
|  "mac_address", |  "mac_address", | ||||||
|  "nix", |  "nix", | ||||||
|  "rand", |  "rand", | ||||||
|  |  "sys-info", | ||||||
|  "thiserror", |  "thiserror", | ||||||
|  "tokio", |  "tokio", | ||||||
| ] | ] | ||||||
|  | @ -249,6 +273,12 @@ dependencies = [ | ||||||
|  "redox_syscall", |  "redox_syscall", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "shlex" | ||||||
|  | version = "1.3.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "syn" | name = "syn" | ||||||
| version = "1.0.82" | version = "1.0.82" | ||||||
|  | @ -260,6 +290,16 @@ dependencies = [ | ||||||
|  "unicode-xid", |  "unicode-xid", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "sys-info" | ||||||
|  | version = "0.9.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "0b3a0d0aba8bf96a0e1ddfdc352fc53b3df7f39318c71854910c3c4b024ae52c" | ||||||
|  | dependencies = [ | ||||||
|  |  "cc", | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "thiserror" | name = "thiserror" | ||||||
| version = "1.0.30" | version = "1.0.30" | ||||||
|  |  | ||||||
|  | @ -9,14 +9,17 @@ license = "AGPL-3.0" | ||||||
| name = "lookaround" | name = "lookaround" | ||||||
| readme = "README.md" | readme = "README.md" | ||||||
| repository = "https://six-five-six-four.com/git/reactor/lookaround" | repository = "https://six-five-six-four.com/git/reactor/lookaround" | ||||||
| version = "0.1.6" | version = "0.1.7" | ||||||
| 
 | 
 | ||||||
| [dependencies] | [dependencies] | ||||||
|  | anyhow = "1.0.97" | ||||||
|  | camino = "1.1.9" | ||||||
| configparser = "3.0.0" | configparser = "3.0.0" | ||||||
| directories = "5.0.0" | directories = "5.0.0" | ||||||
| mac_address = "1.1.8" | mac_address = "1.1.8" | ||||||
| nix = "0.29.0" | nix = "0.29.0" | ||||||
| rand = "0.8.4" | rand = "0.8.4" | ||||||
|  | sys-info = "0.9.1" | ||||||
| thiserror = "1.0.30" | thiserror = "1.0.30" | ||||||
| tokio = { version = "1.14.0", features = ["fs", "net", "rt", "time"] } | tokio = { version = "1.14.0", features = ["fs", "net", "rt", "time"] } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										39
									
								
								README.md
								
								
								
								
							
							
						
						
									
										39
									
								
								README.md
								
								
								
								
							|  | @ -28,51 +28,38 @@ Make sure Cargo is installed from [RustUp.](https://rustup.rs/) | ||||||
| cargo install lookaround | cargo install lookaround | ||||||
| 
 | 
 | ||||||
| # Find your config directory | # Find your config directory | ||||||
| # Prints something like `Using config dir "/home/user/.config/lookaround"` | # e.g. `$HOME/.config/lookaround` | ||||||
| lookaround config | lookaround config | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| Create the files `client.ini` and/or `server.ini` in that directory | Use `$HOME/.config/lookaround/client.ini` as a hosts file if it's more convenient | ||||||
| (e.g. /home/user/.config/lookaround/server.ini) | for your client to be the source of truth for nicknames. | ||||||
| 
 | 
 | ||||||
| ```ini | ```ini | ||||||
| # Clients can store MAC-nickname pairs in client.ini, like a hosts file. |  | ||||||
| # This is useful if your servers are short-lived and you want the clients |  | ||||||
| # to be the source of truth for nicknames. |  | ||||||
| [nicknames] | [nicknames] | ||||||
| 11-11-11-11-11-11 = laptop | 11-11-11-11-11-11 = bob-laptop | ||||||
| 22-22-22-22-22-22 = desktop | 22-22-22-22-22-22 = alice-desktop | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|  | LookAround will use the system's hostname as its server nickname. | ||||||
|  | If you have a generic hostname like `ubuntu`, override this in `$HOME/.config/lookaround/server.ini`: | ||||||
|  | 
 | ||||||
| ```ini | ```ini | ||||||
| # Long-lived servers can have their nickname configured in server.ini |  | ||||||
| [server] | [server] | ||||||
| nickname = my-computer | nickname = alice-desktop | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ## Auto-Start (Linux) | ## Auto-Start (Linux) | ||||||
| 
 | 
 | ||||||
| Put this systemd unit in `~/.config/systemd/user/lookaround.service`: | Run `lookaround install` or `lookaround install $NICKNAME` if you want to set a nickname. | ||||||
| 
 | 
 | ||||||
| ```ini | This will create `$HOME/.config/lookaround/server.ini` and `$HOME/.config/systemd/user/lookaround.service`, and start the systemd user service. | ||||||
| [Unit] |  | ||||||
| Description=LookAround |  | ||||||
| 
 | 
 | ||||||
| [Service] | To check that it's running | ||||||
| ExecStart=/home/user/.cargo/bin/lookaround server |  | ||||||
| Restart=always |  | ||||||
| 
 |  | ||||||
| [Install] |  | ||||||
| WantedBy=default.target |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| Then start the service, check that it's running okay, and enable it for |  | ||||||
| auto-start: |  | ||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| systemctl --user start lookaround |  | ||||||
| systemctl --user status lookaround | systemctl --user status lookaround | ||||||
| systemctl --user enable lookaround | lookaround client | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ## Auto-Start (Windows) | ## Auto-Start (Windows) | ||||||
|  |  | ||||||
|  | @ -1,11 +1,18 @@ | ||||||
| use crate::prelude::*; | use crate::prelude::*; | ||||||
|  | use directories::ProjectDirs; | ||||||
| 
 | 
 | ||||||
| pub const LOOKAROUND_VERSION: &str = env!("CARGO_PKG_VERSION"); | pub fn try_project_dir() -> Option<ProjectDirs> { | ||||||
| 
 |  | ||||||
| pub fn find_project_dirs() -> Option<ProjectDirs> { |  | ||||||
|     ProjectDirs::from("", "ReactorScram", "LookAround") |     ProjectDirs::from("", "ReactorScram", "LookAround") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | pub fn try_config_dir() -> Option<PathBuf> { | ||||||
|  |     Some(try_project_dir()?.config_local_dir().into()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub fn try_server_config_path() -> Option<PathBuf> { | ||||||
|  |     Some(try_config_dir()?.join("server.ini")) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #[derive(Debug, thiserror::Error)] | #[derive(Debug, thiserror::Error)] | ||||||
| pub enum AppError { | pub enum AppError { | ||||||
|     #[error(transparent)] |     #[error(transparent)] | ||||||
|  |  | ||||||
|  | @ -1,26 +0,0 @@ | ||||||
| type Mac = [u8; 6]; |  | ||||||
| 
 |  | ||||||
| pub fn debug() { |  | ||||||
|     for input in [ |  | ||||||
|         [0, 0, 0, 0, 0, 0], |  | ||||||
|         [0, 0, 0, 0, 0, 1], |  | ||||||
|         [1, 0, 0, 0, 0, 0], |  | ||||||
|         [1, 0, 0, 0, 0, 1], |  | ||||||
|     ] { |  | ||||||
|         assert_eq!(unmix(mix(input)), input); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     println!("Passed"); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // NOT intended for any cryptography or security. This is TRIVIALLY reversible.
 |  | ||||||
| // It's just to make it easier for humans to tell apart MACs where only a couple
 |  | ||||||
| // numbers differ.
 |  | ||||||
| 
 |  | ||||||
| fn mix(i: Mac) -> Mac { |  | ||||||
|     [i[0] ^ i[5], i[1] ^ i[4], i[2] ^ i[3], i[3], i[4], i[5]] |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| fn unmix(i: Mac) -> Mac { |  | ||||||
|     [i[0] ^ i[5], i[1] ^ i[4], i[2] ^ i[3], i[3], i[4], i[5]] |  | ||||||
| } |  | ||||||
|  | @ -165,9 +165,9 @@ fn configure_client<I: Iterator<Item = String>>(mut args: I) -> Result<ClientPar | ||||||
| fn load_config_file() -> ConfigFile { | fn load_config_file() -> ConfigFile { | ||||||
|     let mut nicknames: HashMap<String, String> = Default::default(); |     let mut nicknames: HashMap<String, String> = Default::default(); | ||||||
| 
 | 
 | ||||||
|     if let Some(proj_dirs) = find_project_dirs() { |     if let Some(dir) = app_common::try_config_dir() { | ||||||
|  |         let path = dir.join("client.ini"); | ||||||
|         let mut ini = Ini::new_cs(); |         let mut ini = Ini::new_cs(); | ||||||
|         let path = proj_dirs.config_local_dir().join("client.ini"); |  | ||||||
|         if ini.load(&path).is_ok() { |         if ini.load(&path).is_ok() { | ||||||
|             let map_ref = ini.get_map_ref(); |             let map_ref = ini.get_map_ref(); | ||||||
|             if let Some(x) = map_ref.get("nicknames") { |             if let Some(x) = map_ref.get("nicknames") { | ||||||
|  |  | ||||||
|  | @ -0,0 +1,83 @@ | ||||||
|  | use crate::prelude::*; | ||||||
|  | use anyhow::{Context as _, Result, anyhow, ensure}; | ||||||
|  | use std::{io::Write as _, process::Command}; | ||||||
|  | 
 | ||||||
|  | pub(crate) fn main(nickname: Option<&str>) -> Result<()> { | ||||||
|  |     if let Some(nickname) = nickname { | ||||||
|  |         let path = app_common::try_server_config_path().context("can't find config dir")?; | ||||||
|  |         let text = format!( | ||||||
|  |             "[server]
 | ||||||
|  | nickname = {nickname} | ||||||
|  | " | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         write_new(&path, &text).context("Didn't install LookAround server.ini")?; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     setup_autostart()?; | ||||||
|  | 
 | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(not(target_os = "linux"))] | ||||||
|  | fn setup_autostart() -> Result<()> { | ||||||
|  |     anyhow::bail!("LookAround autostart is only implemented for systemd + GNU/Linux"); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(target_os = "linux")] | ||||||
|  | fn setup_autostart() -> Result<()> { | ||||||
|  |     let dirs = directories::BaseDirs::new().context("Could not find home dir")?; | ||||||
|  |     let home = dirs.config_local_dir(); | ||||||
|  |     let path = home.join("systemd").join("user").join("lookaround.service"); | ||||||
|  | 
 | ||||||
|  |     let exe_path = | ||||||
|  |         Utf8PathBuf::from_path_buf(std::env::current_exe().context("Can't get current_exe")?) | ||||||
|  |             .map_err(|_| anyhow!("current_exe isn't valid UTF-8"))?; | ||||||
|  |     let text = format!( | ||||||
|  |         "[Unit]
 | ||||||
|  | Description=LookAround | ||||||
|  | 
 | ||||||
|  | [Service] | ||||||
|  | ExecStart={exe_path} server | ||||||
|  | Restart=always | ||||||
|  | 
 | ||||||
|  | [Install] | ||||||
|  | WantedBy=default.target | ||||||
|  | " | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     write_new(&path, &text).context("Didn't install systemd service unit")?; | ||||||
|  | 
 | ||||||
|  |     ensure!( | ||||||
|  |         Command::new("systemctl") | ||||||
|  |             .args(["--user", "start", "lookaround"]) | ||||||
|  |             .status()? | ||||||
|  |             .success(), | ||||||
|  |         "starting service failed" | ||||||
|  |     ); | ||||||
|  |     ensure!( | ||||||
|  |         Command::new("systemctl") | ||||||
|  |             .args(["--user", "enable", "lookaround"]) | ||||||
|  |             .status()? | ||||||
|  |             .success(), | ||||||
|  |         "enabling service failed" | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn write_new(path: &Path, content: &str) -> Result<()> { | ||||||
|  |     use std::fs; | ||||||
|  |     fs::create_dir_all( | ||||||
|  |         path.parent() | ||||||
|  |             .context("Impossible, path should always have a parent")?, | ||||||
|  |     )?; | ||||||
|  |     match fs::File::create_new(path) { | ||||||
|  |         Err(err) if err.kind() == std::io::ErrorKind::AlreadyExists => { | ||||||
|  |             eprintln!("File already exists, won't overwrite: `{path:?}`") | ||||||
|  |         } | ||||||
|  |         Err(err) => Err(err)?, | ||||||
|  |         Ok(mut f) => f.write_all(content.as_bytes())?, | ||||||
|  |     } | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
							
								
								
									
										42
									
								
								src/main.rs
								
								
								
								
							
							
						
						
									
										42
									
								
								src/main.rs
								
								
								
								
							|  | @ -1,15 +1,16 @@ | ||||||
|  | use anyhow::{Result, bail}; | ||||||
| use prelude::*; | use prelude::*; | ||||||
| 
 | 
 | ||||||
| pub mod app_common; | pub mod app_common; | ||||||
| mod avalanche; |  | ||||||
| mod client; | mod client; | ||||||
|  | mod install; | ||||||
| mod ip; | mod ip; | ||||||
| pub mod message; | pub mod message; | ||||||
| mod prelude; | mod prelude; | ||||||
| mod server; | mod server; | ||||||
| pub mod tlv; | pub mod tlv; | ||||||
| 
 | 
 | ||||||
| fn main() -> Result<(), AppError> { | fn main() -> Result<()> { | ||||||
|     let rt = tokio::runtime::Builder::new_current_thread() |     let rt = tokio::runtime::Builder::new_current_thread() | ||||||
|         .enable_io() |         .enable_io() | ||||||
|         .enable_time() |         .enable_time() | ||||||
|  | @ -20,31 +21,38 @@ fn main() -> Result<(), AppError> { | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| async fn async_main() -> Result<(), AppError> { | async fn async_main() -> Result<()> { | ||||||
|     let mut args = env::args(); |     let mut args = env::args(); | ||||||
| 
 | 
 | ||||||
|     let _exe_name = args.next(); |     let _exe_name = args.next(); | ||||||
| 
 | 
 | ||||||
|     let subcommand: Option<String> = args.next(); |     let Some(subcommand) = args.next() else { | ||||||
|  |         return Err(CliArgError::MissingSubcommand.into()); | ||||||
|  |     }; | ||||||
| 
 | 
 | ||||||
|     match subcommand.as_ref().map(|x| &x[..]) { |     match subcommand.as_ref() { | ||||||
|         None => return Err(CliArgError::MissingSubcommand.into()), |         "--version" => { | ||||||
|         Some("--version") => println!("lookaround v{}", LOOKAROUND_VERSION), |             println!("lookaround v{}", env!("CARGO_PKG_VERSION")); | ||||||
|         Some("client") => client::client(args).await?, |         } | ||||||
|         Some("config") => config(), |         "client" => client::client(args).await?, | ||||||
|         Some("debug-avalanche") => avalanche::debug(), |         "config" => { | ||||||
|         Some("find-nick") => client::find_nick(args).await?, |             config(); | ||||||
|         Some("my-ips") => my_ips()?, |         } | ||||||
|         Some("server") => server::server(args).await?, |         "find-nick" => client::find_nick(args).await?, | ||||||
|         Some(x) => return Err(CliArgError::UnknownSubcommand(x.to_string()).into()), |         "install" => { | ||||||
|  |             let nickname = args.next(); | ||||||
|  |             install::main(nickname.as_deref())? | ||||||
|  |         } | ||||||
|  |         "my-ips" => my_ips()?, | ||||||
|  |         "server" => server::server(args).await?, | ||||||
|  |         x => bail!("Unknown subcommand `{x}`"), | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn config() { | fn config() { | ||||||
|     if let Some(proj_dirs) = ProjectDirs::from("", "ReactorScram", "LookAround") { |     if let Some(dir) = app_common::try_config_dir() { | ||||||
|         println!("Using config dir {:?}", proj_dirs.config_local_dir()); |         println!("config dir = `{dir:?}`"); | ||||||
|     } else { |     } else { | ||||||
|         println!("Can't detect config dir."); |         println!("Can't detect config dir."); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -3,13 +3,14 @@ pub use std::{ | ||||||
|     env, |     env, | ||||||
|     io::{Cursor, Write}, |     io::{Cursor, Write}, | ||||||
|     net::{Ipv4Addr, SocketAddr, SocketAddrV4}, |     net::{Ipv4Addr, SocketAddr, SocketAddrV4}, | ||||||
|  |     path::{Path, PathBuf}, | ||||||
|     str::FromStr, |     str::FromStr, | ||||||
|     sync::Arc, |     sync::Arc, | ||||||
|     time::Duration, |     time::Duration, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | pub use camino::Utf8PathBuf; | ||||||
| pub use configparser::ini::Ini; | pub use configparser::ini::Ini; | ||||||
| pub use directories::ProjectDirs; |  | ||||||
| pub use mac_address::{MacAddress, get_mac_address}; | pub use mac_address::{MacAddress, get_mac_address}; | ||||||
| pub use rand::RngCore; | pub use rand::RngCore; | ||||||
| pub use tokio::{ | pub use tokio::{ | ||||||
|  | @ -18,9 +19,7 @@ pub use tokio::{ | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| pub use crate::{ | pub use crate::{ | ||||||
|     app_common::{ |     app_common::{self, AppError, CliArgError, recv_msg_from}, | ||||||
|         self, AppError, CliArgError, LOOKAROUND_VERSION, find_project_dirs, recv_msg_from, |  | ||||||
|     }, |  | ||||||
|     ip::get_ips, |     ip::get_ips, | ||||||
|     message::{self, Message, PACKET_SIZE}, |     message::{self, Message, PACKET_SIZE}, | ||||||
|     tlv, |     tlv, | ||||||
|  |  | ||||||
|  | @ -42,11 +42,10 @@ pub async fn server<I: Iterator<Item = String>>(args: I) -> Result<(), AppError> | ||||||
| fn configure<I: Iterator<Item = String>>(mut args: I) -> Result<Params, AppError> { | fn configure<I: Iterator<Item = String>>(mut args: I) -> Result<Params, AppError> { | ||||||
|     let common = app_common::Params::default(); |     let common = app_common::Params::default(); | ||||||
|     let mut bind_addrs = vec![]; |     let mut bind_addrs = vec![]; | ||||||
|     let mut nickname = String::new(); |     let mut nickname = sys_info::hostname().unwrap_or_default(); | ||||||
| 
 | 
 | ||||||
|     if let Some(proj_dirs) = find_project_dirs() { |     if let Some(path) = app_common::try_server_config_path() { | ||||||
|         let mut ini = Ini::new_cs(); |         let mut ini = Ini::new_cs(); | ||||||
|         let path = proj_dirs.config_local_dir().join("server.ini"); |  | ||||||
|         if ini.load(&path).is_ok() { |         if ini.load(&path).is_ok() { | ||||||
|             if let Some(x) = ini.get("server", "nickname") { |             if let Some(x) = ini.get("server", "nickname") { | ||||||
|                 nickname = x; |                 nickname = x; | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue