Compare commits

..

No commits in common. "36a1db441e812c5fe3f9ce97cc6795b5a23705d9" and "02b7127d302fdc10f16174451bda2dacd193955a" have entirely different histories.

5 changed files with 66 additions and 109 deletions

34
Cargo.lock generated
View file

@ -332,23 +332,6 @@ dependencies = [
"phf_codegen", "phf_codegen",
] ]
[[package]]
name = "chrono_track"
version = "0.1.0"
dependencies = [
"chrono",
"chrono-tz",
"clap",
"csv",
"lettre",
"log",
"reqwest",
"serde",
"serde_json",
"simplelog",
"toml",
]
[[package]] [[package]]
name = "clap" name = "clap"
version = "4.4.2" version = "4.4.2"
@ -1572,6 +1555,23 @@ dependencies = [
"time-core", "time-core",
] ]
[[package]]
name = "time_tracker"
version = "0.1.0"
dependencies = [
"chrono",
"chrono-tz",
"clap",
"csv",
"lettre",
"log",
"reqwest",
"serde",
"serde_json",
"simplelog",
"toml",
]
[[package]] [[package]]
name = "tinyvec" name = "tinyvec"
version = "1.6.0" version = "1.6.0"

View file

@ -1,5 +1,5 @@
[workspace] [workspace]
members = [ members = [
"status_cloud", "status_cloud",
"chrono_track" "time_tracker"
] ]

View file

@ -62,26 +62,25 @@ rust-project TODO: write shell script for automatically updating `cargoHash`
pname = "status_cloud"; pname = "status_cloud";
version = "0.1.0"; version = "0.1.0";
buildAndTestSubdir = "status_cloud"; buildAndTestSubdir = "status_cloud";
cargoHash = "sha256-mXcD/92WJLgN2efX/8EiV3rOqUiNOKGC4puS8DvGFOc="; cargoHash = "sha256-XIonh2SQ8EichcZvHErSydt0qtKGodXE0yKz4BsntA8=";
postFixup = '' preBuild = ''
wrapProgram $out/bin/status_cloud \ sed -i 's/Command::new("hddtemp")/Command::new("${nixpkgs.lib.escape [ "/" ] "${pkgs.hddtemp}"}")/g' status_cloud/src/main.rs
--prefix PATH : "${nixpkgs.lib.makeBinPath [ pkgs.hddtemp ]}"
''; '';
meta = meta // { meta = meta // {
description = "Server status saved to a nextcloud note service."; description = "Server status saved to a nextcloud note service.";
}; };
}); });
chrono_track = pkgs.rustPlatform.buildRustPackage (rustSettings // { time_tracker = pkgs.rustPlatform.buildRustPackage (rustSettings // {
pname = "chrono_track"; pname = "time_tracker";
version = "0.1.0"; version = "0.1.0";
buildAndTestSubdir = "chrono_track"; buildAndTestSubdir = "time_tracker";
cargoHash = "sha256-51j5eG2ZBJg2TjbvYSEvvmOM4hhPyNohN+LUmvck88k="; cargoHash = "sha256-5Wy2+ef2BriKmvNhllDVh/clAZGV7myDc281ygj05vI=";
meta = meta // { meta = meta // {
description = "Using nextcloud notes to track time usage."; description = "Using nextcloud notes to track time usage.";
}; };
postFixup = '' postFixup = ''
wrapProgram $out/bin/chrono_track \ wrapProgram $out/bin/time_tracker \
--prefix PATH : "${nixpkgs.lib.makeBinPath [ pkgs.msmtp ]}" --prefix PATH : "${nixpkgs.lib.makeBinPath [ pkgs.msmtp ]}"
''; '';
}); });
@ -139,14 +138,14 @@ rust-project TODO: write shell script for automatically updating `cargoHash`
# Time Tracker # Time Tracker
options.services.chrono_track = { options.services.time_tracker = {
enable = lib.mkEnableOption (lib.mdDoc "chrono track, time tracking service"); enable = lib.mkEnableOption (lib.mdDoc "time tracker service");
package = lib.mkOption { package = lib.mkOption {
type = lib.types.package; type = lib.types.package;
default = self.packages.${system}.chrono_track; default = self.packages.${system}.time_tracker;
defaultText = "pkgs.chrono_track"; defaultText = "pkgs.time_tracker";
description = lib.mdDoc '' description = lib.mdDoc ''
The chrono_track package that should be used The time_tracker package that should be used
''; '';
}; };
config_path = lib.mkOption { config_path = lib.mkOption {
@ -155,12 +154,6 @@ rust-project TODO: write shell script for automatically updating `cargoHash`
The file path to the toml that contains user information secrets The file path to the toml that contains user information secrets
''; '';
}; };
from_address = lib.mkOption {
type = lib.types.str;
description = lib.mdDoc ''
The from address for the emails. E.g. noreply@example.com
'';
};
frequency = lib.mkOption { frequency = lib.mkOption {
type = lib.types.int; type = lib.types.int;
description = lib.mdDoc '' description = lib.mdDoc ''
@ -169,25 +162,25 @@ rust-project TODO: write shell script for automatically updating `cargoHash`
}; };
}; };
config.systemd.services.chrono_track = let config.systemd.services.time_tracker = let
cfg = config.services.chrono_track; cfg = config.services.time_tracker;
pkg = self.packages.${system}.chrono_track; pkg = self.packages.${system}.time_tracker;
in lib.mkIf cfg.enable { in lib.mkIf cfg.enable {
description = "Chrono Track"; description = "Nextcloud Time Tracker";
serviceConfig = { serviceConfig = {
Type = "oneshot"; Type = "oneshot";
ExecStart = '' ExecStart = ''
${cfg.package}/bin/chrono_track --config-file ${builtins.toString cfg.config_path} --from_addr ${cfg.from_address} ${cfg.package}/bin/time_tracker --config-file ${builtins.toString cfg.config_path}
''; '';
}; };
}; };
config.systemd.timers.chrono_track = let config.systemd.timers.time_tracker = let
cfg = config.services.chrono_track; cfg = config.services.time_tracker;
pkg = self.packages.${system}.chrono_track; pkg = self.packages.${system}.time_tracker;
in lib.mkIf cfg.enable { in lib.mkIf cfg.enable {
wantedBy = [ "timers.target" ]; wantedBy = [ "timers.target" ];
partOf = [ "chrono_track.service" ]; partOf = [ "time_tracker.service" ];
timerConfig.OnCalendar = [ "*:0/${builtins.toString cfg.frequency}" ]; timerConfig.OnCalendar = [ "*:0/${builtins.toString cfg.frequency}" ];
}; };

View file

@ -1,5 +1,5 @@
[package] [package]
name = "chrono_track" name = "time_tracker"
version = "0.1.0" version = "0.1.0"
edition = "2021" edition = "2021"

View file

@ -2,7 +2,7 @@ use chrono::{DateTime, Datelike, Utc};
use chrono_tz::US::Central; use chrono_tz::US::Central;
use clap::Parser; use clap::Parser;
use lettre::{ use lettre::{
message::{header::ContentType, Attachment, Message, MultiPart, SinglePart}, message::{Message, header::ContentType, Attachment, MultiPart, SinglePart},
SendmailTransport, Transport, SendmailTransport, Transport,
}; };
use log::{debug, error}; use log::{debug, error};
@ -196,7 +196,7 @@ fn main() -> Result<(), Box<dyn std::error::Error>> {
if do_email_summary { if do_email_summary {
debug!("Send email selected"); debug!("Send email selected");
reset_email_checkbox(&cfg); reset_email_checkbox(&cfg);
send_email_summary(&cfg, body_content, &args); send_email_summary(&cfg, body_content);
} }
Ok(()) Ok(())
@ -236,9 +236,6 @@ struct CliArgs {
#[arg(short, long)] #[arg(short, long)]
config_file: PathBuf, config_file: PathBuf,
#[arg(short, long)]
from_addr: String,
#[arg(short, long)] #[arg(short, long)]
debug: bool, debug: bool,
} }
@ -302,14 +299,14 @@ struct SummaryRow {
// All calls after depend on the success of the one before, and cannot run // All calls after depend on the success of the one before, and cannot run
// without. So it would be too much work to make an error type to handle an error // without. So it would be too much work to make an error type to handle an error
// that would cause a crash anyways // that would cause a crash anyways
fn send_email_summary(config: &Config, body_content: Vec<String>, cliargs: &CliArgs) { fn send_email_summary(config: &Config, body_content: Vec<String>) {
let mut wtr = csv::Writer::from_writer(vec![]); let mut wtr = csv::Writer::from_writer(vec![]);
for line in body_content { for line in body_content {
if line.starts_with("### ") { if line.starts_with("### ") {
let mut split: Vec<&str> = line.split('|').collect(); let mut split: Vec<&str> = line.split('|').collect();
if split.len() != 3 { if split.len() != 3 {
error!("There was an issue with this line. 3 splits not found: {line}"); error!("There was an issue with this line: {line}");
continue; continue;
} }
let date: String = match split.pop() { let date: String = match split.pop() {
@ -358,70 +355,36 @@ fn send_email_summary(config: &Config, body_content: Vec<String>, cliargs: &CliA
.to_string() .to_string()
.replace("### ", ""); .replace("### ", "");
match wtr.serialize(SummaryRow { wtr.serialize(SummaryRow { date: date.trim().to_string(), total_time: time, task_name }).unwrap();
date: date.trim().to_string(),
total_time: time,
task_name,
}) {
Ok(_) => continue,
Err(e) => {
error!("There was an error serializing the csv, aborting: {e}");
return;
}
};
} }
} }
let data = match String::from_utf8(wtr.into_inner().unwrap()) { let data = String::from_utf8(wtr.into_inner().unwrap()).unwrap();
Ok(val) => val,
Err(e) => {
error!("There was an error converting the csv writer to a string, aborting: {e}");
return;
}
};
debug!("{:#?}", data); debug!("{:#?}", data);
let attachment = Attachment::new("TimeSummary.csv".to_string()) let attachment = Attachment::new("TimeSummary.csv".to_string()).body(data, ContentType::parse("text/csv").unwrap());
// The unwrap is on a constant value
.body(data, ContentType::parse("text/csv").unwrap());
const HTML: &str = r#" const HTML: &str = r#"
<!DOCTYPE html> !DOCTYPE html>
<html lang="en"> <html lang="en">
<head> <head>
<meta charset="UTF-8"> <meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>ChronoTrack Export</title> <title>Hello from Lettre!</title>
</head> </head>
<body> <body>
<div style="background-color: #33ccff; border-radius: 40px; height: 100px"></div>
<div style="display: flex; flex-direction: column; align-items: center;"> <div style="display: flex; flex-direction: column; align-items: center;">
<img src="https://cdn-icons-png.flaticon.com/512/3938/3938540.png" <h2 style="font-family: Arial, Helvetica, sans-serif;">Hello from Lettre!</h2>
style="height:100px; border-radius: 50%; background-color: whitesmoke; position: absolute; top: 40px;"> <h4 style="font-family: Arial, Helvetica, sans-serif;">A mailer library for Rust</h4>
<h2 style="font-family: Arial, Helvetica, sans-serif; position: absolute; top: 160px;">Prepared with care, and sent through the horrors of the interwebs, ChronoTrack presents! (see attached)
</div> </div>
</body> </body>
</html>"#; </html>"#;
let email = Message::builder() let email = Message::builder()
.from(match cliargs.from_addr.parse() { .from("NoReply@nickiel.net <noreply@nickiel.net>".parse().unwrap())
Ok(val) => val, .to(config.email_addr.parse().unwrap())
Err(e) => { .subject("Testing email")
error!("The provided from_email address was unparsable:\n{e}");
std::process::exit(1);
}
})
.to(match config.email_addr.parse() {
Ok(val) => val,
Err(e) => {
error!(
"The provided destination email in the configuration file was unparsable:\n{e}"
);
std::process::exit(1);
}
})
.subject("ChronoTrack Export")
.multipart( .multipart(
MultiPart::mixed() MultiPart::mixed()
.multipart( .multipart(
@ -429,22 +392,23 @@ fn send_email_summary(config: &Config, body_content: Vec<String>, cliargs: &CliA
.singlepart( .singlepart(
SinglePart::builder() SinglePart::builder()
.header(ContentType::TEXT_PLAIN) .header(ContentType::TEXT_PLAIN)
.body(String::from("This is an automated email from ChronoTrack")), .body(String::from("Hello world"))
) )
.singlepart( .singlepart(
SinglePart::builder() SinglePart::builder()
.header(ContentType::TEXT_HTML) .header(ContentType::TEXT_HTML)
.body(String::from(HTML)), .body(String::from(HTML))
),
) )
.singlepart(attachment),
) )
.unwrap(); .singlepart(attachment)
).unwrap();
let mailer = SendmailTransport::new(); let mailer = SendmailTransport::new();
match mailer.send(&email) { match mailer.send(&email) {
Ok(val) => debug!("email sent: {:?}", val), Ok(val) => debug!("email sent: {:?}", val),
Err(e) => error!("Couldn't send email {}", e), Err(e) => error!("Couldn't send email {}", e)
}; };
} }