pluralsync/src/main.rs

228 lines
6.4 KiB
Rust
Raw Normal View History

2023-11-20 15:39:40 +00:00
use std::path::Path;
use std::{fs, collections::HashMap};
use reqwest::header::{USER_AGENT, AUTHORIZATION};
use serde::{Serialize, Deserialize};
use serde_json::Value;
use dirs;
const PK_URL: &str = "https://api.pluralkit.me/v2";
const SP_URL: &str = "https://api.apparyllis.com/v1";
#[derive(Debug)]
#[derive(Serialize)]
struct System {
pk_userid: String,
sp_userid: String,
members: Vec<Member>,
}
#[derive(Debug)]
#[derive(Serialize)]
#[derive(Clone)]
struct Member {
pk_id: String,
sp_id: String,
name: String,
alias: String
}
fn main() {
if std::env::args().len() > 1 {
let command = std::env::args().nth(1).expect("No command given");
let config_path: String;
match dirs::config_dir() {
Some(x) => {
config_path = format!("{}/pluralshit", x.display());
match command.as_str() {
"sync" => sync(config_path),
"set" => {
if std::env::args().len() > 2 {
set_member(config_path, std::env::args().nth(2).expect("No member given"));
} else {
set_empty(config_path);
}
},
// TODO "get" => get()
// TODO "add op$2" => add() or add($2)
// TODO "setgroup" => setgroup()
&_ => println!("Invalid command"),
}
},
None => println!("Something went wrong")
}
} else {
println!("No arguments given");
}
}
fn set_member(config_path: String, member: String) {
let config = load_json(format!("{}/config.json", config_path));
let system = get_system(config_path);
//for mem in system["members"].as_str().unwrap() {
// TODO
// TODO Hacer que devuelva el json en forma de Vec<Member>
// TODO
//}
}
fn set_empty(config_path: String) {
let config = load_json(format!("{}/config.json", config_path));
let system = get_system(config_path);
}
fn sync(config_path: String) {
// Get config
let config = load_json(format!("{}/config.json", config_path));
let pk_key = &config["pk_key"].as_str().unwrap();
let sp_key = &config["sp_key"].as_str().unwrap();
// Get Pluralkit system id
let pk_sys = pk_get_system(pk_key);
let pk_sysid = pk_sys["id"].as_str().unwrap();
// Get Simplyplural user id
let sp_user_id = sp_get_userid(sp_key);
// Get Simplyplural member ids
let sp_member_ids = sp_get_memberids(sp_key, &sp_user_id);
// get members
let pk_members = pk_get_members(pk_key, pk_sysid);
let mut members: Vec<Member> = Vec::new();
for member in pk_members {
let mut m = Member {
pk_id: member["id"].as_str().unwrap().to_string(),
sp_id: String::new(),
name: member["name"].as_str().unwrap().to_string(),
alias: String::new()
};
if member["display_name"].as_str() != None {
m.alias = member["display_name"].as_str().unwrap().to_string();
} else {
m.alias = String::from(&m.name);
}
m.sp_id = get_sp_id(&m, &sp_member_ids);
members.push(m);
}
let sys = System {
pk_userid: pk_sysid.to_string(),
sp_userid: sp_user_id,
members: members.clone(),
};
let json = serde_json::to_string(&sys);
let _ = fs::write(format!("{}/system.json", config_path), &json.unwrap());
}
// Utilities
fn load_json(path: String) -> Value {
if Path::new(&path).exists() {
let config_data = fs::read_to_string(path).expect("File not found");
return serde_json::from_str(&config_data).unwrap();
} else {
println!("Config file in {path} not found");
return Value::Null;
}
}
// TODO
// TODO fn get_config() -> Value {
// TODO load_json
// TODO if Value:null - > mkconfig dir - > make empty config.json
// TODO else -> return
// TODO }
// TODO
fn get_system(config_path: String) -> Value{
let path = format!("{}/system.json", config_path);
// TODO
// TODO load_json
// TODO if Value:null - > sync - > load_json
// TODO
if Path::new(&path).exists() {
let config_data = fs::read_to_string(path).expect("File not found");
return serde_json::from_str(&config_data).unwrap();
} else {
println!("System file in {path} not found. Syncing");
sync(config_path);
let config_data = fs::read_to_string(path).expect("File not found");
return serde_json::from_str(&config_data).unwrap();
}
}
// Simplyplural
fn sp_get_userid(key: &str) -> String {
let url = format!("{}/me", SP_URL);
let res = http_request(url,key);
let json_res : Value = serde_json::from_str(&res.unwrap()).unwrap();
return json_res["id"].as_str().unwrap().to_string();
}
fn sp_get_memberids(key: &str, system_id: &str) -> HashMap<String, String> {
let url = format!("{}/members/{}", SP_URL, system_id);
let res = http_request(url,key);
let datas: Vec<Value> = serde_json::from_str(&res.unwrap()).unwrap();
let mut sp_memberdata: HashMap<String, String> = HashMap::new();
for data in datas {
sp_memberdata.insert(String::from(data["content"]["name"].as_str().unwrap()), String::from(data["id"].as_str().unwrap()));
}
return sp_memberdata;
}
fn get_sp_id(mem: &Member, ids: &HashMap<String, String>) -> String {
let mut member_id = String::new();
for (mn, mid) in ids {
if &mem.name == mn || &mem.alias == mn {
member_id = String::from(mid);
}
}
return member_id;
}
// Pluralkit
fn pk_get_system(key: &str) -> Value {
let url = format!("{}/systems/@me", PK_URL);
let res = http_request(url,key);
return serde_json::from_str(&res.unwrap()).unwrap();
}
fn pk_get_members(key: &str, sysid: &str) -> Vec<Value> {
let url = format!("{}/systems/{}/members", PK_URL, sysid);
let res = http_request(url,key);
let datas: Vec<Value> = serde_json::from_str(&res.unwrap()).unwrap();
return datas;
}
// HTTP Request
#[tokio::main]
async fn http_request(url: String, key: &str) -> Result<String, Box<dyn std::error::Error>> {
let client = reqwest::Client::new();
let res = client
.get(url)
.header(USER_AGENT, "Pluralsync")
.header(AUTHORIZATION, key)
.send()
.await?
.text()
.await?;
Ok(res)
}