use crate::State::{Active, Closed, Done, Open}; use nostr_sdk::async_utility::futures_util::TryFutureExt; use nostr_sdk::prelude::*; use once_cell::sync::Lazy; use std::borrow::Borrow; use std::collections::HashMap; use std::env::args; use std::fmt; use std::io::{stdin, stdout, Write}; use std::net::{Ipv4Addr, SocketAddr, SocketAddrV4}; use std::ops::Deref; use std::time::Duration; /* 1: Task Description Issue Tracking: https://github.com/nostr-protocol/nips/blob/master/34.md 1621: MD Issue 1622: MD Reply 1630-1633: Status (Time-tracking, Kanban) Calendar: https://github.com/nostr-protocol/nips/blob/master/52.md 31922 (GANTT, only Date) 31923 (Calendar, with Time) */ static MY_KEYS: Lazy = Lazy::new(|| Keys::generate()); static CLIENT: Lazy = Lazy::new(|| Client::new(MY_KEYS.borrow().deref())); #[tokio::main] async fn main() { let proxy = Some(SocketAddr::V4(SocketAddrV4::new(Ipv4Addr::LOCALHOST, 9050))); CLIENT.add_relay("ws://localhost:4736").await; //CLIENT.add_relay("wss://relay.damus.io").await; //CLIENT // .add_relay_with_opts( // "wss://relay.nostr.info", // RelayOptions::new().proxy(proxy).flags(RelayServiceFlags::default().remove(RelayServiceFlags::WRITE)), // ) // .await?; //CLIENT // .add_relay_with_opts( // "ws://jgqaglhautb4k6e6i2g34jakxiemqp6z4wynlirltuukgkft2xuglmqd.onion", // RelayOptions::new().proxy(proxy), // ) // .await?; //let metadata = Metadata::new() // .name("username") // .display_name("My Username") // .about("Description") // .picture(Url::parse("https://example.com/avatar.png")?) // .banner(Url::parse("https://example.com/banner.png")?) // .nip05("username@example.com") // .lud16("yuki@getalby.com") // .custom_field("custom_field", "my value"); //CLIENT.set_metadata(&metadata).await?; CLIENT.connect().await; let timeout = Duration::from_secs(3); let filter = Filter::new(); let sub_id: SubscriptionId = CLIENT.subscribe(vec![filter.clone()], None).await; repl().await; println!("Finding existing events"); let res = CLIENT .get_events_of(vec![filter], Option::from(timeout)) .map_ok(|res| { for event in res { println!("Found {} {:?}", event.content, event.tags) } }) .await; let mut notifications = CLIENT.notifications(); println!("Listening for events..."); while let Ok(notification) = notifications.recv().await { if let RelayPoolNotification::Event { subscription_id, event, .. } = notification { let kind = event.kind; let content = &event.content; println!("{kind}: {content}"); //break; // Exit } } } fn make_event(text: &str, tags: &[Tag]) -> Event { EventBuilder::new(Kind::from(1621), text, tags.to_vec()) .to_event(&MY_KEYS) .unwrap() } async fn repl() { let mut tasks: HashMap = HashMap::new(); let add_task = |tasks: &mut HashMap, event: Event| tasks.insert(event.id, Task::new(event)); for argument in args().skip(1) { add_task( &mut tasks, make_event(&argument, &[Tag::Hashtag("arg".to_string())]), ); } let mut position: Option = None; let mut properties: Vec = vec!["id".into(), "name".into(), "state".into()]; loop { let mut prompt = String::with_capacity(64); let mut pos = position; while pos.is_some() { let id = pos.unwrap(); let task = tasks.get(&id); prompt = task.map_or(id.to_string(), |t| t.event.content.clone()) + " " + &prompt; pos = task.and_then(|t| t.parent_id()); } print!(" {}> ", prompt); stdout().flush().unwrap(); match stdin().lines().next() { Some(Ok(input)) => { if input.trim() == "exit" { break; } if input.starts_with(".") { match input[1..2].parse::() { Ok(index) => { properties.insert(index, input[2..].to_string()); continue; } Err(_) => { let prop = &input[1..]; let pos = properties.iter().position(|s| s == &prop); match pos { None => { properties.push(prop.to_string()); } Some(i) => { properties.remove(i); } } } } } else { match input.split_once(": ") { None => { position = EventId::parse(input).ok().or(position .and_then(|p| tasks.get(&p)) .and_then(|t| t.parent_id())); } Some(s) => { let mut tags: Vec = s.1.split(" ") .map(|t| Tag::Hashtag(t.to_string())) .collect(); if let Some(pos) = position { tags.push(Tag::event(pos)); } let event = make_event(s.0, &tags); for tag in event.tags.iter() { match tag { Tag::Event { event_id, .. } => { tasks .get_mut(event_id) .unwrap() .children .push(event.clone()); } _ => {} } } add_task(&mut tasks, event); } } }; let events: Vec<&Event> = position.map_or(tasks.values().map(|t| &t.event).collect(), |p| { tasks .get(&p) .map_or(Vec::new(), |t| t.children.iter().collect()) }); println!("{}", properties.join(" ")); for event in events { println!("{}: {}", event.id, event.content); } } _ => {} } } CLIENT .batch_event( tasks.into_values().map(|t| t.event).collect(), RelaySendOptions::new().skip_send_confirmation(true), ) .await .unwrap(); } struct Task { event: Event, children: Vec, } impl Task { fn new(event: Event) -> Task { Task { event, children: Vec::new(), } } fn parent_id(&self) -> Option { for tag in self.event.tags.iter() { match tag { Tag::Event { event_id, .. } => return Some(*event_id), _ => {} } } None } fn states(&self) -> impl Iterator + '_ { self.children.iter().filter_map(|event| { match event.kind.as_u32() { 1630 => Some(Open), 1631 => Some(Done), 1632 => Some(Closed), 1633 => Some(Active), _ => None, } .map(|s| TaskState { name: self.event.content.clone(), state: s, time: self.event.created_at.clone(), }) }) } fn descriptions(&self) -> impl Iterator + '_ { self.children.iter().filter_map(|event| { if event.kind == Kind::TextNote { Some(event.content.clone()) } else { None } }) } fn state(&self) -> Option { self.states().max_by_key(|t| t.time) } fn get(&self, property: &str) -> Option { match property { "id" => Some(self.event.id.to_string()), "parentid" => self.parent_id().map(|i| i.to_string()), "state" => self.state().map(|s| s.state.to_string()), "name" => Some(self.event.content.clone()), "desc" | "description" => self.descriptions().fold(None, |total, s| { Some(match total { None => s, Some(i) => i + " " + &s, }) }), _ => None, } } } struct TaskState { name: String, state: State, time: Timestamp, } #[derive(Debug)] enum State { Open, Active, Done, Closed, } impl fmt::Display for State { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { fmt::Debug::fmt(self, f) } }