summaryrefslogtreecommitdiffstats
path: root/src/bin/cloudmqtt-client.rs
blob: 05e4db5e82eebd22da30a7dc12cfd6bdf9c14bc1 (plain)
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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
//
//   This Source Code Form is subject to the terms of the Mozilla Public
//   License, v. 2.0. If a copy of the MPL was not distributed with this
//   file, You can obtain one at http://mozilla.org/MPL/2.0/.
//

use clap::Parser;
use cloudmqtt::{packet_stream::Acknowledge, MqttClient, MqttConnectionParams};
use futures::StreamExt;
use mqtt_format::v3::{
    strings::MString, subscription_request::MSubscriptionRequest, will::MLastWill,
};
use tracing::error;
use tracing_subscriber::layer::SubscriberExt;
use tracing_subscriber::util::SubscriberInitExt;

#[derive(Parser, Debug)]
#[clap(author, version, about)]
struct Args {
    #[clap(long, value_parser)]
    addr: String,
    #[clap(long, value_parser)]
    client_id: String,

    #[clap(long, value_parser)]
    subscriptions: Vec<String>,
}

#[tokio::main]
async fn main() {
    let fmt_layer = tracing_subscriber::fmt::layer()
        .pretty()
        .with_timer(tracing_subscriber::fmt::time::uptime());

    let filter_layer = tracing_subscriber::EnvFilter::from_default_env();

    tracing_subscriber::registry()
        .with(fmt_layer)
        .with(filter_layer)
        .init();

    tracing::info!("Starting up");

    let args = Args::parse();

    let client = MqttClient::connect_v3_unsecured(
        &args.addr,
        MqttConnectionParams {
            clean_session: false,
            will: Some(MLastWill {
                topic: mqtt_format::v3::strings::MString {
                    value: "hello/world",
                },
                payload: b"I died!",
                qos: mqtt_format::v3::qos::MQualityOfService::AtMostOnce,
                retain: false,
            }),
            username: None,
            password: None,
            keep_alive: 5,
            client_id: mqtt_format::v3::strings::MString {
                value: &args.client_id,
            },
        },
    )
    .await
    .unwrap();

    tokio::spawn(client.hearbeat(None));

    client
        .subscribe(
            &args
                .subscriptions
                .iter()
                .map(|sub| MSubscriptionRequest {
                    topic: MString { value: sub },
                    qos: mqtt_format::v3::qos::MQualityOfService::AtLeastOnce,
                })
                .collect::<Vec<_>>(),
        )
        .await
        .unwrap();

    let packet_stream = client
        .build_packet_stream()
        .with_custom_ack_fn(|packet| async move {
            println!("ACKing packet {packet:?}");
            Acknowledge
        })
        .build();
    let mut packet_stream = Box::pin(packet_stream.stream());

    loop {
        let packet = match packet_stream.next().await {
            Some(Ok(packet)) => packet,
            None => {
                error!("Stream closed unexpectedly");
                break;
            }
            Some(Err(error)) => {
                error!(?error, "Stream errored");
                break;
            }
        };

        let packet = packet.get_packet().unwrap();
        println!("Received: {packet:#?}");
    }
}