1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
//
// Copyright (C) 2018 Kubos Corporation
//
// Licensed under the Apache License, Version 2.0 (the "License")
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//

use kubos_telemetry_db::Database;
use log::info;
use serde_json::{self, Value};
use std::net::{SocketAddr, UdpSocket};
use std::sync::{Arc, Mutex};

pub struct DirectUdp {
    db: Arc<Mutex<Database>>,
}

impl DirectUdp {
    pub fn new(db: Arc<Mutex<Database>>) -> Self {
        DirectUdp { db }
    }

    pub fn start(&self, url: String) {
        let socket = UdpSocket::bind(url.parse::<SocketAddr>().unwrap()).unwrap();
        info!("Direct UDP listening on: {}", socket.local_addr().unwrap());

        loop {
            // Wait for an incoming message
            let mut buf = [0; 4096];
            let (size, _peer) = socket
                .recv_from(&mut buf)
                .map_err(|err| format!("Failed to receive a message: {}", err))
                .unwrap();

            if let Ok(msg) = serde_json::from_slice(&buf[0..(size)]) {
                // Go process the request
                let _res = self.process(msg);
            }
        }
    }

    fn process(&self, message: Value) -> Result<(), String> {
        let timestamp = serde_json::from_value::<f64>(message["timestamp"].clone()).ok();

        let subsystem = serde_json::from_value::<String>(message["subsystem"].clone())
            .map_err(|err| format!("Failed to parse subsystem parameter: {}", err))?;

        let param = serde_json::from_value::<String>(message["parameter"].clone())
            .map_err(|err| format!("Failed to parse parameter parameter: {}", err))?;

        let value = serde_json::from_value::<String>(message["value"].clone())
            .map_err(|err| format!("Failed to parse value parameter: {}", err))?;

        if let Some(time) = timestamp {
            self.db
                .lock()
                .map_err(|err| format!("{}", err))?
                .insert(time, &subsystem, &param, &value)
                .map_err(|err| format!("{}", err))?;
        } else {
            self.db
                .lock()
                .map_err(|err| format!("{}", err))?
                .insert_systime(&subsystem, &param, &value)
                .map_err(|err| format!("{}", err))?;
        }

        Ok(())
    }
}