tesla-charge-controller/src/main.rs

174 lines
6.8 KiB
Rust
Raw Normal View History

2024-01-08 12:00:09 +11:00
#![feature(never_type)]
2024-01-06 09:33:56 +11:00
2023-12-28 12:41:05 +11:00
#[macro_use]
extern crate rocket;
2024-01-08 12:00:09 +11:00
use api_interface::TeslaInterface;
use charge_controllers::pl::Pli;
2023-12-25 21:22:08 +11:00
use clap::{Parser, Subcommand};
use config::{access_config, CONFIG_PATH};
2024-01-12 09:09:42 +11:00
use errors::PrintErrors;
2024-01-08 12:00:09 +11:00
use std::path::PathBuf;
2024-01-16 11:00:11 +11:00
use tesla_charge_rate::TeslaChargeRateController;
2023-12-28 12:41:05 +11:00
2024-01-08 12:00:09 +11:00
use crate::config::Config;
2023-12-25 21:22:08 +11:00
2024-01-08 12:00:09 +11:00
mod api_interface;
mod charge_controllers;
2023-12-25 21:22:08 +11:00
mod config;
mod errors;
2023-12-28 12:41:05 +11:00
mod server;
2024-01-16 11:00:11 +11:00
mod tesla_charge_rate;
2024-01-07 10:08:16 +11:00
mod types;
2023-12-25 21:22:08 +11:00
#[derive(Parser, Debug, Clone)]
#[clap(author, version, about, long_about = None)]
struct Args {
#[command(subcommand)]
command: Commands,
#[clap(long, default_value = "/etc/tesla-charge-controller")]
config_dir: PathBuf,
}
#[derive(Subcommand, Debug, Clone)]
enum Commands {
/// Run charge controller server
2023-12-27 18:29:12 +11:00
Watch,
/// Print the default config file
GenerateConfig,
2023-12-27 16:42:29 +11:00
}
2023-12-25 21:22:08 +11:00
#[tokio::main]
async fn main() {
let args = Args::parse();
2024-01-15 10:22:10 +11:00
env_logger::builder()
.format_module_path(false)
.format_timestamp(
if std::env::var("LOG_TIMESTAMP").is_ok_and(|v| v == "false") {
None
} else {
Some(env_logger::TimestampPrecision::Seconds)
},
)
.init();
2023-12-28 12:41:05 +11:00
2023-12-25 21:22:08 +11:00
let auth_path = args.config_dir.join("auth");
2024-01-17 08:35:05 +11:00
let _ = CONFIG_PATH.set(args.config_dir.join("config.json"));
2023-12-27 16:42:29 +11:00
2024-01-10 15:22:28 +11:00
let _recorder = metrics_prometheus::install();
2023-12-27 18:29:12 +11:00
match args.command {
Commands::GenerateConfig => {
println!(
"{}",
2024-01-17 08:35:05 +11:00
serde_json::ser::to_string_pretty(&Config::default()).unwrap(),
2023-12-27 18:29:12 +11:00
);
}
2024-01-12 09:09:42 +11:00
Commands::Watch => {
2024-01-15 10:13:46 +11:00
if let Some(mut interface) = TeslaInterface::load(auth_path)
.await
.some_or_print_with("loading tesla interface")
{
let config = access_config();
2024-01-09 11:50:37 +11:00
// build the channel that takes messages from the webserver thread to the api thread
2024-01-11 08:57:41 +11:00
let (api_requests, api_receiver) = async_channel::unbounded();
// and to the pli thread
let (pli_requests, pli_receiver) = async_channel::unbounded();
2024-01-16 11:00:11 +11:00
// and to the charge rate controller thread
let (tcrc_requests, tcrc_receiver) = async_channel::unbounded();
charge_controllers::register_metrics();
// try to spawn the pli loop
let pl_state = match Pli::new(
config.serial_port.clone(),
config.baud_rate,
config.pl_timeout_milliseconds,
) {
Ok(mut pli) => {
let pl_state = pli.state.clone();
tokio::task::spawn(async move {
let mut interval = tokio::time::interval(
std::time::Duration::from_secs(config.pl_watch_interval_seconds),
);
loop {
tokio::select! {
_ = interval.tick() => pli.refresh(),
message = pli_receiver.recv() => match message {
Ok(message) => pli.process_request(message),
Err(e) => panic!("Error on PLI receive channel: {e:?}")
}
}
}
});
Some(pl_state)
}
Err(e) => {
log::error!("Error connecting to serial device for PLI: {e:?}");
None
}
};
2024-01-16 11:00:11 +11:00
let mut tesla_charge_rate_controller =
TeslaChargeRateController::new(interface.state.clone(), pl_state.clone());
2024-01-08 12:00:09 +11:00
let server_handle = server::launch_server(server::ServerState {
car_state: interface.state.clone(),
pl_state,
2024-01-16 11:00:11 +11:00
tcrc_state: tesla_charge_rate_controller.tcrc_state.clone(),
api_requests,
2024-01-11 08:57:41 +11:00
pli_requests,
2024-01-16 11:00:11 +11:00
tcrc_requests,
2024-01-08 12:00:09 +11:00
});
2024-01-06 09:33:56 +11:00
2024-01-09 11:50:37 +11:00
// spawn the api loop
2024-01-08 12:00:09 +11:00
tokio::task::spawn(async move {
let mut normal_data_update_interval = tokio::time::interval(
std::time::Duration::from_secs(config.tesla_update_interval_seconds),
);
let mut charge_data_update_interval =
tokio::time::interval(std::time::Duration::from_secs(
config.tesla_update_interval_while_charging_seconds,
));
let mut charge_rate_update_interval = tokio::time::interval(
std::time::Duration::from_secs(config.charge_rate_update_interval_seconds),
);
2024-01-08 12:00:09 +11:00
loop {
2024-01-09 11:50:37 +11:00
// await either the next interval OR a message from the other thread
tokio::select! {
_ = normal_data_update_interval.tick() => {
if !interface.state.read().unwrap().is_charging_at_home() {
interface.refresh().await
2024-01-16 11:00:11 +11:00
}
},
_ = charge_data_update_interval.tick() => {
if interface.state.read().unwrap().is_charging_at_home() {
interface.refresh().await
}
},
_ = charge_rate_update_interval.tick() => {
if interface.state.read().unwrap().is_charging_at_home() {
if let Some(request) = tesla_charge_rate_controller.control_charge_rate() {
interface.process_request(request).await;
}
}
},
2024-01-16 11:00:11 +11:00
api_message = api_receiver.recv() => match api_message {
Ok(message) => interface.process_request(message).await,
Err(e) => panic!("Error on Tesla receive channel: {e:?}")
2024-01-16 11:00:11 +11:00
},
tcrc_message = tcrc_receiver.recv() => match tcrc_message {
Ok(message) => tesla_charge_rate_controller.process_request(message),
Err(e) => panic!("Error on TCRC receive channel: {e:?}")
}
}
2024-01-06 09:33:56 +11:00
}
2024-01-08 12:00:09 +11:00
});
server_handle.await;
2024-01-06 09:33:56 +11:00
}
2024-01-12 09:09:42 +11:00
}
2024-01-06 09:33:56 +11:00
}
2023-12-27 12:20:45 +11:00
}