recorder/ot-recorder.c

1018 lines
26 KiB
C
Raw Normal View History

2015-09-01 08:19:52 -07:00
/*
* Copyright (C) 2015 Jan-Piet Mens <jpmens@gmail.com> and OwnTracks
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* JAN-PIET MENS OR OWNTRACKS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
* IN THE SOFTWARE.
*/
2015-08-14 09:40:35 -07:00
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>
#include <mosquitto.h>
#include <getopt.h>
#include <time.h>
#include "json.h"
#include <sys/utsname.h>
#include "utstring.h"
#include "utarray.h"
#include "geo.h"
2015-09-01 04:31:49 -07:00
#include "geohash.h"
2015-08-14 09:40:35 -07:00
#include "base64.h"
2015-08-14 23:14:34 -07:00
#include "misc.h"
2015-08-22 06:53:11 -07:00
#include "util.h"
2015-08-27 14:42:18 -07:00
#include "storage.h"
2015-09-01 04:31:49 -07:00
#ifdef HAVE_LMDB
# include "gcache.h"
#endif
2015-08-27 14:42:18 -07:00
#ifdef HAVE_HTTP
2015-08-29 06:04:22 -07:00
# include "http.h"
2015-08-27 14:42:18 -07:00
#endif
2015-08-14 09:40:35 -07:00
#define SSL_VERIFY_PEER (1)
#define SSL_VERIFY_NONE (0)
#define TOPIC_PARTS (4) /* owntracks/user/device/info */
#define TOPIC_SUFFIX "info"
#define DEFAULT_QOS (2)
#define CLEAN_SESSION false
2015-08-14 09:40:35 -07:00
static int run = 1;
double number(JsonNode *j, char *element)
{
JsonNode *m;
double d;
2015-08-14 09:40:35 -07:00
if ((m = json_find_member(j, element)) != NULL) {
if (m->tag == JSON_NUMBER) {
return (m->number_);
} else if (m->tag == JSON_STRING) {
d = atof(m->string_);
/* Normalize to number */
json_remove_from_parent(m);
json_append_member(j, element, json_mknumber(d));
return (d);
2015-08-14 09:40:35 -07:00
}
}
return (-7.0L);
}
JsonNode *extract(struct udata *ud, char *payload, char *tid, char *t, double *lat, double *lon, long *tst)
{
JsonNode *json, *j;
*tid = *t = 0;
*lat = *lon = -1.0L;
if ((json = json_decode(payload)) == NULL)
return (NULL);
if (ud->skipdemo && (json_find_member(json, "_demo") != NULL)) {
json_delete(json);
return (NULL);
}
if ((j = json_find_member(json, "_type")) == NULL) {
json_delete(json);
return (NULL);
}
if ((j->tag != JSON_STRING) || (strcmp(j->string_, "location") != 0)) {
json_delete(json);
return (NULL);
}
if ((j = json_find_member(json, "tid")) != NULL) {
if (j->tag == JSON_STRING) {
// printf("I got: [%s]\n", m->string_);
strcpy(tid, j->string_);
}
}
if ((j = json_find_member(json, "t")) != NULL) {
if (j && j->tag == JSON_STRING) {
strcpy(t, j->string_);
}
}
/*
* Normalize tst, lat, lon to numbers, particularly for Greenwich
* which produces strings currently.
*/
2015-08-14 09:40:35 -07:00
*tst = time(NULL);
if ((j = json_find_member(json, "tst")) != NULL) {
if (j && j->tag == JSON_STRING) {
*tst = strtoul(j->string_, NULL, 10);
json_remove_from_parent(j);
json_append_member(json, "tst", json_mknumber(*tst));
2015-08-14 09:40:35 -07:00
} else {
*tst = (unsigned long)j->number_;
}
}
*lat = number(json, "lat");
*lon = number(json, "lon");
return (json);
}
static const char *ltime(time_t t) {
static char buf[] = "HH:MM:SS";
strftime(buf, sizeof(buf), "%T", localtime(&t));
return(buf);
}
/*
* Process info/ message containing a CARD. If the payload is a card, return TRUE.
*/
int do_info(void *userdata, UT_string *username, UT_string *device, char *payload)
2015-08-14 09:40:35 -07:00
{
struct udata *ud = (struct udata *)userdata;
JsonNode *json, *j;
static UT_string *name = NULL, *face = NULL;
FILE *fp;
char *img;
int rc = FALSE;
2015-08-14 09:40:35 -07:00
utstring_renew(name);
utstring_renew(face);
if ((json = json_decode(payload)) == NULL) {
2015-08-14 23:41:44 -07:00
fprintf(stderr, "Can't decode INFO payload for username=%s\n", utstring_body(username));
return (FALSE);
2015-08-14 09:40:35 -07:00
}
if (ud->skipdemo && (json_find_member(json, "_demo") != NULL)) {
goto cleanup;
}
if ((j = json_find_member(json, "_type")) == NULL) {
goto cleanup;
}
if ((j->tag != JSON_STRING) || (strcmp(j->string_, "card") != 0)) {
goto cleanup;
}
2015-09-01 04:31:49 -07:00
/* I know the payload is valid JSON: write card */
2015-08-14 09:40:35 -07:00
2015-09-01 04:31:49 -07:00
if ((fp = pathn("wb", "cards", username, NULL, "json")) != NULL) {
fprintf(fp, "%s\n", payload);
fclose(fp);
2015-08-14 09:40:35 -07:00
}
2015-09-01 04:31:49 -07:00
rc = TRUE;
2015-08-14 09:40:35 -07:00
if ((j = json_find_member(json, "name")) != NULL) {
if (j->tag == JSON_STRING) {
// printf("I got: [%s]\n", j->string_);
utstring_printf(name, "%s", j->string_);
}
}
if ((j = json_find_member(json, "face")) != NULL) {
if (j->tag == JSON_STRING) {
// printf("I got: [%s]\n", j->string_);
utstring_printf(face, "%s", j->string_);
}
}
fprintf(stderr, "* CARD: %s-%s %s\n", utstring_body(username), utstring_body(device), utstring_body(name));
2015-09-01 04:31:49 -07:00
#ifdef HAVE_REDIS /* TODO: LMDB? */
2015-08-14 09:40:35 -07:00
if (ud->useredis) {
redis_ping(&ud->redis);
r = redisCommand(ud->redis, "HMSET card:%s name %s face %s", utstring_body(username), utstring_body(name), utstring_body(face));
}
#endif
/* We have a base64-encoded "face". Decode it and store binary image */
if ((img = malloc(utstring_len(face))) != NULL) {
int imglen;
if ((imglen = base64_decode(utstring_body(face), img)) > 0) {
2015-09-01 04:31:49 -07:00
if ((fp = pathn("wb", "photos", username, NULL, "png")) != NULL) {
fwrite(img, sizeof(char), imglen, fp);
fclose(fp);
2015-08-14 09:40:35 -07:00
}
2015-09-01 04:31:49 -07:00
#ifdef HAVE_REDIS /* TODO: LMDB ? */
2015-08-14 09:40:35 -07:00
if (ud->useredis) {
/* Add photo (binary) to Redis as photo:username */
redis_ping(&ud->redis);
r = redisCommand(ud->redis, "SET photo:%s %b", utstring_body(username), img, imglen);
}
#endif
}
free(img);
}
cleanup:
json_delete(json);
return (rc);
2015-08-14 09:40:35 -07:00
}
void do_msg(void *userdata, UT_string *username, UT_string *device, char *payload)
{
struct udata *ud = (struct udata *)userdata;
JsonNode *json, *j;
FILE *fp;
if ((json = json_decode(payload)) == NULL) {
2015-08-14 23:41:44 -07:00
fprintf(stderr, "Can't decode MSG payload for username=%s, device=%s\n",
utstring_body(username), utstring_body(device));
2015-08-14 09:40:35 -07:00
return;
}
if (ud->skipdemo && (json_find_member(json, "_demo") != NULL)) {
goto cleanup;
}
if ((j = json_find_member(json, "_type")) == NULL) {
goto cleanup;
}
if ((j->tag != JSON_STRING) || (strcmp(j->string_, "msg") != 0)) {
goto cleanup;
}
2015-09-01 04:31:49 -07:00
/* I know the payload is valid JSON: write message */
2015-08-14 09:40:35 -07:00
2015-09-01 04:31:49 -07:00
if ((fp = pathn("ab", "msg", username, NULL, "json")) != NULL) {
fprintf(fp, "%s\n", payload);
fclose(fp);
2015-08-14 09:40:35 -07:00
}
fprintf(stderr, "* MSG: %s-%s\n", utstring_body(username), utstring_body(device));
cleanup:
json_delete(json);
}
void republish(struct mosquitto *mosq, struct udata *userdata, char *username, char *topic, double lat, double lon, char *cc, char *addr, long tst, char *t)
{
struct udata *ud = (struct udata *)userdata;
JsonNode *json;
static UT_string *newtopic = NULL;
char *payload;
if (ud->pubprefix == NULL)
return;
if ((json = json_mkobject()) == NULL) {
return;
}
utstring_renew(newtopic);
utstring_printf(newtopic, "%s/%s", ud->pubprefix, topic);
json_append_member(json, "username", json_mkstring(username));
json_append_member(json, "topic", json_mkstring(topic));
json_append_member(json, "cc", json_mkstring(cc));
json_append_member(json, "addr", json_mkstring(addr));
json_append_member(json, "t", json_mkstring(t));
json_append_member(json, "tst", json_mknumber(tst));
json_append_member(json, "lat", json_mknumber(lat));
json_append_member(json, "lon", json_mknumber(lon));
if ((payload = json_stringify(json, NULL)) != NULL) {
mosquitto_publish(mosq, NULL, utstring_body(newtopic),
strlen(payload), payload, 1, true);
fprintf(stderr, "%s %s\n", utstring_body(newtopic), payload);
free(payload);
}
json_delete(json);
}
/*
* Decode OwnTracks CSV and return a new JsonNode to a JSON object.
*/
#define MILL 1000000.0
JsonNode *csv(char *payload, char *tid, char *t, double *lat, double *lon, long *tst)
{
JsonNode *json = NULL;
double dist = 0;
char tmptst[40];
double vel, trip, alt, cog;
if (sscanf(payload, "%[^,],%[^,],%[^,],%lf,%lf,%lf,%lf,%lf,%lf,%lf", tid, tmptst, t, lat, lon, &cog, &vel, &alt, &dist, &trip) != 10) {
2015-08-16 08:38:11 -07:00
// fprintf(stderr, "**** payload not CSV: %s\n", payload);
2015-08-14 09:40:35 -07:00
return (NULL);
}
*lat /= MILL;
*lon /= MILL;
cog *= 10;
alt *= 10;
trip *= 1000;
*tst = strtoul(tmptst, NULL, 16);
json = json_mkobject();
json_append_member(json, "_type", json_mkstring("location"));
json_append_member(json, "t", json_mkstring(t));
json_append_member(json, "tid", json_mkstring(tid));
2015-08-16 02:24:37 -07:00
json_append_member(json, "tst", json_mknumber(*tst));
2015-08-14 09:40:35 -07:00
json_append_member(json, "lat", json_mknumber(*lat));
json_append_member(json, "lon", json_mknumber(*lon));
json_append_member(json, "cog", json_mknumber(cog));
json_append_member(json, "vel", json_mknumber(vel));
json_append_member(json, "alt", json_mknumber(alt));
json_append_member(json, "dist", json_mknumber(dist));
json_append_member(json, "trip", json_mknumber(trip));
json_append_member(json, "csv", json_mkbool(1));
return (json);
}
#define RECFORMAT "%s\t%-18s\t%s\n"
void on_message(struct mosquitto *mosq, void *userdata, const struct mosquitto_message *m)
{
JsonNode *json;
char tid[BUFSIZ], t[BUFSIZ], *p;
double lat, lon;
long tst;
struct udata *ud = (struct udata *)userdata;
FILE *fp;
char **topics;
int count = 0, cached;
static UT_string *basetopic = NULL, *username = NULL, *device = NULL, *addr = NULL, *cc = NULL, *ghash = NULL, *ts = NULL;
2015-08-15 06:19:27 -07:00
static UT_string *reltopic = NULL;
2015-08-14 09:40:35 -07:00
char *jsonstring;
time_t now;
int pingping = FALSE;
2015-08-14 09:40:35 -07:00
/*
* mosquitto_message->
* int mid;
* char *topic;
* void *payload;
* int payloadlen;
* int qos;
* bool retain;
*/
time(&now);
monitorhook(ud, now, m->topic);
2015-08-14 09:40:35 -07:00
if (m->payloadlen == 0) {
return;
}
if (m->retain == TRUE && ud->ignoreretained) {
return;
}
2015-08-16 08:38:11 -07:00
// printf("%s %s\n", m->topic, bindump(m->payload, m->payloadlen)); fflush(stdout);
2015-08-14 09:40:35 -07:00
utstring_renew(ts);
utstring_renew(basetopic);
utstring_renew(username);
utstring_renew(device);
if (mosquitto_sub_topic_tokenise(m->topic, &topics, &count) != MOSQ_ERR_SUCCESS) {
return;
}
2015-08-15 01:42:55 -07:00
2015-08-14 09:40:35 -07:00
/* FIXME: handle null leading topic `/` */
utstring_printf(basetopic, "%s/%s/%s", topics[0], topics[1], topics[2]);
utstring_printf(username, "%s", topics[1]);
utstring_printf(device, "%s", topics[2]);
#ifdef HAVE_PING
if (!strcmp(utstring_body(username), "ping") && !strcmp(utstring_body(device), "ping")) {
pingping = TRUE;
}
#endif
2015-08-14 09:40:35 -07:00
if ((count == TOPIC_PARTS) && (strcmp(topics[count-1], TOPIC_SUFFIX) == 0)) {
if (do_info(ud, username, device, m->payload) == TRUE) { /* this was a card */
return;
}
2015-08-14 09:40:35 -07:00
}
/* owntracks/user/device/msg */
if ((count == TOPIC_PARTS) && (strcmp(topics[count-1], "msg") == 0)) {
do_msg(ud, username, device, m->payload);
2015-08-14 09:40:35 -07:00
}
2015-08-15 06:19:27 -07:00
/*
* Determine "relative topic", relative to base, i.e. whatever comes behind
* ownntracks/user/device/
*/
utstring_renew(reltopic);
2015-08-14 09:40:35 -07:00
if (count != 3) {
/*
* Not a normal location publish. Build up a string consisting of the remaining
* topic parts, i.e. whatever is after base topic, and record with whatever
* (hopefully non-binary) payload we got.
*/
int j;
for (j = 3; j < count; j++) {
2015-08-15 06:19:27 -07:00
utstring_printf(reltopic, "%s%c", topics[j], (j < count - 1) ? '/' : ' ');
2015-08-14 09:40:35 -07:00
}
2015-09-01 04:31:49 -07:00
if ((fp = pathn("a", "rec", username, device, "rec")) != NULL) {
2015-08-14 09:40:35 -07:00
2015-09-01 04:31:49 -07:00
fprintf(fp, RECFORMAT, isotime(now), utstring_body(reltopic), bindump(m->payload, m->payloadlen));
fclose(fp);
2015-08-14 09:40:35 -07:00
}
mosquitto_sub_topic_tokens_free(&topics, count);
return;
}
2015-08-15 07:26:41 -07:00
if (utstring_len(reltopic) == 0)
utstring_printf(reltopic, "-");
2015-08-14 09:40:35 -07:00
mosquitto_sub_topic_tokens_free(&topics, count);
2015-08-15 00:09:09 -07:00
/*
* Try to decode JSON payload to find _type: location. If that doesn't work,
* see if it's OwnTracks' Greenwich CSV
*/
2015-08-14 09:40:35 -07:00
json = extract(ud, m->payload, tid, t, &lat, &lon, &tst);
if (json == NULL) {
/* Is it OwnTracks Greenwich CSV? */
if ((json = csv(m->payload, tid, t, &lat, &lon, &tst)) == NULL) {
/* It's not JSON or it's not a location CSV; store it */
2015-08-16 08:38:11 -07:00
/* It may be an lwt */
if ((fp = pathn("a", "rec", username, device, "rec")) != NULL) {
fprintf(fp, RECFORMAT, isotime(now),
2015-08-15 06:19:27 -07:00
utstring_body(reltopic),
bindump(m->payload, m->payloadlen));
fclose(fp);
}
2015-08-14 09:40:35 -07:00
return;
}
// fprintf(stderr, "+++++ %s\n", json_stringify(json, NULL));
2015-08-14 09:40:35 -07:00
}
#if 0
if (*t && (!strcmp(t, "p") || !strcmp(t, "b"))) {
// fprintf(stderr, "Ignore `t:%s' for %s\n", t, m->topic);
json_delete(json);
return;
}
#endif
2015-08-15 00:09:09 -07:00
/*
* We are now processing a _type location.
*/
2015-08-14 09:40:35 -07:00
utstring_renew(ghash);
p = geohash_encode(lat, lon, geohash_prec());
2015-08-14 09:40:35 -07:00
if (p != NULL) {
utstring_printf(ghash, "%s", p);
free(p);
}
utstring_renew(addr);
utstring_renew(cc);
cached = FALSE;
if (ud->revgeo == TRUE) {
2015-09-01 04:31:49 -07:00
JsonNode *geo, *j;
2015-08-14 09:40:35 -07:00
2015-09-01 04:31:49 -07:00
if ((geo = gcache_json_get(ud->gc, utstring_body(ghash))) != NULL) {
/* Habemus cached data */
cached = TRUE;
2015-08-14 09:40:35 -07:00
2015-09-01 04:31:49 -07:00
if ((j = json_find_member(geo, "cc")) != NULL) {
utstring_printf(cc, "%s", j->string_);
}
if ((j = json_find_member(geo, "addr")) != NULL) {
utstring_printf(addr, "%s", j->string_);
}
} else {
if ((geo = revgeo(lat, lon, addr, cc)) != NULL) {
2015-09-01 04:31:49 -07:00
gcache_json_put(ud->gc, utstring_body(ghash), geo);
2015-08-21 10:31:26 -07:00
} else {
/* We didn't obtain reverse Geo, maybe because of over
* quota; make a note of the missing geohash */
char gfile[BUFSIZ];
FILE *fp;
snprintf(gfile, BUFSIZ, "%s/ghash/missing", STORAGEDIR);
if ((fp = fopen(gfile, "a")) != NULL) {
fprintf(fp, "%s %lf %lf\n", utstring_body(ghash), lat, lon);
fclose(fp);
}
}
2015-08-14 09:40:35 -07:00
}
} else {
utstring_printf(cc, "??");
utstring_printf(addr, "n.a.");
2015-08-14 09:40:35 -07:00
}
/*
* We have exactly three topic parts (owntracks/user/device), and valid JSON.
*/
2015-09-08 03:57:09 -07:00
#ifdef HAVE_HTTP
if (ud->mgserver && !pingping) {
2015-09-08 03:57:09 -07:00
/*
* Create a new location object containing all the bits and
* pieces we need and push that into connected Websockets.
* TODO: clean up
*/
JsonNode *geo, *wso = json_mkobject();
json_copy_to_object(wso, json, TRUE);
if ((geo = gcache_json_get(ud->gc, utstring_body(ghash))) != NULL) {
json_copy_to_object(wso, geo, FALSE);
json_delete(geo);
}
/*
* I need a unique "key" in the Websocket clients to keep track
* of which device is being updated; use topic.
*/
json_append_member(wso, "topic", json_mkstring(m->topic));
/*
* We have to know which user/device this is for in order to
* determine whether a connected Websocket client is authorized
* to see this. Add user/device
*/
json_append_member(wso, "user", json_mkstring(utstring_body(username)));
json_append_member(wso, "device", json_mkstring(utstring_body(device)));
http_ws_push_json(ud->mgserver, wso);
2015-09-08 03:57:09 -07:00
json_delete(wso);
}
#endif
2015-08-14 09:40:35 -07:00
if ((jsonstring = json_stringify(json, NULL)) != NULL) {
2015-09-01 04:31:49 -07:00
char *js;
2015-08-14 09:40:35 -07:00
#ifdef HAVE_REDIS /* TODO: shall we store last positions in LMDB? */
2015-08-14 09:40:35 -07:00
if (ud->useredis) {
/* add last to Redis as "lastpos:username-device" */
last_storeredis(&ud->redis, utstring_body(username), utstring_body(device), jsonstring);
}
#endif
if (!pingping) {
if ((fp = pathn("a", "rec", username, device, "rec")) != NULL) {
fprintf(fp, RECFORMAT, isotime(now), "*", jsonstring);
fclose(fp);
}
2015-09-01 04:31:49 -07:00
}
2015-09-01 04:31:49 -07:00
/* Keep track of original username & device name in LAST. */
json_append_member(json, "username", json_mkstring(utstring_body(username)));
json_append_member(json, "device", json_mkstring(utstring_body(device)));
json_append_member(json, "topic", json_mkstring(m->topic));
json_append_member(json, "ghash", json_mkstring(utstring_body(ghash)));
2015-09-01 04:31:49 -07:00
if ((js = json_stringify(json, NULL)) != NULL) {
/* Now safewrite the last location */
utstring_printf(ts, "%s/last/%s/%s",
STORAGEDIR, utstring_body(username), utstring_body(device));
if (mkpath(utstring_body(ts)) < 0) {
perror(utstring_body(ts));
}
2015-09-01 04:31:49 -07:00
utstring_printf(ts, "/%s-%s.json",
utstring_body(username), utstring_body(device));
safewrite(utstring_body(ts), js);
free(js);
2015-08-14 09:40:35 -07:00
}
free(jsonstring);
}
/* publish */
2015-08-19 07:55:46 -07:00
// republish(mosq, ud, utstring_body(username), m->topic, lat, lon, utstring_body(cc), utstring_body(addr), tst, t);
2015-08-14 09:40:35 -07:00
2015-08-16 08:38:11 -07:00
if (*t == 0) {
strcpy(t, " ");
}
2015-08-17 09:54:49 -07:00
fprintf(stderr, "%c %s %-35s t=%-1.1s tid=%-2.2s loc=%.5f,%.5f [%s] %s (%s)\n",
2015-08-14 09:40:35 -07:00
(cached) ? '*' : '-',
2015-08-16 08:38:11 -07:00
ltime(tst),
m->topic,
t,
tid,
lat, lon,
utstring_body(cc),
2015-08-17 09:54:49 -07:00
utstring_body(addr),
utstring_body(ghash)
2015-08-16 08:38:11 -07:00
);
2015-08-14 09:40:35 -07:00
json_delete(json);
}
void on_connect(struct mosquitto *mosq, void *userdata, int rc)
{
struct udata *ud = (struct udata *)userdata;
int mid;
char **m = NULL;
while ((m = (char **)utarray_next(ud->topics, m))) {
2015-09-03 23:06:53 -07:00
olog(LOG_DEBUG, "Subscribing to %s (qos=%d)", *m, ud->qos);
mosquitto_subscribe(mosq, &mid, *m, ud->qos);
2015-08-14 09:40:35 -07:00
}
}
void on_disconnect(struct mosquitto *mosq, void *userdata, int reason)
{
2015-09-01 04:31:49 -07:00
#ifdef HAVE_LMDB
2015-08-14 09:40:35 -07:00
struct udata *ud = (struct udata *)userdata;
#endif
2015-09-03 23:06:53 -07:00
olog(LOG_INFO, "Disconnected. Reason: %d [%s]", reason, mosquitto_strerror(reason));
2015-08-14 09:40:35 -07:00
if (reason == 0) { // client wish
2015-09-01 04:31:49 -07:00
#ifdef HAVE_LMDB
gcache_close(ud->gc);
#endif
2015-08-14 09:40:35 -07:00
}
}
static void catcher(int sig)
{
fprintf(stderr, "Going down on signal %d\n", sig);
exit(1);
}
void usage(char *prog)
{
2015-08-26 12:32:12 -07:00
printf("Usage: %s [options..] topic [topic ...]\n", prog);
printf(" --help -h this message\n");
printf(" --storage -S storage dir (%s)\n", STORAGEDEFAULT);
2015-08-26 12:32:12 -07:00
printf(" --norevgeo -G disable ghash to reverge-geo lookups\n");
printf(" --skipdemo -D do not handle objects with _demo\n");
printf(" --useretained -R process retained messages (default: no)\n");
printf(" --clientid -i MQTT client-ID\n");
printf(" --qos -q MQTT QoS (dflt: 2)\n");
printf(" --pubprefix -P republish prefix (dflt: no republish)\n");
printf(" --host -H MQTT host (localhost)\n");
printf(" --port -p MQTT port (1883)\n");
printf(" --logfacility syslog facility (local0)\n");
2015-08-27 14:42:18 -07:00
#ifdef HAVE_HTTP
printf(" --http-host <host> HTTP addr to bind to (localhost)\n");
2015-08-27 14:42:18 -07:00
printf(" --http-port <port> -A HTTP port (8083)\n");
printf(" --doc-root <directory> document root (./wdocs)\n");
#endif
printf(" --precision ghash precision (dflt: %d)\n", GHASHPREC);
printf("\n");
printf("Options override these environment variables:\n");
printf(" $OTR_HOST MQTT hostname\n");
printf(" $OTR_PORT MQTT port\n");
printf(" $OTR_STORAGEDIR\n");
printf(" $OTR_USER\n");
printf(" $OTR_PASS\n");
printf(" $OTR_CAFILE PEM CA certificate chain\n");
2015-08-26 12:32:12 -07:00
exit(1);
2015-08-14 09:40:35 -07:00
}
2015-08-26 12:32:12 -07:00
2015-08-14 09:40:35 -07:00
int main(int argc, char **argv)
{
struct mosquitto *mosq = NULL;
char err[1024], *p, *username, *password, *cafile;
char *hostname = "localhost", *logfacility = "local0";
2015-08-14 09:40:35 -07:00
int port = 1883;
int rc, i, ch;
static struct udata udata, *ud = &udata;
struct utsname uts;
UT_string *clientid;
2015-08-27 14:42:18 -07:00
#ifdef HAVE_HTTP
int http_port = 8083;
char *doc_root = "./wdocs";
char *http_host = "localhost";
2015-08-14 09:40:35 -07:00
#endif
char *progname = *argv;
udata.qos = DEFAULT_QOS;
2015-08-14 09:40:35 -07:00
udata.ignoreretained = TRUE;
udata.pubprefix = NULL;
udata.skipdemo = TRUE;
2015-08-19 08:39:00 -07:00
udata.revgeo = TRUE;
2015-09-01 04:31:49 -07:00
#ifdef HAVE_LMDB
udata.gc = NULL;
#endif
2015-08-27 14:42:18 -07:00
#ifdef HAVE_HTTP
udata.mgserver = NULL;
2015-08-27 14:42:18 -07:00
#endif
2015-08-14 09:40:35 -07:00
2015-08-26 12:32:12 -07:00
if ((p = getenv("OTR_HOST")) != NULL) {
hostname = strdup(p);
}
if ((p = getenv("OTR_PORT")) != NULL) {
port = atoi(p);
}
if ((p = getenv("OTR_STORAGEDIR")) != NULL) {
strcpy(STORAGEDIR, p);
}
utstring_new(clientid);
utstring_printf(clientid, "ot-recorder");
if (uname(&uts) == 0) {
utstring_printf(clientid, "-%s", uts.nodename);
}
utstring_printf(clientid, "-%d", getpid());
2015-08-26 12:32:12 -07:00
while (1) {
static struct option long_options[] = {
{ "help", no_argument, 0, 'h'},
{ "skipdemo", no_argument, 0, 'D'},
{ "norevgeo", no_argument, 0, 'G'},
{ "useretained", no_argument, 0, 'R'},
{ "clientid", required_argument, 0, 'i'},
{ "pubprefix", required_argument, 0, 'P'},
{ "qos", required_argument, 0, 'q'},
{ "host", required_argument, 0, 'H'},
{ "port", required_argument, 0, 'p'},
{ "storage", required_argument, 0, 'S'},
{ "logfacility", required_argument, 0, 4},
{ "precision", required_argument, 0, 5},
2015-08-27 14:42:18 -07:00
#ifdef HAVE_HTTP
{ "http-host", required_argument, 0, 3},
2015-08-27 14:42:18 -07:00
{ "http-port", required_argument, 0, 'A'},
{ "doc-root", required_argument, 0, 2},
#endif
2015-08-26 12:32:12 -07:00
{0, 0, 0, 0}
};
int optindex = 0;
2015-09-01 04:31:49 -07:00
ch = getopt_long(argc, argv, "hDGRi:P:q:S:H:p:A:", long_options, &optindex);
2015-08-26 12:32:12 -07:00
if (ch == -1)
break;
2015-08-14 09:40:35 -07:00
switch (ch) {
case 5:
geohash_setprec(atoi(optarg));
break;
case 4:
logfacility = strdup(optarg);
break;
2015-08-27 14:42:18 -07:00
#ifdef HAVE_HTTP
case 'A': /* API */
http_port = atoi(optarg);
break;
case 2: /* no short char */
doc_root = strdup(optarg);
break;
case 3: /* no short char */
http_host = strdup(optarg);
break;
2015-08-27 14:42:18 -07:00
#endif
2015-08-14 09:40:35 -07:00
case 'D':
ud->skipdemo = FALSE;
break;
case 'G':
ud->revgeo = FALSE;
break;
case 'i':
utstring_clear(clientid);
utstring_printf(clientid, "%s", optarg);
break;
2015-08-14 09:40:35 -07:00
case 'P':
2015-08-26 12:32:12 -07:00
udata.pubprefix = strdup(optarg); /* TODO: do we want this? */
2015-08-14 09:40:35 -07:00
break;
case 'q':
ud->qos = atoi(optarg);
if (ud->qos < 0 || ud->qos > 2) {
fprintf(stderr, "%s: illegal qos\n", progname);
exit(2);
}
break;
2015-08-14 09:40:35 -07:00
case 'R':
ud->ignoreretained = FALSE;
break;
2015-08-26 12:32:12 -07:00
case 'H':
hostname = strdup(optarg);
break;
case 'p':
port = atoi(optarg);
2015-08-14 09:40:35 -07:00
break;
2015-08-26 12:32:12 -07:00
case 'S':
strcpy(STORAGEDIR, optarg);
break;
case 'h':
usage(progname);
exit(0);
default:
abort();
2015-08-14 09:40:35 -07:00
}
2015-08-26 12:32:12 -07:00
2015-08-14 09:40:35 -07:00
}
2015-08-26 12:32:12 -07:00
argc -= (optind);
argv += (optind);
2015-08-14 09:40:35 -07:00
2015-08-26 12:32:12 -07:00
if (argc < 1) {
2015-08-14 09:40:35 -07:00
usage(progname);
return (-1);
}
openlog("ot-recorder", LOG_PID | LOG_PERROR, syslog_facility_code(logfacility));
2015-09-01 05:08:28 -07:00
#ifdef HAVE_HTTP
if (http_port) {
if (!is_directory(doc_root)) {
2015-09-03 23:06:53 -07:00
olog(LOG_ERR, "%s is not a directory", doc_root);
exit(1);
}
/* First arg is user data which I can grab via conn->server_param */
udata.mgserver = mg_create_server(ud, ev_handler);
2015-09-01 05:08:28 -07:00
}
#endif
2015-09-03 23:06:53 -07:00
olog(LOG_DEBUG, "starting");
if (ud->revgeo == TRUE) {
2015-09-01 04:31:49 -07:00
#ifdef HAVE_LMDB
char db_filename[BUFSIZ], *pa;
snprintf(db_filename, BUFSIZ, "%s/ghash", STORAGEDIR);
pa = strdup(db_filename);
mkpath(pa);
free(pa);
2015-09-07 05:08:09 -07:00
udata.gc = gcache_open(db_filename, NULL, FALSE);
2015-09-01 04:31:49 -07:00
if (udata.gc == NULL) {
2015-09-03 23:06:53 -07:00
olog(LOG_ERR, "Can't initialize gcache in %s", db_filename);
2015-09-01 04:31:49 -07:00
exit(1);
}
storage_init(ud->revgeo); /* For the HTTP server */
2015-09-01 04:31:49 -07:00
#endif
revgeo_init();
}
2015-08-14 09:40:35 -07:00
mosquitto_lib_init();
signal(SIGINT, catcher);
mosq = mosquitto_new(utstring_body(clientid), CLEAN_SESSION, (void *)&udata);
2015-08-14 09:40:35 -07:00
if (!mosq) {
fprintf(stderr, "Error: Out of memory.\n");
mosquitto_lib_cleanup();
return 1;
}
/*
* Pushing list of topics into the array so that we can (re)subscribe on_connect()
*/
utarray_new(ud->topics, &ut_str_icd);
2015-08-26 12:32:12 -07:00
for (i = 0; i < argc; i++) {
2015-08-14 09:40:35 -07:00
utarray_push_back(ud->topics, &argv[i]);
}
mosquitto_reconnect_delay_set(mosq,
2, /* delay */
20, /* delay_max */
0); /* exponential backoff */
mosquitto_message_callback_set(mosq, on_message);
mosquitto_connect_callback_set(mosq, on_connect);
mosquitto_disconnect_callback_set(mosq, on_disconnect);
if ((username = getenv("OTR_USER")) != NULL) {
if ((password = getenv("OTR_PASS")) != NULL) {
mosquitto_username_pw_set(mosq, username, password);
}
}
cafile = getenv("OTR_CAFILE");
if (cafile && *cafile) {
rc = mosquitto_tls_set(mosq,
cafile, /* cafile */
NULL, /* capath */
NULL, /* certfile */
NULL, /* keyfile */
NULL /* pw_callback() */
);
if (rc != MOSQ_ERR_SUCCESS) {
fprintf(stderr, "Cannot set TLS CA: %s (check path names)\n",
mosquitto_strerror(rc));
exit(3);
}
mosquitto_tls_opts_set(mosq,
SSL_VERIFY_PEER,
NULL, /* tls_version: "tlsv1.2", "tlsv1" */
NULL /* ciphers */
);
}
rc = mosquitto_connect(mosq, hostname, port, 60);
if (rc) {
if (rc == MOSQ_ERR_ERRNO) {
strerror_r(errno, err, 1024);
fprintf(stderr, "Error: %s\n", err);
} else {
fprintf(stderr, "Unable to connect (%d) [%s].\n", rc, mosquitto_strerror(rc));
}
mosquitto_lib_cleanup();
return rc;
}
2015-08-27 14:42:18 -07:00
#ifdef HAVE_HTTP
if (http_port) {
char address[BUFSIZ];
sprintf(address, "%s:%d", http_host, http_port);
mg_set_option(udata.mgserver, "listening_port", address);
// mg_set_option(udata.mgserver, "listening_port", "8090,ssl://8091:cert.pem");
2015-08-27 14:42:18 -07:00
// mg_set_option(udata.mgserver, "ssl_certificate", "cert.pem");
// mg_set_option(udata.mgserver, "listening_port", "8091");
2015-08-27 14:42:18 -07:00
mg_set_option(udata.mgserver, "document_root", doc_root);
mg_set_option(udata.mgserver, "enable_directory_listing", "yes");
// mg_set_option(udata.mgserver, "access_log_file", "access.log");
// mg_set_option(udata.mgserver, "cgi_pattern", "**.cgi");
2015-08-27 14:42:18 -07:00
2015-09-03 23:06:53 -07:00
olog(LOG_INFO, "HTTP listener started on %s", mg_get_option(udata.mgserver, "listening_port"));
}
2015-08-27 14:42:18 -07:00
#endif
2015-08-14 09:40:35 -07:00
while (run) {
2015-08-27 14:42:18 -07:00
rc = mosquitto_loop(mosq, /* timeout */ 200, /* max-packets */ 1);
if (run && rc) {
2015-09-03 23:06:53 -07:00
olog(LOG_INFO, "MQTT connection: rc=%d [%s]. Sleeping...", rc, mosquitto_strerror(rc));
sleep(10);
mosquitto_reconnect(mosq);
}
2015-08-27 14:42:18 -07:00
#ifdef HAVE_HTTP
if (udata.mgserver) {
mg_poll_server(udata.mgserver, 50);
}
2015-08-27 14:42:18 -07:00
#endif
2015-08-14 09:40:35 -07:00
}
#ifdef HAVE_HTTP
mg_destroy_server(&udata.mgserver);
#endif
2015-08-14 09:40:35 -07:00
mosquitto_disconnect(mosq);
mosquitto_destroy(mosq);
mosquitto_lib_cleanup();
return (0);
}