forked from flightaware/dump978
-
Notifications
You must be signed in to change notification settings - Fork 0
/
skyaware978_main.cc
138 lines (112 loc) · 4.63 KB
/
skyaware978_main.cc
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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
// Copyright (c) 2019, FlightAware LLC.
// All rights reserved.
// Licensed under the 2-clause BSD license; see the LICENSE file
#include <boost/exception/diagnostic_information.hpp>
#include <boost/program_options.hpp>
#include <boost/regex.hpp>
#include <iostream>
#include <memory>
#include "message_source.h"
#include "skyaware_writer.h"
#include "socket_input.h"
#include "uat_message.h"
using namespace flightaware::uat;
using namespace flightaware::skyaware;
namespace po = boost::program_options;
using boost::asio::ip::tcp;
struct connect_option {
std::string host;
std::string port;
};
// Specializations of validate for --connect
void validate(boost::any &v, const std::vector<std::string> &values, connect_option *target_type, int) {
po::validators::check_first_occurrence(v);
const std::string &s = po::validators::get_single_string(values);
static const boost::regex r("(?:([^:]+):)?(\\d+)");
boost::smatch match;
if (boost::regex_match(s, match, r)) {
v = boost::any(connect_option{match[1], match[2]});
} else {
throw po::validation_error(po::validation_error::invalid_option_value);
}
}
#define EXIT_NO_RESTART (64)
static int realmain(int argc, char **argv) {
boost::asio::io_service io_service;
// clang-format off
po::options_description desc("Allowed options");
desc.add_options()
("help", "produce help message")
("version", "show version")
("connect", po::value<connect_option>(), "connect to host:port for raw UAT data")
("reconnect-interval", po::value<unsigned>()->default_value(0), "on connection failure, attempt to reconnect after this interval (seconds); 0 disables")
("json-dir", po::value<std::string>(), "write json files to given directory")
("history-count", po::value<unsigned>()->default_value(120), "number of history files to maintain")
("history-interval", po::value<unsigned>()->default_value(30), "interval between history files (seconds)")
("lat", po::value<double>(), "latitude of receiver")
("lon", po::value<double>(), "longitude of receiver");
// clang-format on
po::variables_map opts;
try {
po::store(po::parse_command_line(argc, argv, desc), opts);
po::notify(opts);
} catch (boost::program_options::error &err) {
std::cerr << err.what() << std::endl;
std::cerr << desc << std::endl;
return EXIT_NO_RESTART;
}
if (opts.count("help")) {
std::cerr << "skyaware978 " << VERSION << std::endl;
std::cerr << desc << std::endl;
return EXIT_NO_RESTART;
}
if (opts.count("version")) {
std::cerr << "skyaware978 " << VERSION << std::endl;
return EXIT_NO_RESTART;
}
if (!opts.count("connect")) {
std::cerr << "--connect option is required" << std::endl;
return EXIT_NO_RESTART;
}
if (!opts.count("json-dir")) {
std::cerr << "--json-dir option is required" << std::endl;
return EXIT_NO_RESTART;
}
auto connect = opts["connect"].as<connect_option>();
auto reconnect_interval = opts["reconnect-interval"].as<unsigned>();
auto input = RawInput::Create(io_service, connect.host, connect.port, std::chrono::milliseconds(reconnect_interval * 1000));
auto tracker = Tracker::Create(io_service);
input->SetConsumer(std::bind(&Tracker::HandleMessages, tracker, std::placeholders::_1));
input->SetErrorHandler([&io_service, reconnect_interval](const boost::system::error_code &ec) {
std::cerr << "Connection failed: " << ec.message() << std::endl;
if (!reconnect_interval) {
io_service.stop();
}
});
boost::optional<std::pair<double, double>> location = boost::make_optional(false, std::make_pair(0.0, 0.0));
if (opts.count("lat") && opts.count("lon")) {
location.emplace(opts["lat"].as<double>(), opts["lon"].as<double>());
}
auto dir = opts["json-dir"].as<std::string>();
auto writer = SkyAwareWriter::Create(io_service, tracker, dir, std::chrono::milliseconds(1000), opts["history-count"].as<unsigned>(), std::chrono::milliseconds(opts["history-interval"].as<unsigned>() * 1000), location);
writer->Start();
tracker->Start();
input->Start();
io_service.run();
input->Stop();
tracker->Stop();
writer->Stop();
return 1; // connection loss is abnormal
}
int main(int argc, char **argv) {
#ifndef DEBUG_EXCEPTIONS
try {
return realmain(argc, argv);
} catch (...) {
std::cerr << "Uncaught exception: " << boost::current_exception_diagnostic_information() << std::endl;
return 2;
}
#else
return realmain(argc, argv);
#endif
}