/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/

/***
  This file is part of systemd.

  Copyright 2013 Lennart Poettering

  systemd is free software; you can redistribute it and/or modify it
  under the terms of the GNU Lesser General Public License as published by
  the Free Software Foundation; either version 2.1 of the License, or
  (at your option) any later version.

  systemd is distributed in the hope that it will be useful, but
  WITHOUT ANY WARRANTY; without even the implied warranty of
  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  Lesser General Public License for more details.

  You should have received a copy of the GNU Lesser General Public License
  along with systemd; If not, see <http://www.gnu.org/licenses/>.
***/

#include "bus-util.h"
#include "dbus-client-track.h"

static unsigned tracked_client_hash(const void *a) {
        const BusTrackedClient *x = a;

        return string_hash_func(x->name) ^ PTR_TO_UINT(x->bus);
}

static int tracked_client_compare(const void *a, const void *b) {
        const BusTrackedClient *x = a, *y = b;
        int r;

        r = strcmp(x->name, y->name);
        if (r != 0)
                return r;

        if (x->bus < y->bus)
                return -1;
        if (x->bus > y->bus)
                return 1;

        return 0;
}

static int on_name_owner_changed(sd_bus *bus, sd_bus_message *message, void *userdata, sd_bus_error *error) {
        BusTrackedClient *c = userdata;
        const char *name, *old, *new;
        int r;

        assert(bus);
        assert(message);

        r = sd_bus_message_read(message, "sss", &name, &old, &new);
        if (r < 0) {
                bus_log_parse_error(r);
                return r;
        }

        bus_client_untrack(c->set, bus, name);
        return 0;
}

static char *build_match(const char *name) {

        return strjoin("type='signal',"
                       "sender='org.freedesktop.DBus',"
                       "path='/org/freedesktop/DBus',"
                       "interface='org.freedesktop.DBus',"
                       "member='NameOwnerChanged',"
                       "arg0='", name, "'", NULL);
}

int bus_client_track(Set **s, sd_bus *bus, const char *name) {
        BusTrackedClient *c, *found;
        size_t l;
        int r;

        assert(s);
        assert(bus);

        r = set_ensure_allocated(s, tracked_client_hash, tracked_client_compare);
        if (r < 0)
                return r;

        name = strempty(name);

        l = strlen(name);

        c = alloca(offsetof(BusTrackedClient, name) + l + 1);
        c->set = *s;
        c->bus = bus;
        strcpy(c->name, name);

        found = set_get(*s, c);
        if (found)
                return 0;

        c = memdup(c, offsetof(BusTrackedClient, name) + l + 1);
        if (!c)
                return -ENOMEM;

        r = set_put(*s, c);
        if (r < 0) {
                free(c);
                return r;
        }

        if (!isempty(name)) {
                _cleanup_free_ char *match = NULL;

                match = build_match(name);
                if (!match) {
                        set_remove(*s, c);
                        free(c);
                        return -ENOMEM;
                }

                r = sd_bus_add_match(bus, match, on_name_owner_changed, c);
                if (r < 0) {
                        set_remove(*s, c);
                        free(c);
                        return r;
                }
        }

        sd_bus_ref(c->bus);
        return 1;
}

static void bus_client_free_one(Set *s, BusTrackedClient *c) {
        assert(s);
        assert(c);

        if (!isempty(c->name)) {
                _cleanup_free_ char *match = NULL;

                match = build_match(c->name);
                if (match)
                        sd_bus_remove_match(c->bus, match, on_name_owner_changed, c);
        }

        sd_bus_unref(c->bus);
        set_remove(s, c);
        free(c);
}

int bus_client_untrack(Set *s, sd_bus *bus, const char *name) {
        BusTrackedClient *c, *found;
        size_t l;

        assert(bus);
        assert(s);
        assert(name);

        name = strempty(name);

        l = strlen(name);

        c = alloca(offsetof(BusTrackedClient, name) + l + 1);
        c->bus = bus;
        strcpy(c->name, name);

        found = set_get(s, c);
        if (!found)
                return 0;

        bus_client_free_one(s, found);
        return 1;
}

void bus_client_track_free(Set *s) {
        BusTrackedClient *c;

        while ((c = set_first(s)))
                bus_client_free_one(s, c);

        set_free(s);
}

int bus_client_untrack_bus(Set *s, sd_bus *bus) {
        BusTrackedClient *c;
        Iterator i;
        int r = 0;

        SET_FOREACH(c, s, i)
                if (c->bus == bus) {
                        bus_client_free_one(s, c);
                        r++;
                }

        return r;
}

void bus_client_track_serialize(Manager *m, FILE *f, Set *s) {
        BusTrackedClient *c;
        Iterator i;

        assert(m);
        assert(f);

        SET_FOREACH(c, s, i) {
                if (c->bus == m->api_bus)
                        fprintf(f, "subscribed=%s\n", isempty(c->name) ? "*" : c->name);
                else
                        fprintf(f, "subscribed=%p %s\n", c->bus, isempty(c->name) ? "*" : c->name);
        }
}

int bus_client_track_deserialize_item(Manager *m, Set **s, const char *line) {
        const char *e, *q, *name;
        sd_bus *bus;
        void *p;
        int r;

        e = startswith(line, "subscribed=");
        if (!e)
                return 0;

        q = strpbrk(e, WHITESPACE);
        if (!q) {
                if (m->api_bus) {
                        bus = m->api_bus;
                        name = e;
                        goto finish;
                }

                return 1;
        }

        if (sscanf(e, "%p", &p) != 1) {
                log_debug("Failed to parse subscription pointer.");
                return -EINVAL;
        }

        bus = set_get(m->private_buses, p);
        if (!bus)
                return 1;

        name = q + strspn(q, WHITESPACE);

finish:
        r = bus_client_track(s, bus, streq(name, "*") ? NULL : name);
        if (r < 0) {
                log_debug("Failed to deserialize client subscription: %s", strerror(-r));
                return r;
        }

        return 1;
}