Compare commits
	
		
			20 Commits
		
	
	
		
			1.0.0
			...
			c35722fe17
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| c35722fe17 | |||
| 1735077db3 | |||
| 31bb956a29 | |||
| ff0e548422 | |||
| 837835da7e | |||
| 2d262bb4c9 | |||
| f594ebfbaa | |||
| 57a9c03308 | |||
| 7b9db9c7c3 | |||
| b7720df305 | |||
| 445c1b014e | |||
| aa732af571 | |||
| c365f959e7 | |||
| 9a4c6d2de2 | |||
| 3c20cca915 | |||
| 5aaad54de3 | |||
| b3edfb05d9 | |||
| 45029f24cc | |||
| ec594c0e4d | |||
| fdfbdf093f | 
							
								
								
									
										57
									
								
								central_backend/Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										57
									
								
								central_backend/Cargo.lock
									
									
									
										generated
									
									
									
								
							| @@ -492,9 +492,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "anyhow" | name = "anyhow" | ||||||
| version = "1.0.89" | version = "1.0.90" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "86fdf8605db99b54d3cd748a44c6d04df638eb5dafb219b135d0149bd0db01f6" | checksum = "37bf3594c4c988a53154954629820791dde498571819ae4ca50ca811e060cc95" | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "asn1" | name = "asn1" | ||||||
| @@ -645,9 +645,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "cc" | name = "cc" | ||||||
| version = "1.1.30" | version = "1.1.31" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "b16803a61b81d9eabb7eae2588776c4c1e584b738ede45fdbb4c972cec1e9945" | checksum = "c2e7962b54006dcfcc61cb72735f4d89bb97061dd6a7ed882ec6b8ee53714c6f" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "jobserver", |  "jobserver", | ||||||
|  "libc", |  "libc", | ||||||
| @@ -656,7 +656,7 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "central_backend" | name = "central_backend" | ||||||
| version = "0.1.0" | version = "1.0.2" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "actix", |  "actix", | ||||||
|  "actix-cors", |  "actix-cors", | ||||||
| @@ -1387,9 +1387,9 @@ checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4" | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "hyper" | name = "hyper" | ||||||
| version = "1.4.1" | version = "1.5.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "50dfd22e0e76d0f662d429a5f80fcaf3855009297eab6a0a9f8543834744ba05" | checksum = "bbbff0a806a4728c99295b254c8838933b5b082d75e3cb70c8dab21fdfbcfa9a" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "bytes", |  "bytes", | ||||||
|  "futures-channel", |  "futures-channel", | ||||||
| @@ -1498,9 +1498,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "impl-more" | name = "impl-more" | ||||||
| version = "0.1.6" | version = "0.1.8" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "206ca75c9c03ba3d4ace2460e57b189f39f43de612c2f85836e65c929701bb2d" | checksum = "aae21c3177a27788957044151cc2800043d127acaa460a47ebb9b84dfa2c6aa0" | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "indexmap" | name = "indexmap" | ||||||
| @@ -1955,9 +1955,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "proc-macro2" | name = "proc-macro2" | ||||||
| version = "1.0.87" | version = "1.0.88" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "b3e4daa0dcf6feba26f985457cdf104d4b4256fc5a09547140f3631bb076b19a" | checksum = "7c3a7fc5db1e57d5a779a352c8cdb57b29aa4c40cc69c3a68a7fedc815fbf2f9" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "unicode-ident", |  "unicode-ident", | ||||||
| ] | ] | ||||||
| @@ -2178,9 +2178,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "rustls" | name = "rustls" | ||||||
| version = "0.23.14" | version = "0.23.15" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "415d9944693cb90382053259f89fbb077ea730ad7273047ec63b19bc9b160ba8" | checksum = "5fbb44d7acc4e873d613422379f69f237a1b141928c02f6bc6ccfddddc2d7993" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "once_cell", |  "once_cell", | ||||||
|  "rustls-pki-types", |  "rustls-pki-types", | ||||||
| @@ -2200,9 +2200,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "rustls-pki-types" | name = "rustls-pki-types" | ||||||
| version = "1.9.0" | version = "1.10.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "0e696e35370c65c9c541198af4543ccd580cf17fc25d8e05c5a242b202488c55" | checksum = "16f1201b3c9a7ee8039bcadc17b7e605e2945b27eee7631788c1bd2b0643674b" | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "rustls-webpki" | name = "rustls-webpki" | ||||||
| @@ -2285,18 +2285,18 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "serde" | name = "serde" | ||||||
| version = "1.0.210" | version = "1.0.213" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "c8e3592472072e6e22e0a54d5904d9febf8508f65fb8552499a1abc7d1078c3a" | checksum = "3ea7893ff5e2466df8d720bb615088341b295f849602c6956047f8f80f0e9bc1" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "serde_derive", |  "serde_derive", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "serde_derive" | name = "serde_derive" | ||||||
| version = "1.0.210" | version = "1.0.213" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "243902eda00fad750862fc144cea25caca5e20d615af0a81bee94ca738f1df1f" | checksum = "7e85ad2009c50b58e87caa8cd6dac16bdf511bbfb7af6c33df902396aa480fa5" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
| @@ -2305,9 +2305,9 @@ dependencies = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "serde_json" | name = "serde_json" | ||||||
| version = "1.0.129" | version = "1.0.132" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "6dbcf9b78a125ee667ae19388837dd12294b858d101fdd393cb9d5501ef09eb2" | checksum = "d726bfaff4b320266d395898905d0eba0345aae23b54aee3a737e260fd46db03" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "itoa", |  "itoa", | ||||||
|  "memchr", |  "memchr", | ||||||
| @@ -2445,9 +2445,9 @@ checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "syn" | name = "syn" | ||||||
| version = "2.0.79" | version = "2.0.82" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "89132cd0bf050864e1d38dc3bbc07a0eb8e7530af26344d3d2bbbef83499f590" | checksum = "83540f837a8afc019423a8edb95b52a8effe46957ee402287f4292fae35be021" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "proc-macro2", |  "proc-macro2", | ||||||
|  "quote", |  "quote", | ||||||
| @@ -2698,12 +2698,9 @@ checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "unicase" | name = "unicase" | ||||||
| version = "2.7.0" | version = "2.8.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "f7d2d4dafb69621809a81864c9c1b864479e1235c0dd4e199924b9742439ed89" | checksum = "7e51b68083f157f853b6379db119d1c1be0e6e4dec98101079dec41f6f5cf6df" | ||||||
| dependencies = [ |  | ||||||
|  "version_check", |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "unicode-bidi" | name = "unicode-bidi" | ||||||
| @@ -2773,9 +2770,9 @@ checksum = "06abde3611657adf66d383f00b093d7faecc7fa57071cce2578660c9f1010821" | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "uuid" | name = "uuid" | ||||||
| version = "1.10.0" | version = "1.11.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "81dfa00651efa65069b0b6b651f4aaa31ba9e3c3ce0137aaad053604ee7e0314" | checksum = "f8c5f0a0af699448548ad1a2fbf920fb4bee257eae39953ba95cb84891a0446a" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "getrandom", |  "getrandom", | ||||||
|  "serde", |  "serde", | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| [package] | [package] | ||||||
| name = "central_backend" | name = "central_backend" | ||||||
| version = "0.1.0" | version = "1.0.2" | ||||||
| edition = "2021" | edition = "2021" | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| @@ -20,7 +20,7 @@ actix-web = { version = "4", features = ["openssl"] } | |||||||
| futures = "0.3.31" | futures = "0.3.31" | ||||||
| serde = { version = "1.0.210", features = ["derive"] } | serde = { version = "1.0.210", features = ["derive"] } | ||||||
| reqwest = { version = "0.12.7", features = ["json"] } | reqwest = { version = "0.12.7", features = ["json"] } | ||||||
| serde_json = "1.0.128" | serde_json = "1.0.131" | ||||||
| rand = "0.8.5" | rand = "0.8.5" | ||||||
| actix = "0.13.5" | actix = "0.13.5" | ||||||
| actix-identity = "0.8.0" | actix-identity = "0.8.0" | ||||||
| @@ -29,7 +29,7 @@ actix-cors = "0.7.0" | |||||||
| actix-multipart = { version ="0.7.2", features = ["derive"] } | actix-multipart = { version ="0.7.2", features = ["derive"] } | ||||||
| actix-remote-ip = "0.1.0" | actix-remote-ip = "0.1.0" | ||||||
| futures-util = "0.3.31" | futures-util = "0.3.31" | ||||||
| uuid = { version = "1.10.0", features = ["v4", "serde"] } | uuid = { version = "1.11.0", features = ["v4", "serde"] } | ||||||
| semver = { version = "1.0.23", features = ["serde"] } | semver = { version = "1.0.23", features = ["serde"] } | ||||||
| lazy-regex = "3.3.0" | lazy-regex = "3.3.0" | ||||||
| tokio = { version = "1.40.0", features = ["full"] } | tokio = { version = "1.40.0", features = ["full"] } | ||||||
|   | |||||||
| @@ -39,8 +39,12 @@ pub enum ConsumptionBackend { | |||||||
|     /// Fronius inverter consumption |     /// Fronius inverter consumption | ||||||
|     Fronius { |     Fronius { | ||||||
|         /// The origin of the domain where the webserver of the Fronius Symo can be reached |         /// The origin of the domain where the webserver of the Fronius Symo can be reached | ||||||
|         #[clap(short, long, env = "FRONIUS_ORIG")] |         #[clap(short, long, env)] | ||||||
|         origin: String, |         fronius_orig: String, | ||||||
|  |  | ||||||
|  |         /// Use cURL instead of reqwest to perform request | ||||||
|  |         #[clap(short, long)] | ||||||
|  |         curl: bool, | ||||||
|     }, |     }, | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,6 +9,8 @@ pub enum ConsumptionError { | |||||||
|     NonExistentFile, |     NonExistentFile, | ||||||
|     #[error("The file that should contain the consumption has an invalid content!")] |     #[error("The file that should contain the consumption has an invalid content!")] | ||||||
|     FileInvalidContent(#[source] ParseIntError), |     FileInvalidContent(#[source] ParseIntError), | ||||||
|  |     #[error("Failed to execute cURL request!")] | ||||||
|  |     CurlReqFailed, | ||||||
| } | } | ||||||
|  |  | ||||||
| pub type EnergyConsumption = i32; | pub type EnergyConsumption = i32; | ||||||
| @@ -63,9 +65,25 @@ pub async fn get_curr_consumption() -> anyhow::Result<EnergyConsumption> { | |||||||
|                 .map_err(ConsumptionError::FileInvalidContent)?) |                 .map_err(ConsumptionError::FileInvalidContent)?) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         ConsumptionBackend::Fronius { origin } => { |         ConsumptionBackend::Fronius { fronius_orig, curl } => { | ||||||
|             let url = format!("{origin}/solar_api/v1/GetPowerFlowRealtimeData.fcgi"); |             let url = format!("{fronius_orig}/solar_api/v1/GetPowerFlowRealtimeData.fcgi"); | ||||||
|             let response = reqwest::get(url).await?.json::<FroniusResponse>().await?; |  | ||||||
|  |             let response = match curl { | ||||||
|  |                 false => reqwest::get(url).await?.json::<FroniusResponse>().await?, | ||||||
|  |                 true => { | ||||||
|  |                     let res = std::process::Command::new("curl") | ||||||
|  |                         .arg("--connect-timeout") | ||||||
|  |                         .arg("1.5") | ||||||
|  |                         .arg(url) | ||||||
|  |                         .output()?; | ||||||
|  |  | ||||||
|  |                     if !res.status.success() { | ||||||
|  |                         return Err(ConsumptionError::CurlReqFailed.into()); | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     serde_json::from_slice::<FroniusResponse>(&res.stdout)? | ||||||
|  |                 } | ||||||
|  |             }; | ||||||
|  |  | ||||||
|             Ok(response.body.data.site.grid_production as i32) |             Ok(response.body.data.site.grid_production as i32) | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -25,7 +25,14 @@ impl EnergyActor { | |||||||
|     pub async fn new() -> anyhow::Result<Self> { |     pub async fn new() -> anyhow::Result<Self> { | ||||||
|         let consumption_cache_size = |         let consumption_cache_size = | ||||||
|             AppConfig::get().refresh_interval / AppConfig::get().energy_fetch_interval; |             AppConfig::get().refresh_interval / AppConfig::get().energy_fetch_interval; | ||||||
|         let curr_consumption = consumption::get_curr_consumption().await?; |         let curr_consumption = match consumption::get_curr_consumption().await { | ||||||
|  |             Ok(v) => v, | ||||||
|  |             Err(e) => { | ||||||
|  |                 log::warn!("Failed to fetch consumption, using default value! {e}"); | ||||||
|  |                 constants::FALLBACK_PRODUCTION_VALUE | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |         log::info!("Initial consumption value: {curr_consumption}"); | ||||||
|         let mut consumption_cache = ConsumptionCache::new(consumption_cache_size as usize); |         let mut consumption_cache = ConsumptionCache::new(consumption_cache_size as usize); | ||||||
|         consumption_cache.add_value(curr_consumption); |         consumption_cache.add_value(curr_consumption); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,14 +9,20 @@ use actix_web::HttpResponse; | |||||||
|  |  | ||||||
| #[derive(serde::Serialize)] | #[derive(serde::Serialize)] | ||||||
| struct Consumption { | struct Consumption { | ||||||
|     consumption: i32, |     consumption: Option<i32>, | ||||||
| } | } | ||||||
|  |  | ||||||
| /// Get current energy consumption | /// Get current energy consumption | ||||||
| pub async fn curr_consumption() -> HttpResult { | pub async fn curr_consumption() -> HttpResult { | ||||||
|     let consumption = consumption::get_curr_consumption().await?; |     Ok(match consumption::get_curr_consumption().await { | ||||||
|  |         Ok(v) => HttpResponse::Ok().json(Consumption { | ||||||
|     Ok(HttpResponse::Ok().json(Consumption { consumption })) |             consumption: Some(v), | ||||||
|  |         }), | ||||||
|  |         Err(e) => { | ||||||
|  |             log::error!("Failed to fetch current consumption! {e}"); | ||||||
|  |             HttpResponse::Ok().json(Consumption { consumption: None }) | ||||||
|  |         } | ||||||
|  |     }) | ||||||
| } | } | ||||||
|  |  | ||||||
| /// Get curr consumption history | /// Get curr consumption history | ||||||
| @@ -34,7 +40,9 @@ pub async fn curr_consumption_history() -> HttpResult { | |||||||
| pub async fn cached_consumption(energy_actor: WebEnergyActor) -> HttpResult { | pub async fn cached_consumption(energy_actor: WebEnergyActor) -> HttpResult { | ||||||
|     let consumption = energy_actor.send(energy_actor::GetCurrConsumption).await?; |     let consumption = energy_actor.send(energy_actor::GetCurrConsumption).await?; | ||||||
|  |  | ||||||
|     Ok(HttpResponse::Ok().json(Consumption { consumption })) |     Ok(HttpResponse::Ok().json(Consumption { | ||||||
|  |         consumption: Some(consumption), | ||||||
|  |     })) | ||||||
| } | } | ||||||
|  |  | ||||||
| /// Get current relays consumption | /// Get current relays consumption | ||||||
| @@ -42,7 +50,9 @@ pub async fn relays_consumption(energy_actor: WebEnergyActor) -> HttpResult { | |||||||
|     let consumption = |     let consumption = | ||||||
|         energy_actor.send(energy_actor::RelaysConsumption).await? as EnergyConsumption; |         energy_actor.send(energy_actor::RelaysConsumption).await? as EnergyConsumption; | ||||||
|  |  | ||||||
|     Ok(HttpResponse::Ok().json(Consumption { consumption })) |     Ok(HttpResponse::Ok().json(Consumption { | ||||||
|  |         consumption: Some(consumption), | ||||||
|  |     })) | ||||||
| } | } | ||||||
|  |  | ||||||
| pub async fn relays_consumption_history() -> HttpResult { | pub async fn relays_consumption_history() -> HttpResult { | ||||||
|   | |||||||
| @@ -13,6 +13,7 @@ struct ServerConfig { | |||||||
|     auth_disabled: bool, |     auth_disabled: bool, | ||||||
|     constraints: StaticConstraints, |     constraints: StaticConstraints, | ||||||
|     unsecure_origin: String, |     unsecure_origin: String, | ||||||
|  |     backend_version: &'static str, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Default for ServerConfig { | impl Default for ServerConfig { | ||||||
| @@ -21,6 +22,7 @@ impl Default for ServerConfig { | |||||||
|             auth_disabled: AppConfig::get().unsecure_disable_login, |             auth_disabled: AppConfig::get().unsecure_disable_login, | ||||||
|             constraints: Default::default(), |             constraints: Default::default(), | ||||||
|             unsecure_origin: AppConfig::get().unsecure_origin(), |             unsecure_origin: AppConfig::get().unsecure_origin(), | ||||||
|  |             backend_version: env!("CARGO_PKG_VERSION"), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										409
									
								
								central_frontend/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										409
									
								
								central_frontend/package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -13,10 +13,10 @@ | |||||||
|         "@fontsource/roboto": "^5.1.0", |         "@fontsource/roboto": "^5.1.0", | ||||||
|         "@mdi/js": "^7.4.47", |         "@mdi/js": "^7.4.47", | ||||||
|         "@mdi/react": "^1.6.1", |         "@mdi/react": "^1.6.1", | ||||||
|         "@mui/icons-material": "^6.1.3", |         "@mui/icons-material": "^6.1.4", | ||||||
|         "@mui/material": "^6.1.3", |         "@mui/material": "^6.1.4", | ||||||
|         "@mui/x-charts": "^7.20.0", |         "@mui/x-charts": "^7.21.0", | ||||||
|         "@mui/x-date-pickers": "^7.20.0", |         "@mui/x-date-pickers": "^7.21.0", | ||||||
|         "@types/semver": "^7.5.8", |         "@types/semver": "^7.5.8", | ||||||
|         "date-and-time": "^3.6.0", |         "date-and-time": "^3.6.0", | ||||||
|         "dayjs": "^1.11.13", |         "dayjs": "^1.11.13", | ||||||
| @@ -29,14 +29,14 @@ | |||||||
|       "devDependencies": { |       "devDependencies": { | ||||||
|         "@types/react": "^18.3.11", |         "@types/react": "^18.3.11", | ||||||
|         "@types/react-dom": "^18.3.1", |         "@types/react-dom": "^18.3.1", | ||||||
|         "@typescript-eslint/eslint-plugin": "^8.8.0", |         "@typescript-eslint/eslint-plugin": "^8.10.0", | ||||||
|         "@typescript-eslint/parser": "^8.8.0", |         "@typescript-eslint/parser": "^8.10.0", | ||||||
|         "@vitejs/plugin-react": "^4.3.2", |         "@vitejs/plugin-react": "^4.3.2", | ||||||
|         "eslint": "^8.57.1", |         "eslint": "^8.57.1", | ||||||
|         "eslint-plugin-react-hooks": "^5.0.0", |         "eslint-plugin-react-hooks": "^5.0.0", | ||||||
|         "eslint-plugin-react-refresh": "^0.4.12", |         "eslint-plugin-react-refresh": "^0.4.12", | ||||||
|         "typescript": "^5.6.3", |         "typescript": "^5.6.3", | ||||||
|         "vite": "^5.4.8" |         "vite": "^5.4.9" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@ampproject/remapping": { |     "node_modules/@ampproject/remapping": { | ||||||
| @@ -1383,17 +1383,17 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@mui/x-charts": { |     "node_modules/@mui/x-charts": { | ||||||
|       "version": "7.20.0", |       "version": "7.21.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@mui/x-charts/-/x-charts-7.20.0.tgz", |       "resolved": "https://registry.npmjs.org/@mui/x-charts/-/x-charts-7.21.0.tgz", | ||||||
|       "integrity": "sha512-mm3ERanuxWWc16dYLC54jqQp1CrHFSvWYvaXvhaXhWZdNrSIWNEY4inCbrDuGvl+i7/uNJgxeINw4SOtQ/BOFA==", |       "integrity": "sha512-Qv7U1Koo7hxinn1ncbn+Yfcwd8h3bSJDVCpjyKKgO0247kGIAK4ecrBlFHwVLol4bNTY36Ir1prEaA0G1MmUrg==", | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@babel/runtime": "^7.25.7", |         "@babel/runtime": "^7.25.7", | ||||||
|         "@mui/utils": "^5.16.6 || ^6.0.0", |         "@mui/utils": "^5.16.6 || ^6.0.0", | ||||||
|         "@mui/x-charts-vendor": "7.20.0", |         "@mui/x-charts-vendor": "7.20.0", | ||||||
|         "@mui/x-internals": "7.20.0", |         "@mui/x-internals": "7.21.0", | ||||||
|         "@react-spring/rafz": "^9.7.4", |         "@react-spring/rafz": "^9.7.5", | ||||||
|         "@react-spring/web": "^9.7.4", |         "@react-spring/web": "^9.7.5", | ||||||
|         "clsx": "^2.1.1", |         "clsx": "^2.1.1", | ||||||
|         "prop-types": "^15.8.1" |         "prop-types": "^15.8.1" | ||||||
|       }, |       }, | ||||||
| @@ -1441,14 +1441,14 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@mui/x-date-pickers": { |     "node_modules/@mui/x-date-pickers": { | ||||||
|       "version": "7.20.0", |       "version": "7.21.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@mui/x-date-pickers/-/x-date-pickers-7.20.0.tgz", |       "resolved": "https://registry.npmjs.org/@mui/x-date-pickers/-/x-date-pickers-7.21.0.tgz", | ||||||
|       "integrity": "sha512-LnijrF8IF3r7c7sAVXRX4pDurozJSMUGAJdd5xuTT7ZPQIOp5ry0kDKqx79WAjXA/ZgjropLNt/nk15GE+6ZNw==", |       "integrity": "sha512-WLpuTu3PvhYwd7IAJSuDWr1Zd8c5C8Cc7rpAYCaV5+tGBoEP0C2UKqClMR4F1wTiU2a7x3dzgQzkcgK72yyqDw==", | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@babel/runtime": "^7.25.7", |         "@babel/runtime": "^7.25.7", | ||||||
|         "@mui/utils": "^5.16.6 || ^6.0.0", |         "@mui/utils": "^5.16.6 || ^6.0.0", | ||||||
|         "@mui/x-internals": "7.20.0", |         "@mui/x-internals": "7.21.0", | ||||||
|         "@types/react-transition-group": "^4.4.11", |         "@types/react-transition-group": "^4.4.11", | ||||||
|         "clsx": "^2.1.1", |         "clsx": "^2.1.1", | ||||||
|         "prop-types": "^15.8.1", |         "prop-types": "^15.8.1", | ||||||
| @@ -1507,9 +1507,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@mui/x-internals": { |     "node_modules/@mui/x-internals": { | ||||||
|       "version": "7.20.0", |       "version": "7.21.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@mui/x-internals/-/x-internals-7.20.0.tgz", |       "resolved": "https://registry.npmjs.org/@mui/x-internals/-/x-internals-7.21.0.tgz", | ||||||
|       "integrity": "sha512-ScXdEwtnxmBEq9umeusnotfeVQnnhjOZcM2ddXyIupmzeGmgDDtEcXGyTgrS/GOc91J74g81s6eJ4UCrlYZ2sg==", |       "integrity": "sha512-94YNyZ0BhK5Z+Tkr90RKf47IVCW8R/1MvdUhh6MCQg6sZa74jsX+x+gEZ4kzuCqOsuyTyxikeQ8vVuCIQiP7UQ==", | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@babel/runtime": "^7.25.7", |         "@babel/runtime": "^7.25.7", | ||||||
| @@ -2030,17 +2030,17 @@ | |||||||
|       "license": "MIT" |       "license": "MIT" | ||||||
|     }, |     }, | ||||||
|     "node_modules/@typescript-eslint/eslint-plugin": { |     "node_modules/@typescript-eslint/eslint-plugin": { | ||||||
|       "version": "8.8.1", |       "version": "8.11.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.8.1.tgz", |       "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.11.0.tgz", | ||||||
|       "integrity": "sha512-xfvdgA8AP/vxHgtgU310+WBnLB4uJQ9XdyP17RebG26rLtDrQJV3ZYrcopX91GrHmMoH8bdSwMRh2a//TiJ1jQ==", |       "integrity": "sha512-KhGn2LjW1PJT2A/GfDpiyOfS4a8xHQv2myUagTM5+zsormOmBlYsnQ6pobJ8XxJmh6hnHwa2Mbe3fPrDJoDhbA==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@eslint-community/regexpp": "^4.10.0", |         "@eslint-community/regexpp": "^4.10.0", | ||||||
|         "@typescript-eslint/scope-manager": "8.8.1", |         "@typescript-eslint/scope-manager": "8.11.0", | ||||||
|         "@typescript-eslint/type-utils": "8.8.1", |         "@typescript-eslint/type-utils": "8.11.0", | ||||||
|         "@typescript-eslint/utils": "8.8.1", |         "@typescript-eslint/utils": "8.11.0", | ||||||
|         "@typescript-eslint/visitor-keys": "8.8.1", |         "@typescript-eslint/visitor-keys": "8.11.0", | ||||||
|         "graphemer": "^1.4.0", |         "graphemer": "^1.4.0", | ||||||
|         "ignore": "^5.3.1", |         "ignore": "^5.3.1", | ||||||
|         "natural-compare": "^1.4.0", |         "natural-compare": "^1.4.0", | ||||||
| @@ -2063,17 +2063,67 @@ | |||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/scope-manager": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-Uholz7tWhXmA4r6epo+vaeV7yjdKy5QFCERMjs1kMVsLRKIrSdM6o21W2He9ftp5PP6aWOVpD5zvrvuHZC0bMQ==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@typescript-eslint/types": "8.11.0", | ||||||
|  |         "@typescript-eslint/visitor-keys": "8.11.0" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/types": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-tn6sNMHf6EBAYMvmPUaKaVeYvhUsrE6x+bXQTxjQRp360h1giATU0WvgeEys1spbvb5R+VpNOZ+XJmjD8wOUHw==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/visitor-keys": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-EaewX6lxSjRJnc+99+dqzTeoDZUfyrA52d2/HRrkI830kgovWsmIiTfmr0NZorzqic7ga+1bS60lRBUgR3n/Bw==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@typescript-eslint/types": "8.11.0", | ||||||
|  |         "eslint-visitor-keys": "^3.4.3" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "node_modules/@typescript-eslint/parser": { |     "node_modules/@typescript-eslint/parser": { | ||||||
|       "version": "8.8.1", |       "version": "8.11.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.8.1.tgz", |       "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.11.0.tgz", | ||||||
|       "integrity": "sha512-hQUVn2Lij2NAxVFEdvIGxT9gP1tq2yM83m+by3whWFsWC+1y8pxxxHUFE1UqDu2VsGi2i6RLcv4QvouM84U+ow==", |       "integrity": "sha512-lmt73NeHdy1Q/2ul295Qy3uninSqi6wQI18XwSpm8w0ZbQXUpjCAWP1Vlv/obudoBiIjJVjlztjQ+d/Md98Yxg==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "BSD-2-Clause", |       "license": "BSD-2-Clause", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@typescript-eslint/scope-manager": "8.8.1", |         "@typescript-eslint/scope-manager": "8.11.0", | ||||||
|         "@typescript-eslint/types": "8.8.1", |         "@typescript-eslint/types": "8.11.0", | ||||||
|         "@typescript-eslint/typescript-estree": "8.8.1", |         "@typescript-eslint/typescript-estree": "8.11.0", | ||||||
|         "@typescript-eslint/visitor-keys": "8.8.1", |         "@typescript-eslint/visitor-keys": "8.11.0", | ||||||
|         "debug": "^4.3.4" |         "debug": "^4.3.4" | ||||||
|       }, |       }, | ||||||
|       "engines": { |       "engines": { | ||||||
| @@ -2092,15 +2142,94 @@ | |||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@typescript-eslint/scope-manager": { |     "node_modules/@typescript-eslint/parser/node_modules/@typescript-eslint/scope-manager": { | ||||||
|       "version": "8.8.1", |       "version": "8.11.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.8.1.tgz", |       "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.11.0.tgz", | ||||||
|       "integrity": "sha512-X4JdU+66Mazev/J0gfXlcC/dV6JI37h+93W9BRYXrSn0hrE64IoWgVkO9MSJgEzoWkxONgaQpICWg8vAN74wlA==", |       "integrity": "sha512-Uholz7tWhXmA4r6epo+vaeV7yjdKy5QFCERMjs1kMVsLRKIrSdM6o21W2He9ftp5PP6aWOVpD5zvrvuHZC0bMQ==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@typescript-eslint/types": "8.8.1", |         "@typescript-eslint/types": "8.11.0", | ||||||
|         "@typescript-eslint/visitor-keys": "8.8.1" |         "@typescript-eslint/visitor-keys": "8.11.0" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/parser/node_modules/@typescript-eslint/types": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-tn6sNMHf6EBAYMvmPUaKaVeYvhUsrE6x+bXQTxjQRp360h1giATU0WvgeEys1spbvb5R+VpNOZ+XJmjD8wOUHw==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/parser/node_modules/@typescript-eslint/typescript-estree": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-yHC3s1z1RCHoCz5t06gf7jH24rr3vns08XXhfEqzYpd6Hll3z/3g23JRi0jM8A47UFKNc3u/y5KIMx8Ynbjohg==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "BSD-2-Clause", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@typescript-eslint/types": "8.11.0", | ||||||
|  |         "@typescript-eslint/visitor-keys": "8.11.0", | ||||||
|  |         "debug": "^4.3.4", | ||||||
|  |         "fast-glob": "^3.3.2", | ||||||
|  |         "is-glob": "^4.0.3", | ||||||
|  |         "minimatch": "^9.0.4", | ||||||
|  |         "semver": "^7.6.0", | ||||||
|  |         "ts-api-utils": "^1.3.0" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       }, | ||||||
|  |       "peerDependenciesMeta": { | ||||||
|  |         "typescript": { | ||||||
|  |           "optional": true | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/parser/node_modules/@typescript-eslint/visitor-keys": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-EaewX6lxSjRJnc+99+dqzTeoDZUfyrA52d2/HRrkI830kgovWsmIiTfmr0NZorzqic7ga+1bS60lRBUgR3n/Bw==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@typescript-eslint/types": "8.11.0", | ||||||
|  |         "eslint-visitor-keys": "^3.4.3" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/scope-manager": { | ||||||
|  |       "version": "8.10.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.10.0.tgz", | ||||||
|  |       "integrity": "sha512-AgCaEjhfql9MDKjMUxWvH7HjLeBqMCBfIaBbzzIcBbQPZE7CPh1m6FF+L75NUMJFMLYhCywJXIDEMa3//1A0dw==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@typescript-eslint/types": "8.10.0", | ||||||
|  |         "@typescript-eslint/visitor-keys": "8.10.0" | ||||||
|       }, |       }, | ||||||
|       "engines": { |       "engines": { | ||||||
|         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
| @@ -2111,14 +2240,14 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@typescript-eslint/type-utils": { |     "node_modules/@typescript-eslint/type-utils": { | ||||||
|       "version": "8.8.1", |       "version": "8.11.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.8.1.tgz", |       "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.11.0.tgz", | ||||||
|       "integrity": "sha512-qSVnpcbLP8CALORf0za+vjLYj1Wp8HSoiI8zYU5tHxRVj30702Z1Yw4cLwfNKhTPWp5+P+k1pjmD5Zd1nhxiZA==", |       "integrity": "sha512-ItiMfJS6pQU0NIKAaybBKkuVzo6IdnAhPFZA/2Mba/uBjuPQPet/8+zh5GtLHwmuFRShZx+8lhIs7/QeDHflOg==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@typescript-eslint/typescript-estree": "8.8.1", |         "@typescript-eslint/typescript-estree": "8.11.0", | ||||||
|         "@typescript-eslint/utils": "8.8.1", |         "@typescript-eslint/utils": "8.11.0", | ||||||
|         "debug": "^4.3.4", |         "debug": "^4.3.4", | ||||||
|         "ts-api-utils": "^1.3.0" |         "ts-api-utils": "^1.3.0" | ||||||
|       }, |       }, | ||||||
| @@ -2135,10 +2264,71 @@ | |||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/types": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-tn6sNMHf6EBAYMvmPUaKaVeYvhUsrE6x+bXQTxjQRp360h1giATU0WvgeEys1spbvb5R+VpNOZ+XJmjD8wOUHw==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/typescript-estree": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-yHC3s1z1RCHoCz5t06gf7jH24rr3vns08XXhfEqzYpd6Hll3z/3g23JRi0jM8A47UFKNc3u/y5KIMx8Ynbjohg==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "BSD-2-Clause", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@typescript-eslint/types": "8.11.0", | ||||||
|  |         "@typescript-eslint/visitor-keys": "8.11.0", | ||||||
|  |         "debug": "^4.3.4", | ||||||
|  |         "fast-glob": "^3.3.2", | ||||||
|  |         "is-glob": "^4.0.3", | ||||||
|  |         "minimatch": "^9.0.4", | ||||||
|  |         "semver": "^7.6.0", | ||||||
|  |         "ts-api-utils": "^1.3.0" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       }, | ||||||
|  |       "peerDependenciesMeta": { | ||||||
|  |         "typescript": { | ||||||
|  |           "optional": true | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/type-utils/node_modules/@typescript-eslint/visitor-keys": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-EaewX6lxSjRJnc+99+dqzTeoDZUfyrA52d2/HRrkI830kgovWsmIiTfmr0NZorzqic7ga+1bS60lRBUgR3n/Bw==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@typescript-eslint/types": "8.11.0", | ||||||
|  |         "eslint-visitor-keys": "^3.4.3" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "node_modules/@typescript-eslint/types": { |     "node_modules/@typescript-eslint/types": { | ||||||
|       "version": "8.8.1", |       "version": "8.10.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.8.1.tgz", |       "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.10.0.tgz", | ||||||
|       "integrity": "sha512-WCcTP4SDXzMd23N27u66zTKMuEevH4uzU8C9jf0RO4E04yVHgQgW+r+TeVTNnO1KIfrL8ebgVVYYMMO3+jC55Q==", |       "integrity": "sha512-k/E48uzsfJCRRbGLapdZgrX52csmWJ2rcowwPvOZ8lwPUv3xW6CcFeJAXgx4uJm+Ge4+a4tFOkdYvSpxhRhg1w==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "engines": { |       "engines": { | ||||||
| @@ -2150,14 +2340,14 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@typescript-eslint/typescript-estree": { |     "node_modules/@typescript-eslint/typescript-estree": { | ||||||
|       "version": "8.8.1", |       "version": "8.10.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.8.1.tgz", |       "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.10.0.tgz", | ||||||
|       "integrity": "sha512-A5d1R9p+X+1js4JogdNilDuuq+EHZdsH9MjTVxXOdVFfTJXunKJR/v+fNNyO4TnoOn5HqobzfRlc70NC6HTcdg==", |       "integrity": "sha512-3OE0nlcOHaMvQ8Xu5gAfME3/tWVDpb/HxtpUZ1WeOAksZ/h/gwrBzCklaGzwZT97/lBbbxJ16dMA98JMEngW4w==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "BSD-2-Clause", |       "license": "BSD-2-Clause", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@typescript-eslint/types": "8.8.1", |         "@typescript-eslint/types": "8.10.0", | ||||||
|         "@typescript-eslint/visitor-keys": "8.8.1", |         "@typescript-eslint/visitor-keys": "8.10.0", | ||||||
|         "debug": "^4.3.4", |         "debug": "^4.3.4", | ||||||
|         "fast-glob": "^3.3.2", |         "fast-glob": "^3.3.2", | ||||||
|         "is-glob": "^4.0.3", |         "is-glob": "^4.0.3", | ||||||
| @@ -2179,16 +2369,16 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@typescript-eslint/utils": { |     "node_modules/@typescript-eslint/utils": { | ||||||
|       "version": "8.8.1", |       "version": "8.11.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.8.1.tgz", |       "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.11.0.tgz", | ||||||
|       "integrity": "sha512-/QkNJDbV0bdL7H7d0/y0qBbV2HTtf0TIyjSDTvvmQEzeVx8jEImEbLuOA4EsvE8gIgqMitns0ifb5uQhMj8d9w==", |       "integrity": "sha512-CYiX6WZcbXNJV7UNB4PLDIBtSdRmRI/nb0FMyqHPTQD1rMjA0foPLaPUV39C/MxkTd/QKSeX+Gb34PPsDVC35g==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@eslint-community/eslint-utils": "^4.4.0", |         "@eslint-community/eslint-utils": "^4.4.0", | ||||||
|         "@typescript-eslint/scope-manager": "8.8.1", |         "@typescript-eslint/scope-manager": "8.11.0", | ||||||
|         "@typescript-eslint/types": "8.8.1", |         "@typescript-eslint/types": "8.11.0", | ||||||
|         "@typescript-eslint/typescript-estree": "8.8.1" |         "@typescript-eslint/typescript-estree": "8.11.0" | ||||||
|       }, |       }, | ||||||
|       "engines": { |       "engines": { | ||||||
|         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
| @@ -2201,14 +2391,93 @@ | |||||||
|         "eslint": "^8.57.0 || ^9.0.0" |         "eslint": "^8.57.0 || ^9.0.0" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/@typescript-eslint/visitor-keys": { |     "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/scope-manager": { | ||||||
|       "version": "8.8.1", |       "version": "8.11.0", | ||||||
|       "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.8.1.tgz", |       "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.11.0.tgz", | ||||||
|       "integrity": "sha512-0/TdC3aeRAsW7MDvYRwEc1Uwm0TIBfzjPFgg60UU2Haj5qsCs9cc3zNgY71edqE3LbWfF/WoZQd3lJoDXFQpag==", |       "integrity": "sha512-Uholz7tWhXmA4r6epo+vaeV7yjdKy5QFCERMjs1kMVsLRKIrSdM6o21W2He9ftp5PP6aWOVpD5zvrvuHZC0bMQ==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|         "@typescript-eslint/types": "8.8.1", |         "@typescript-eslint/types": "8.11.0", | ||||||
|  |         "@typescript-eslint/visitor-keys": "8.11.0" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/types": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-tn6sNMHf6EBAYMvmPUaKaVeYvhUsrE6x+bXQTxjQRp360h1giATU0WvgeEys1spbvb5R+VpNOZ+XJmjD8wOUHw==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/typescript-estree": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-yHC3s1z1RCHoCz5t06gf7jH24rr3vns08XXhfEqzYpd6Hll3z/3g23JRi0jM8A47UFKNc3u/y5KIMx8Ynbjohg==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "BSD-2-Clause", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@typescript-eslint/types": "8.11.0", | ||||||
|  |         "@typescript-eslint/visitor-keys": "8.11.0", | ||||||
|  |         "debug": "^4.3.4", | ||||||
|  |         "fast-glob": "^3.3.2", | ||||||
|  |         "is-glob": "^4.0.3", | ||||||
|  |         "minimatch": "^9.0.4", | ||||||
|  |         "semver": "^7.6.0", | ||||||
|  |         "ts-api-utils": "^1.3.0" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       }, | ||||||
|  |       "peerDependenciesMeta": { | ||||||
|  |         "typescript": { | ||||||
|  |           "optional": true | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/utils/node_modules/@typescript-eslint/visitor-keys": { | ||||||
|  |       "version": "8.11.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.11.0.tgz", | ||||||
|  |       "integrity": "sha512-EaewX6lxSjRJnc+99+dqzTeoDZUfyrA52d2/HRrkI830kgovWsmIiTfmr0NZorzqic7ga+1bS60lRBUgR3n/Bw==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@typescript-eslint/types": "8.11.0", | ||||||
|  |         "eslint-visitor-keys": "^3.4.3" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": "^18.18.0 || ^20.9.0 || >=21.1.0" | ||||||
|  |       }, | ||||||
|  |       "funding": { | ||||||
|  |         "type": "opencollective", | ||||||
|  |         "url": "https://opencollective.com/typescript-eslint" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@typescript-eslint/visitor-keys": { | ||||||
|  |       "version": "8.10.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.10.0.tgz", | ||||||
|  |       "integrity": "sha512-k8nekgqwr7FadWk548Lfph6V3r9OVqjzAIVskE7orMZR23cGJjAOVazsZSJW+ElyjfTM4wx/1g88Mi70DDtG9A==", | ||||||
|  |       "dev": true, | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@typescript-eslint/types": "8.10.0", | ||||||
|         "eslint-visitor-keys": "^3.4.3" |         "eslint-visitor-keys": "^3.4.3" | ||||||
|       }, |       }, | ||||||
|       "engines": { |       "engines": { | ||||||
| @@ -2227,9 +2496,9 @@ | |||||||
|       "license": "ISC" |       "license": "ISC" | ||||||
|     }, |     }, | ||||||
|     "node_modules/@vitejs/plugin-react": { |     "node_modules/@vitejs/plugin-react": { | ||||||
|       "version": "4.3.2", |       "version": "4.3.3", | ||||||
|       "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.3.2.tgz", |       "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.3.3.tgz", | ||||||
|       "integrity": "sha512-hieu+o05v4glEBucTcKMK3dlES0OeJlD9YVOAPraVMOInBCwzumaIFiUjr4bHK7NPgnAHgiskUoceKercrN8vg==", |       "integrity": "sha512-NooDe9GpHGqNns1i8XDERg0Vsg5SSYRhRxxyTGogUdkdNt47jal+fbuYi+Yfq6pzRCKXyoPcWisfxE6RIM3GKA==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
| @@ -2851,9 +3120,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/eslint-plugin-react-refresh": { |     "node_modules/eslint-plugin-react-refresh": { | ||||||
|       "version": "0.4.12", |       "version": "0.4.13", | ||||||
|       "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.12.tgz", |       "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.13.tgz", | ||||||
|       "integrity": "sha512-9neVjoGv20FwYtCP6CB1dzR1vr57ZDNOXst21wd2xJ/cTlM2xLq0GWVlSNTdMn/4BtP6cHYBMCSp1wFBJ9jBsg==", |       "integrity": "sha512-f1EppwrpJRWmqDTyvAyomFVDYRtrS7iTEqv3nokETnMiMzs2SSTmKRTACce4O2p4jYyowiSMvpdwC/RLcMFhuQ==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "peerDependencies": { |       "peerDependencies": { | ||||||
|   | |||||||
| @@ -15,10 +15,10 @@ | |||||||
|     "@fontsource/roboto": "^5.1.0", |     "@fontsource/roboto": "^5.1.0", | ||||||
|     "@mdi/js": "^7.4.47", |     "@mdi/js": "^7.4.47", | ||||||
|     "@mdi/react": "^1.6.1", |     "@mdi/react": "^1.6.1", | ||||||
|     "@mui/icons-material": "^6.1.3", |     "@mui/icons-material": "^6.1.4", | ||||||
|     "@mui/material": "^6.1.3", |     "@mui/material": "^6.1.4", | ||||||
|     "@mui/x-charts": "^7.20.0", |     "@mui/x-charts": "^7.21.0", | ||||||
|     "@mui/x-date-pickers": "^7.20.0", |     "@mui/x-date-pickers": "^7.21.0", | ||||||
|     "@types/semver": "^7.5.8", |     "@types/semver": "^7.5.8", | ||||||
|     "date-and-time": "^3.6.0", |     "date-and-time": "^3.6.0", | ||||||
|     "dayjs": "^1.11.13", |     "dayjs": "^1.11.13", | ||||||
| @@ -31,13 +31,13 @@ | |||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@types/react": "^18.3.11", |     "@types/react": "^18.3.11", | ||||||
|     "@types/react-dom": "^18.3.1", |     "@types/react-dom": "^18.3.1", | ||||||
|     "@typescript-eslint/eslint-plugin": "^8.8.0", |     "@typescript-eslint/eslint-plugin": "^8.10.0", | ||||||
|     "@typescript-eslint/parser": "^8.8.0", |     "@typescript-eslint/parser": "^8.10.0", | ||||||
|     "@vitejs/plugin-react": "^4.3.2", |     "@vitejs/plugin-react": "^4.3.2", | ||||||
|     "eslint": "^8.57.1", |     "eslint": "^8.57.1", | ||||||
|     "eslint-plugin-react-hooks": "^5.0.0", |     "eslint-plugin-react-hooks": "^5.0.0", | ||||||
|     "eslint-plugin-react-refresh": "^0.4.12", |     "eslint-plugin-react-refresh": "^0.4.12", | ||||||
|     "typescript": "^5.6.3", |     "typescript": "^5.6.3", | ||||||
|     "vite": "^5.4.8" |     "vite": "^5.4.9" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ export interface ServerConfig { | |||||||
|   auth_disabled: boolean; |   auth_disabled: boolean; | ||||||
|   constraints: ServerConstraint; |   constraints: ServerConstraint; | ||||||
|   unsecure_origin: string; |   unsecure_origin: string; | ||||||
|  |   backend_version: string; | ||||||
| } | } | ||||||
|  |  | ||||||
| export interface ServerConstraint { | export interface ServerConstraint { | ||||||
|   | |||||||
| @@ -12,9 +12,11 @@ import { | |||||||
|   ListItemButton, |   ListItemButton, | ||||||
|   ListItemIcon, |   ListItemIcon, | ||||||
|   ListItemText, |   ListItemText, | ||||||
|  |   Typography, | ||||||
| } from "@mui/material"; | } from "@mui/material"; | ||||||
| import { useLocation } from "react-router-dom"; | import { useLocation } from "react-router-dom"; | ||||||
| import { RouterLink } from "./RouterLink"; | import { RouterLink } from "./RouterLink"; | ||||||
|  | import { ServerApi } from "../api/ServerApi"; | ||||||
|  |  | ||||||
| export function SolarEnergyNavList(): React.ReactElement { | export function SolarEnergyNavList(): React.ReactElement { | ||||||
|   return ( |   return ( | ||||||
| @@ -52,6 +54,13 @@ export function SolarEnergyNavList(): React.ReactElement { | |||||||
|         uri="/logs" |         uri="/logs" | ||||||
|         icon={<Icon path={mdiNotebookMultiple} size={1} />} |         icon={<Icon path={mdiNotebookMultiple} size={1} />} | ||||||
|       /> |       /> | ||||||
|  |       <Typography | ||||||
|  |         variant="caption" | ||||||
|  |         component="div" | ||||||
|  |         style={{ textAlign: "center", width: "100%", marginTop: "30px" }} | ||||||
|  |       > | ||||||
|  |         Version {ServerApi.Config.backend_version} | ||||||
|  |       </Typography> | ||||||
|     </List> |     </List> | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										1709
									
								
								custom_consumption/Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1709
									
								
								custom_consumption/Cargo.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -7,6 +7,6 @@ edition = "2021" | |||||||
| env_logger = "0.11.5" | env_logger = "0.11.5" | ||||||
| log = "0.4.22" | log = "0.4.22" | ||||||
| clap = { version = "4.5.18", features = ["derive", "env"] } | clap = { version = "4.5.18", features = ["derive", "env"] } | ||||||
| egui = "0.28.1" | egui = "0.29.1" | ||||||
| eframe = "0.28.1" | eframe = "0.29.1" | ||||||
| lazy_static = "1.5.0" | lazy_static = "1.5.0" | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
| # Configure project for production | # Configure project for production | ||||||
|  |  | ||||||
|  | Note: This guide assumes that you use the default hostname, `central.internal` as hostname for your central system. | ||||||
|  |  | ||||||
| ## Create production build | ## Create production build | ||||||
|  |  | ||||||
| ### Central | ### Central | ||||||
| @@ -44,5 +46,146 @@ The OTA update is then located in `build/main.bin` | |||||||
| * A server running a recent Linux (Debian / Ubuntu preferred) with `central` as hostname | * A server running a recent Linux (Debian / Ubuntu preferred) with `central` as hostname | ||||||
| * DHCP configured on the network | * DHCP configured on the network | ||||||
|  |  | ||||||
|  | ## Configure DNS server | ||||||
|  |  | ||||||
|  | If you need to setup a DNS server / proxy to point `central.internal` to the central server IP, you can follow this guide. | ||||||
|  |  | ||||||
|  | ### Retrieve DNS server binary | ||||||
|  | Use [DNSProxy](https://gitlab.com/pierre42100/dnsproxy) as DNS server. Get and compile the sources: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | git clone https://gitlab.com/pierre42100/dnsproxy | ||||||
|  | cd dnsproxy | ||||||
|  | cargo build --release | ||||||
|  | scp target/release/dns_proxy USER@CENTRAL_IP:/home/USER | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Then, on the target server, install the binary to its final destination: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | sudo mv dns_proxy /usr/local/bin/ | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Configure DNS server | ||||||
|  | Configure the server as a service `/etc/systemd/system/dns.service`: | ||||||
|  |  | ||||||
|  | ```conf | ||||||
|  | [Unit] | ||||||
|  | Description=DNS server | ||||||
|  | After=syslog.target | ||||||
|  | After=network.target | ||||||
|  |  | ||||||
|  | [Service] | ||||||
|  | RestartSec=2s | ||||||
|  | Type=simple | ||||||
|  | User=root | ||||||
|  | Group=root | ||||||
|  | WorkingDirectory=/tmp | ||||||
|  | ExecStart=/usr/local/bin/dns_proxy -l "CENTRAL_IP:53" -c "central.internal. A CENTRAL_IP" | ||||||
|  | Restart=always | ||||||
|  |  | ||||||
|  | [Install] | ||||||
|  | WantedBy=multi-user.target | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Enable and start the new service: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | sudo systemctl enable dns | ||||||
|  | sudo systemctl start dns | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Check that it works correctly: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | dig central.internal. @CENTRAL_IP | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | You should get an entry like this if it works: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | ;; ANSWER SECTION: | ||||||
|  | central.internal.	0	IN	A	CENTRAL_IP | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Then, in your DHCP service, define the central as the DNS server. | ||||||
|  |  | ||||||
| ## Configure server | ## Configure server | ||||||
| TODO |  | ||||||
|  | ### Create a user dedicated to the central | ||||||
|  | ```bash | ||||||
|  | sudo adduser --disabled-login central | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Install binary | ||||||
|  | You can use `scp` to copy the binary to the target server: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | scp central_backend/target/release/central_backend pierre@central:/home/pierre | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Then the executable must be installed system-wide: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | sudo mv central_backend /usr/local/bin/ | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Create configuration file | ||||||
|  | Create a configuration file in `/home/central/config.yaml`: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | sudo touch /home/central/config.yaml | ||||||
|  | sudo chown central:central /home/central/config.yaml | ||||||
|  | sudo chmod 400 /home/central/config.yaml | ||||||
|  | sudo nano /home/central/config.yaml | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Sample configuration: | ||||||
|  |  | ||||||
|  | ```conf | ||||||
|  | SECRET=RANDOM_VALUE | ||||||
|  | COOKIE_SECURE=true | ||||||
|  | LISTEN_ADDRESS=0.0.0.0:443 | ||||||
|  | ADMIN_USERNAME=admin | ||||||
|  | ADMIN_PASSWORD=FIXME | ||||||
|  | HOSTNAME=central.internal | ||||||
|  | STORAGE=/home/central/storage | ||||||
|  | FRONIUS_ORIG=http://10.0.0.10 | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Test configuration | ||||||
|  | Run the following command to check if the configuration is working: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | sudo -u central central_backend -c /home/central/config.yaml fronius -c | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Create systemd unit file | ||||||
|  | Once you confirmed the configuration is working, you can configure a system service, in `/etc/systemd/system/central.service`: | ||||||
|  |  | ||||||
|  | ```conf | ||||||
|  | [Unit] | ||||||
|  | Description=Central backend server | ||||||
|  | After=syslog.target | ||||||
|  | After=network.target | ||||||
|  |  | ||||||
|  | [Service] | ||||||
|  | RestartSec=2s | ||||||
|  | Type=simple | ||||||
|  | User=central | ||||||
|  | Group=central | ||||||
|  | WorkingDirectory=/home/central | ||||||
|  | ExecStart=/usr/local/bin/central_backend -c /home/central/config.yaml fronius -c | ||||||
|  | Restart=always | ||||||
|  | Environment=USER=central  | ||||||
|  | HOME=/home/central | ||||||
|  |  | ||||||
|  | [Install] | ||||||
|  | WantedBy=multi-user.target | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Enable & start service: | ||||||
|  | ```bash | ||||||
|  | sudo systemctl enable central | ||||||
|  | sudo systemctl start central | ||||||
|  | ``` | ||||||
| @@ -8,7 +8,7 @@ | |||||||
| /** | /** | ||||||
|  * Backend unsecure API URL |  * Backend unsecure API URL | ||||||
|  */ |  */ | ||||||
| #define BACKEND_UNSECURE_URL "http://devweb.internal:8080" | #define BACKEND_UNSECURE_URL "http://central.internal:8080" | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Device name len |  * Device name len | ||||||
|   | |||||||
| @@ -1 +1 @@ | |||||||
| 1.0.0 | 1.0.2-b1 | ||||||
		Reference in New Issue
	
	Block a user