telegram_notes_bot/src/main.rs

432 lines
15 KiB
Rust
Raw Normal View History

2019-09-12 07:53:43 +00:00
use chrono::prelude::*;
2020-05-04 15:14:20 +00:00
use futures_util::stream::StreamExt;
2019-09-12 07:53:43 +00:00
use maplit::hashmap;
use mime::Mime;
2020-05-04 15:14:20 +00:00
use once_cell::sync::Lazy;
use reqwest::Client;
use serde_derive::Deserialize;
use serde_json::json;
2019-09-12 07:53:43 +00:00
use telegram_bot::*;
2020-05-04 15:14:20 +00:00
use telegram_bot::types::{EditMessageText, InlineKeyboardButton, InlineKeyboardMarkup, SendMessage};
use thiserror::Error;
use tokio::task;
2019-09-12 07:53:43 +00:00
use url::Url;
use std::collections::HashMap;
use std::env;
2020-05-04 15:14:20 +00:00
use std::sync::Arc;
use std::time::Duration;
2019-09-12 07:53:43 +00:00
mod ical_parsing;
static TELEGRAM_BOT_TOKEN: Lazy<String> = Lazy::new(|| {
env::var("TELEGRAM_BOT_TOKEN").expect("TELEGRAM_BOT_TOKEN not set")
});
2020-05-04 15:14:20 +00:00
static API: Lazy<Arc<Api>> = Lazy::new(|| {
println!("Initializing Telegram API..");
Arc::new(Api::new(&*TELEGRAM_BOT_TOKEN))
2020-05-04 15:14:20 +00:00
});
2020-05-04 15:36:19 +00:00
static TRILIUM_TOKEN: Lazy<String> = Lazy::new(|| {
println!("Loading passwords..");
let trilium_user = env::var("TRILIUM_USER").expect("TRILIUM_USER not set");
let trilium_password = env::var("TRILIUM_PASSWORD").expect("TRILIUM_PASSWORD not set");
let client = reqwest::blocking::Client::new();
let resp: HashMap<String, String> = client.post("http://localhost:9001/api/login/token")
.json(&hashmap!{ "username" => &trilium_user, "password" => &trilium_password })
.send().unwrap().json().unwrap();
resp["token"].clone()
});
2020-05-04 15:14:20 +00:00
static OWNER: Lazy<UserId> = Lazy::new(|| {
2019-09-12 07:53:43 +00:00
println!("Loading configuration..");
2020-05-04 15:14:20 +00:00
UserId::new(env::var("TELEGRAM_USER_ID").expect("TELEGRAM_USER_ID not set").parse().expect("TELEGRAM_USER_ID not numeric"))
});
static CLIENT: Lazy<Client> = Lazy::new(Client::new);
#[tokio::main]
async fn main() -> Result<(), Error> {
&*OWNER;
2019-09-12 07:53:43 +00:00
2020-05-04 15:14:20 +00:00
&*API;
2019-09-12 07:53:43 +00:00
println!("Initializing Trilium API..");
// Trilium login:
// curl 'http://localhost:9001/api/login/token' -H 'User-Agent: Mozilla/5.0 ..' -H 'Accept: application/json' -H 'Accept-Language: en' --compressed -H 'Content-Type: application/json' -H 'Origin: moz-extension://13bc3fd7-5cb0-4d48-b368-76e389fd7c5f' -H 'DNT: 1' -H 'Connection: keep-alive' --data '{"username":"username","password":"insert_password_here"}'
// -> {"token":"icB3xohFDpkVt7YFpbTflUYC8pucmryVGpb1DFpd6ns="}
2020-05-04 15:36:19 +00:00
println!("Acquired token: {}", *TRILIUM_TOKEN);
2019-09-12 07:53:43 +00:00
println!("Init done!");
2020-05-04 15:14:20 +00:00
task::spawn(async move {
start_polling().await;
});
2020-05-23 15:23:36 +00:00
task::spawn(async move {
event_alerts().await;
});
2020-05-04 15:14:20 +00:00
let mut reminder_msg = MessageId::new(1);
let mut reminder_text = String::new();
let mut reminder_start = Local::now();
let mut reminder_time = chrono::Duration::minutes(0);
2019-09-12 07:53:43 +00:00
// Fetch new updates via long poll method
2020-05-04 15:14:20 +00:00
let mut stream = API.stream();
2019-09-12 07:53:43 +00:00
while let Some(update) = stream.next().await {
2020-05-04 15:14:20 +00:00
if update.is_err() {
2020-05-04 15:36:19 +00:00
println!("Telegram error: {:?}", update.err().unwrap());
2020-05-04 15:14:20 +00:00
continue;
}
let update = update.unwrap();
2019-09-12 07:53:43 +00:00
2020-05-04 15:36:19 +00:00
if let Err(e) = process_one(update, &mut reminder_msg, &mut reminder_text, &mut reminder_start, &mut reminder_time).await {
println!("Error: {}", e);
}
}
Ok(())
}
async fn process_one(update: Update, reminder_msg: &mut MessageId, reminder_text: &mut String, reminder_start: &mut DateTime<Local>, reminder_time: &mut chrono::Duration) -> Result<(), Error> {
if let UpdateKind::Message(message) = update.kind {
let now = Local::now();
2019-09-12 07:53:43 +00:00
2020-05-04 15:36:19 +00:00
println!("[{}-{:02}-{:02} {:02}:{:02}] Receiving msg {:?}", now.year(), now.month(), now.day(), now.hour(), now.minute(), message);
if message.from.id != *OWNER {
// don't handle message
return Ok(());
}
if let MessageKind::Text { ref data, .. } = message.kind {
if data == "/remindme" {
let mut msg = SendMessage::new(*OWNER, "in 0m: new reminder");
msg.reply_markup(get_keyboard());
*reminder_msg = API.send(msg).await?.to_message_id();
*reminder_text = "new reminder".to_owned();
*reminder_time = chrono::Duration::minutes(0);
*reminder_start = Local::now();
return Ok(());
} else if !reminder_text.is_empty() {
*reminder_text = data.to_owned();
let mut edit = EditMessageText::new(*OWNER, *reminder_msg, format!("in {}: {}", format_time(*reminder_time), reminder_text));
edit.reply_markup(get_keyboard());
API.send(edit).await?;
return Ok(());
2019-09-12 07:53:43 +00:00
}
2020-05-04 15:36:19 +00:00
let is_url = false; //Url::parse(&data).is_ok(); // TODO: read this data from the Telegram json data (utf16 idxes..)
let formatted_text = if is_url {
format!("<ul><li><a href=\"{}\">{}</a></li></ul>", data, data)
} else {
format!("<ul><li>{}</li></ul>", data)
};
let title = format!("{} found at {:02}:{:02}", if is_url { "URL" } else { "content" }, now.hour(), now.minute());
create_text_note(&CLIENT, &*TRILIUM_TOKEN,
&title,
&formatted_text
).await?;
// answer message
if is_url {
API.send(message.text_reply("URL saved :-)")).await?;
} else {
API.send(message.text_reply("Text saved :-)")).await?;
2020-05-04 15:14:20 +00:00
}
} else if let MessageKind::Document { ref data, ref caption, .. } = message.kind {
let document = data;
let get_file = GetFile::new(&document);
let file = API.send(get_file).await?;
let url = file.get_url(&TELEGRAM_BOT_TOKEN).ok_or_else(|| error("url is none"))?;
let data = CLIENT.get(&url).send().await?.bytes().await?;
let mime: Mime = document.mime_type.as_ref().ok_or_else(|| error("no mime type"))?.parse()?;
match (mime.type_(), mime.subtype()) {
(mime::TEXT, x) if x == "calendar" => {
let text = String::from_utf8_lossy(&data);
let text = text.replace("\n<", "<"); // newlines in HTML values
2020-05-23 15:04:00 +00:00
//send_message(&text).await?;
let calendar = ical_parsing::parse_calendar(&text)?;
2020-05-23 15:04:00 +00:00
//send_message(format!("{:?}", calendar)).await?;
if calendar.events.len() != 1 {
return Ok(());
}
if CLIENT.get("http://localhost:9001/custom/new_event").form(&json!({
"name": calendar.events[0].summary,
"summary": calendar.events[0].description_html.as_deref().unwrap_or(&calendar.events[0].description),
"fileName": document.file_name,
"fileData": text,
"location": calendar.events[0].location,
"startTime": calendar.events[0].start.format("%Y-%m-%dT%H:%M:%S").to_string(),
"endTime": calendar.events[0].end.format("%Y-%m-%dT%H:%M:%S").to_string(),
})).send().await?.status().is_success() {
send_message("Event saved :-)").await?;
} else {
send_message("error saving event").await?;
}
},
2020-05-23 15:04:00 +00:00
_ => {
send_message(format!("Document {:?} {:?} {:?} {:?}", caption, document.file_id, document.file_name, document.mime_type)).await?;
}
}
2019-09-12 07:53:43 +00:00
}
2020-05-04 15:36:19 +00:00
} else if let UpdateKind::CallbackQuery(cb) = update.kind {
match &*cb.data.unwrap_or_default() {
"10m_cb" => {
*reminder_time = reminder_time.checked_add(&chrono::Duration::minutes(10)).unwrap();
let mut edit = EditMessageText::new(*OWNER, *reminder_msg, format!("in {}: {}", format_time(*reminder_time), reminder_text));
edit.reply_markup(get_keyboard());
API.send(edit).await?;
},
"1h_cb" => {
*reminder_time = reminder_time.checked_add(&chrono::Duration::hours(1)).unwrap();
let mut edit = EditMessageText::new(*OWNER, *reminder_msg, format!("in {}: {}", format_time(*reminder_time), reminder_text));
edit.reply_markup(get_keyboard());
API.send(edit).await?;
},
"1d_cb" => {
*reminder_time = reminder_time.checked_add(&chrono::Duration::days(1)).unwrap();
let mut edit = EditMessageText::new(*OWNER, *reminder_msg, format!("in {}: {}", format_time(*reminder_time), reminder_text));
edit.reply_markup(get_keyboard());
API.send(edit).await?;
},
"1w_cb" => {
*reminder_time = reminder_time.checked_add(&chrono::Duration::days(7)).unwrap();
let mut edit = EditMessageText::new(*OWNER, *reminder_msg, format!("in {}: {}", format_time(*reminder_time), reminder_text));
edit.reply_markup(get_keyboard());
API.send(edit).await?;
},
"save_cb" => {
let remind_time = *reminder_start + *reminder_time;
CLIENT.get("http://localhost:9001/custom/new_reminder").form(&json!({
"time": remind_time.to_rfc3339(),
"task": *reminder_text
2020-05-23 15:04:00 +00:00
})).send().await?;
2020-05-04 15:36:19 +00:00
API.send(SendMessage::new(*OWNER, "Reminder saved :-)")).await?;
*reminder_text = String::new();
},
_ => {}
}
} else {
println!("{:?}", update.kind);
2019-09-12 07:53:43 +00:00
}
2020-05-04 15:36:19 +00:00
Ok(())
2019-09-12 07:53:43 +00:00
}
2020-05-04 15:14:20 +00:00
fn get_keyboard() -> InlineKeyboardMarkup {
let mut keyboard = InlineKeyboardMarkup::new();
let key = InlineKeyboardButton::callback("10m", "10m_cb");
let key2 = InlineKeyboardButton::callback("1h", "1h_cb");
let key3 = InlineKeyboardButton::callback("1d", "1d_cb");
let key4 = InlineKeyboardButton::callback("1w", "1w_cb");
keyboard.add_row(vec![key, key2, key3, key4]);
let key = InlineKeyboardButton::callback("save", "save_cb");
keyboard.add_row(vec![key]);
keyboard
}
async fn create_text_note(client: &Client, trilium_token: &str, title: &str, content: &str) -> Result<(), Error> {
2019-09-12 07:53:43 +00:00
// creating a note:
// curl 'http://localhost:9001/api/clipper/notes'
// -H 'Accept: */*' -H 'Accept-Language: en' --compressed -H 'Content-Type: application/json'
// -H 'Authorization: icB3xohFDpkVt7YFpbTflUYC8pucmryVGpb1DFpd6ns='
// -H 'Origin: moz-extension://13bc3fd7-5cb0-4d48-b368-76e389fd7c5f'
// --data '{"title":"line 1","content":"<p>line 2</p><p>line 3</p>","clipType":"note"}'
client.post("http://localhost:9001/api/clipper/notes")
.header("Authorization", trilium_token)
.json(&hashmap!{ "title" => title, "content" => content, "clipType" => "note" })
.send().await?;
Ok(())
}
// image note:
2020-05-04 15:14:20 +00:00
// curl 'http://localhost:9001/api/clipper/clippings' -H 'Accept: */*' -H 'Accept-Language: en' --compressed -H 'Content-Type: application/json' -H 'Authorization: icB3xohFDpkVt7YFpbTflUYC8pucmryVGpb1DFpd6ns=' -H 'Origin: moz-extension://13bc3fd7-5cb0-4d48-b368-76e389fd7c5f' --data $'{"title":"trilium/clipper.js at master \xb7 zadam/trilium","content":"<img src=\\"BoCpsLz9je8a01MdGbj4\\">","images":[{"imageId":"BoCpsLz9je8a01MdGbj4","src":"inline.png","dataUrl":""}]}'
2020-05-23 15:23:36 +00:00
async fn event_alerts() {
loop {
let last_min = Local::now().minute();
if let Err(e) = event_alerts_soon().await {
println!("error: {}", e);
}
while Local::now().minute() == last_min {
tokio::time::delay_for(Duration::from_secs(1)).await;
}
tokio::time::delay_for(Duration::from_secs(16)).await;
}
}
async fn event_alerts_soon() -> Result<(), Error> {
let now = Local::now();
let events: Vec<Event> = CLIENT.get("http://localhost:9001/custom/event_alerts").send().await?.json().await?;
for event in events {
let todo_time: DateTime<Local> = TimeZone::from_local_datetime(&Local, &NaiveDateTime::parse_from_str(&event.start_time, "%Y-%m-%dT%H:%M:%S")?).unwrap();
if todo_time <= now {
continue;
}
let diff = todo_time - now;
let minutes = diff.num_minutes();
if minutes == 7 * 24 * 60 || minutes == 48 * 60 || minutes == 24 * 60 || minutes == 60 || minutes == 10 {
event_alert_notify(&format_time(diff), event).await?;
}
}
Ok(())
}
async fn event_alert_notify(time_left: &str, event: Event) -> Result<(), Error> {
API.send(SendMessage::new(*OWNER, format!("{}: {}", time_left, event.name))).await?;
Ok(())
}
#[derive(Deserialize)]
#[serde(rename_all = "camelCase")]
struct Event {
name: String,
start_time: String,
}
2020-05-04 15:14:20 +00:00
async fn start_polling() {
loop {
let last_min = Local::now().minute();
if let Err(e) = one_req().await {
println!("error: {}", e);
}
while Local::now().minute() == last_min {
tokio::time::delay_for(Duration::from_secs(1)).await;
}
tokio::time::delay_for(Duration::from_secs(1)).await;
}
}
async fn one_req() -> Result<(), Error> {
let now = Local::now();
//println!("{}", CLIENT.get("http://localhost:9001/custom/task_alerts").send().await?.text().await?);
let tasks: Vec<Task> = CLIENT.get("http://localhost:9001/custom/task_alerts").send().await?.json().await?;
//println!("{:?}", tasks);
'task: for task in tasks {
let mut todo_date = None;
let mut todo_time = None;
let mut is_reminder = false;
//println!("considering {:?} with {:?}", task.title, task.attributes);
for attribute in &task.attributes {
if attribute.r#type != "label" {
continue;
}
match &*attribute.name {
"todoDate" => todo_date = Some(attribute.value.as_str().unwrap().to_owned()),
"todoTime" => todo_time = Some(attribute.value.as_str().unwrap().to_owned()),
"doneDate" => continue 'task,
"reminder" => is_reminder = true,
_ => {}
}
}
if todo_date.is_none() {
continue;
}
let todo_date = todo_date.unwrap();
let parts = todo_date.split('-').collect::<Vec<_>>();
let (year, month, day) = (parts[0].parse().unwrap(), parts[1].parse().unwrap(), parts[2].parse().unwrap());
let (hour, minute, second) = if let Some(todo_time) = todo_time {
let parts = todo_time.split(':').collect::<Vec<_>>();
(parts.get(0).map(|x| x.parse().unwrap()).unwrap_or(0), parts.get(1).map(|x| x.parse().unwrap()).unwrap_or(0), parts.get(2).map(|x| x.parse().unwrap()).unwrap_or(0))
} else { (0, 0, 0) };
let todo_time: DateTime<Local> = TimeZone::from_local_datetime(&Local, &NaiveDate::from_ymd(year, month, day).and_hms(hour, minute, second)).unwrap();
if todo_time <= now {
continue;
}
let diff = todo_time - now;
let minutes = diff.num_minutes();
if !is_reminder && (minutes == 7 * 24 * 60 || minutes == 48 * 60 || minutes == 24 * 60 || minutes == 60 || minutes == 10) {
notify_owner_impl(&format_time(diff), task).await?;
} else if is_reminder && minutes == 0 {
notify_owner_impl("", task).await?;
}
}
Ok(())
}
fn format_time(diff: chrono::Duration) -> String {
if diff.num_weeks() > 0 {
format!("{}w", diff.num_weeks())
} else if diff.num_days() > 0 {
format!("{}d", diff.num_days())
} else if diff.num_hours() > 0 {
if diff.num_minutes() % 60 != 0 {
format!("{}h{:02}m", diff.num_hours(), diff.num_minutes() % 60)
} else {
format!("{}h", diff.num_hours())
}
} else {
format!("{}m", diff.num_minutes())
}
}
2020-05-04 15:36:19 +00:00
async fn notify_owner_impl(time_left: &str, task: Task) -> Result<(), Error> {
2020-05-04 15:14:20 +00:00
API.send(SendMessage::new(*OWNER, format!("{}: {}", time_left, task.title))).await?;
2020-05-04 15:36:19 +00:00
Ok(())
2020-05-04 15:14:20 +00:00
}
async fn send_message<S: Into<String>>(msg: S) -> Result<(), Error> {
API.send(SendMessage::new(*OWNER, msg.into())).await?;
Ok(())
}
2020-05-04 15:14:20 +00:00
#[derive(Deserialize, Debug)]
#[allow(non_snake_case)]
struct Task {
attributes: Vec<Attribute>,
contentLength: usize,
dateCreated: DateTime<FixedOffset>,
dateModified: DateTime<FixedOffset>,
deleteId: Option<serde_json::Value>,
hash: String,
isContentAvailable: bool,
isDeleted: bool,
isErased: i64,
isProtected: bool,
mime: String,
noteId: String,
title: String,
r#type: String,
utcDateCreated: DateTime<Utc>,
utcDateModified: DateTime<Utc>,
}
#[derive(Deserialize, Debug)]
#[allow(non_snake_case)]
struct Attribute {
attributeId: String,
noteId: String,
r#type: String,
name: String,
value: serde_json::Value,
position: usize,
utcDateCreated: DateTime<Utc>,
utcDateModified: DateTime<Utc>,
isDeleted: bool,
deleteId: Option<serde_json::Value>,
hash: String,
isInheritable: bool,
2020-05-13 20:50:49 +00:00
//isOwned: bool, // removed in 0.42.2
2020-05-04 15:14:20 +00:00
}
#[derive(Error, Debug)]
pub enum Error {
#[error("network error: {0}")]
Network(#[from] reqwest::Error),
#[error("telegram error: {0}")]
Telegram(#[from] telegram_bot::Error),
#[error("mime parsing error: {0}")]
Mime(#[from] mime::FromStrError),
2020-05-23 15:23:36 +00:00
#[error("chrono parsing error: {0}")]
Chrono(#[from] chrono::format::ParseError),
#[error("ical parsing error: {0}")]
Ical(#[from] ical_parsing::Error),
#[error("internal error: {0}")]
CustomMessage(String),
}
fn error<S: Into<String>>(msg: S) -> Error {
Error::CustomMessage(msg.into())
2020-05-04 15:14:20 +00:00
}