First step toward supporting multiple database writes
This commit is contained in:
parent
aefd8cde25
commit
c88858e43f
@ -45,20 +45,22 @@ pub struct UserAuth {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Deserialize)]
|
#[derive(Debug, Deserialize)]
|
||||||
|
#[serde(rename_all = "kebab-case", tag = "type")]
|
||||||
|
pub enum Database {
|
||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
pub struct InfluxDBConfig {
|
Influxdb {
|
||||||
pub url: String,
|
url: String,
|
||||||
pub auth: Option<UserAuth>,
|
auth: Option<UserAuth>,
|
||||||
pub db_name: String,
|
db_name: String,
|
||||||
pub measurement: String,
|
measurement: String,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Deserialize, Clone)]
|
#[derive(Debug, Deserialize, Clone)]
|
||||||
#[serde(rename_all = "camelCase", tag = "type")]
|
#[serde(rename_all = "kebab-case", tag = "type")]
|
||||||
#[allow(non_camel_case_types)]
|
|
||||||
pub enum Payload {
|
pub enum Payload {
|
||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
json {
|
Json {
|
||||||
value_field_path: String,
|
value_field_path: String,
|
||||||
timestamp_field_path: Option<String>,
|
timestamp_field_path: Option<String>,
|
||||||
},
|
},
|
||||||
@ -79,7 +81,7 @@ pub struct Mapping {
|
|||||||
pub struct Config {
|
pub struct Config {
|
||||||
pub log_level: Option<LevelFilter>,
|
pub log_level: Option<LevelFilter>,
|
||||||
pub mqtt: MqttConfig,
|
pub mqtt: MqttConfig,
|
||||||
pub database: InfluxDBConfig,
|
pub databases: Vec<Database>,
|
||||||
pub mappings: Vec<Mapping>,
|
pub mappings: Vec<Mapping>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
46
src/main.rs
46
src/main.rs
@ -1,7 +1,7 @@
|
|||||||
#[macro_use]
|
#[macro_use]
|
||||||
extern crate log;
|
extern crate log;
|
||||||
|
|
||||||
use config::{Config, InfluxDBConfig, MqttAuth, MqttConfig, UserAuth};
|
use config::{Config, Database as ConfigDatabase, MqttAuth, MqttConfig, UserAuth};
|
||||||
use futures::TryFutureExt;
|
use futures::TryFutureExt;
|
||||||
use influxdb::InfluxDbWriteable;
|
use influxdb::InfluxDbWriteable;
|
||||||
use influxdb::{Client as InfluxClient, Timestamp, Type};
|
use influxdb::{Client as InfluxClient, Timestamp, Type};
|
||||||
@ -89,16 +89,20 @@ async fn init_mqtt(config: &MqttConfig) -> Result<(MqttAsyncClient, MqttEventLoo
|
|||||||
Ok(MqttAsyncClient::new(options, 100))
|
Ok(MqttAsyncClient::new(options, 100))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn init_db(config: &InfluxDBConfig) -> Result<Database, String> {
|
fn init_db(config: &ConfigDatabase) -> Result<Database, String> {
|
||||||
let mut client = InfluxClient::new(config.url.clone(), config.db_name.clone());
|
match config {
|
||||||
if let Some(UserAuth { username, password }) = &config.auth {
|
ConfigDatabase::Influxdb { url, auth, db_name, measurement } => {
|
||||||
|
let mut client = InfluxClient::new(url.clone(), db_name.clone());
|
||||||
|
if let Some(UserAuth { username, password }) = auth {
|
||||||
client = client.with_auth(username, password);
|
client = client.with_auth(username, password);
|
||||||
}
|
}
|
||||||
Ok(Database {
|
Ok(Database {
|
||||||
client,
|
client,
|
||||||
measurement: config.measurement.clone(),
|
measurement: measurement.clone(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async fn init_subscriptions(
|
async fn init_subscriptions(
|
||||||
mqtt_client: &mut MqttAsyncClient,
|
mqtt_client: &mut MqttAsyncClient,
|
||||||
@ -121,7 +125,7 @@ async fn init_subscriptions(
|
|||||||
async fn handle_publish(
|
async fn handle_publish(
|
||||||
publish: &Publish,
|
publish: &Publish,
|
||||||
mapping: Arc<Mapping>,
|
mapping: Arc<Mapping>,
|
||||||
database: Arc<Database>,
|
databases: Arc<Vec<Database>>,
|
||||||
) -> Result<(), String> {
|
) -> Result<(), String> {
|
||||||
debug!("Got publish: {:?}; {:?}", publish, publish.payload);
|
debug!("Got publish: {:?}; {:?}", publish, publish.payload);
|
||||||
|
|
||||||
@ -170,9 +174,11 @@ async fn handle_publish(
|
|||||||
.unwrap()
|
.unwrap()
|
||||||
.as_millis()
|
.as_millis()
|
||||||
);
|
);
|
||||||
|
|
||||||
|
for database in databases.iter() {
|
||||||
let mut query = Timestamp::Milliseconds(timestamp)
|
let mut query = Timestamp::Milliseconds(timestamp)
|
||||||
.into_query(&database.measurement)
|
.into_query(&database.measurement)
|
||||||
.add_field(&field_name, influx_value);
|
.add_field(&field_name, influx_value.clone());
|
||||||
for tag in mapping.tags.iter() {
|
for tag in mapping.tags.iter() {
|
||||||
let value = match &tag.1 {
|
let value = match &tag.1 {
|
||||||
TagValue::Literal(v) => v.clone(),
|
TagValue::Literal(v) => v.clone(),
|
||||||
@ -189,6 +195,7 @@ async fn handle_publish(
|
|||||||
.await
|
.await
|
||||||
.map_err(|err| format!("Failed to write to DB: {}", err))?;
|
.map_err(|err| format!("Failed to write to DB: {}", err))?;
|
||||||
debug!("wrote to influx: {:?}", query);
|
debug!("wrote to influx: {:?}", query);
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -216,17 +223,20 @@ fn find_mapping<'a>(mappings: &'a Vec<Arc<Mapping>>, topic: &String) -> Option<&
|
|||||||
|
|
||||||
async fn run_event_loop(
|
async fn run_event_loop(
|
||||||
mut event_loop: MqttEventLoop,
|
mut event_loop: MqttEventLoop,
|
||||||
mappings: &Vec<Arc<Mapping>>,
|
mappings: Vec<Mapping>,
|
||||||
database: Arc<Database>,
|
databases: Vec<Database>,
|
||||||
) {
|
) {
|
||||||
|
let mappings = mappings.into_iter().map(Arc::new).collect();
|
||||||
|
let databases = Arc::new(databases);
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
match event_loop.poll().await {
|
match event_loop.poll().await {
|
||||||
Ok(Event::Incoming(Packet::Publish(publish))) => {
|
Ok(Event::Incoming(Packet::Publish(publish))) => {
|
||||||
if let Some(mapping) = find_mapping(&mappings, &publish.topic) {
|
if let Some(mapping) = find_mapping(&mappings, &publish.topic) {
|
||||||
let mapping = Arc::clone(mapping);
|
let mapping = Arc::clone(mapping);
|
||||||
let database = Arc::clone(&database);
|
let databases = Arc::clone(&databases);
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
if let Err(err) = handle_publish(&publish, mapping, database).await {
|
if let Err(err) = handle_publish(&publish, mapping, databases).await {
|
||||||
warn!("{}", err);
|
warn!("{}", err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -256,14 +266,11 @@ async fn main() -> Result<(), String> {
|
|||||||
}
|
}
|
||||||
logger_builder.init();
|
logger_builder.init();
|
||||||
|
|
||||||
let mappings: Vec<Arc<Mapping>> = config
|
let mappings: Vec<Mapping> = config
|
||||||
.mappings
|
.mappings
|
||||||
.iter()
|
.iter()
|
||||||
.map(Mapping::try_from)
|
.map(Mapping::try_from)
|
||||||
.collect::<Result<Vec<Mapping>, String>>()?
|
.collect::<Result<Vec<Mapping>, String>>()?;
|
||||||
.into_iter()
|
|
||||||
.map(Arc::new)
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let (mut mqtt_client, mqtt_event_loop) = init_mqtt(&config.mqtt).await?;
|
let (mut mqtt_client, mqtt_event_loop) = init_mqtt(&config.mqtt).await?;
|
||||||
init_subscriptions(
|
init_subscriptions(
|
||||||
@ -276,9 +283,12 @@ async fn main() -> Result<(), String> {
|
|||||||
)
|
)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
let database = Arc::new(init_db(&config.database)?);
|
let databases = config.databases
|
||||||
|
.iter()
|
||||||
|
.map(init_db)
|
||||||
|
.collect::<Result<Vec<Database>, String>>()?;
|
||||||
|
|
||||||
run_event_loop(mqtt_event_loop, &mappings, database).await;
|
run_event_loop(mqtt_event_loop, mappings, databases).await;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -113,7 +113,7 @@ impl TryFrom<&ConfigMapping> for Mapping {
|
|||||||
|
|
||||||
let payload = match &mapping.payload {
|
let payload = match &mapping.payload {
|
||||||
None => Payload::Raw,
|
None => Payload::Raw,
|
||||||
Some(ConfigPayload::json { value_field_path, timestamp_field_path }) => {
|
Some(ConfigPayload::Json { value_field_path, timestamp_field_path }) => {
|
||||||
let value_field_selector = Selector::new(&value_field_path)
|
let value_field_selector = Selector::new(&value_field_path)
|
||||||
.map_err(|err| format!("Value field path '{}' is invalid: {}'", value_field_path, err))?;
|
.map_err(|err| format!("Value field path '{}' is invalid: {}'", value_field_path, err))?;
|
||||||
let timestamp_field_selector = timestamp_field_path.as_ref()
|
let timestamp_field_selector = timestamp_field_path.as_ref()
|
||||||
|
Loading…
Reference in New Issue
Block a user