leaseshelper.c 8.0 KB
Newer Older
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
/*
 * leaseshelper.c: Helper program to create custom leases file
 *
 * Copyright (C) 2014 Red Hat, Inc.
 * Copyright (C) 2014 Nehal J Wani
 *
 * This library 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.
 *
 * This library 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 this library.  If not, see
 * <http://www.gnu.org/licenses/>.
 *
 * For IPv6 support, use dnsmasq >= 2.67
 */

#include <config.h>


#include "virthread.h"
#include "virfile.h"
#include "virpidfile.h"
#include "virstring.h"
#include "virerror.h"
#include "viralloc.h"
#include "virjson.h"
34
#include "virlease.h"
35
#include "virenum.h"
36
#include "configmake.h"
37
#include "virgettext.h"
J
Ján Tomko 已提交
38
#include "virutil.h"
39 40 41 42 43 44 45 46 47 48 49 50

#define VIR_FROM_THIS VIR_FROM_NETWORK

static const char *program_name;

/* Display version information. */
static void
helperVersion(const char *argv0)
{
    printf("%s (%s) %s\n", argv0, PACKAGE_NAME, PACKAGE_VERSION);
}

51
G_GNUC_NORETURN static void
52 53 54 55 56
usage(int status)
{
    if (status) {
        fprintf(stderr, _("%s: try --help for more details\n"), program_name);
    } else {
57
        printf(_("Usage: %s add|old|del|init mac|clientid ip [hostname]\n"
58 59 60 61 62 63 64 65 66 67 68 69
                 "Designed for use with 'dnsmasq --dhcp-script'\n"
                 "Refer to man page of dnsmasq for more details'\n"),
               program_name);
    }
    exit(status);
}

/* Flags denoting actions for a lease */
enum virLeaseActionFlags {
    VIR_LEASE_ACTION_ADD,       /* Create new lease */
    VIR_LEASE_ACTION_OLD,       /* Lease already exists, renew it */
    VIR_LEASE_ACTION_DEL,       /* Delete the lease */
70
    VIR_LEASE_ACTION_INIT,      /* Tell dnsmasq of existing leases on restart */
71 72 73 74 75 76

    VIR_LEASE_ACTION_LAST
};

VIR_ENUM_DECL(virLeaseAction);

77 78
VIR_ENUM_IMPL(virLeaseAction,
              VIR_LEASE_ACTION_LAST,
79 80
              "add", "old", "del", "init",
);
81 82 83 84 85 86 87 88

int
main(int argc, char **argv)
{
    char *pid_file = NULL;
    char *custom_lease_file = NULL;
    const char *ip = NULL;
    const char *mac = NULL;
89
    const char *leases_str = NULL;
90 91 92 93
    const char *iaid = getenv("DNSMASQ_IAID");
    const char *clientid = getenv("DNSMASQ_CLIENT_ID");
    const char *interface = getenv("DNSMASQ_INTERFACE");
    const char *hostname = getenv("DNSMASQ_SUPPLIED_HOSTNAME");
94
    char *server_duid = NULL;
95 96 97 98 99 100 101 102 103 104 105 106
    int action = -1;
    int pid_file_fd = -1;
    int rv = EXIT_FAILURE;
    bool delete = false;
    virJSONValuePtr lease_new = NULL;
    virJSONValuePtr leases_array_new = NULL;

    virSetErrorFunc(NULL, NULL);
    virSetErrorLogPriorityFunc(NULL);

    program_name = argv[0];

107
    if (virGettextInitialize() < 0 ||
108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123
        virErrorInitialize() < 0) {
        fprintf(stderr, _("%s: initialization failed\n"), program_name);
        exit(EXIT_FAILURE);
    }

    /* Doesn't hurt to check */
    if (argc > 1) {
        if (STREQ(argv[1], "--help"))
            usage(EXIT_SUCCESS);

        if (STREQ(argv[1], "--version")) {
            helperVersion(argv[0]);
            exit(EXIT_SUCCESS);
        }
    }

124
    if (argc != 4 && argc != 5 && argc != 2) {
125 126 127 128 129
        /* Refer man page of dnsmasq --dhcp-script for more details */
        usage(EXIT_FAILURE);
    }

    /* Make sure dnsmasq knows the interface. The interface name is not known
130 131 132 133
     * via env variable set by dnsmasq when dnsmasq (re)starts and throws 'del'
     * events for expired leases. So, libvirtd sets another env var for this
     * purpose */
    if (!interface &&
134 135 136 137
        !(interface = getenv("VIR_BRIDGE_NAME"))) {
        fprintf(stderr, _("interface not set\n"));
        exit(EXIT_FAILURE);
    }
138 139 140

    ip = argv[3];
    mac = argv[2];
P
Peter Krempa 已提交
141 142 143 144 145

    if ((action = virLeaseActionTypeFromString(argv[1])) < 0) {
        fprintf(stderr, _("Unsupported action: %s\n"), argv[1]);
        exit(EXIT_FAILURE);
    }
146 147 148 149 150 151

    /* In case hostname is known, it is the 5th argument */
    if (argc == 5)
        hostname = argv[4];

    /* Check if it is an IPv6 lease */
152
    if (iaid) {
153
        mac = getenv("DNSMASQ_MAC");
154 155 156
        clientid = argv[2];
    }

157
    server_duid = g_strdup(getenv("DNSMASQ_SERVER_DUID"));
158

159 160
    custom_lease_file = g_strdup_printf(LOCALSTATEDIR "/lib/libvirt/dnsmasq/%s.status",
                                        interface);
161

162
    pid_file = g_strdup(RUNSTATEDIR "/leaseshelper.pid");
163 164

    /* Try to claim the pidfile, exiting if we can't */
165 166 167 168
    if ((pid_file_fd = virPidFileAcquirePath(pid_file, true, getpid())) < 0) {
        fprintf(stderr,
                _("Unable to acquire PID file: %s\n errno=%d"),
                pid_file, errno);
169
        goto cleanup;
170
    }
171 172 173

    /* Since interfaces can be hot plugged, we need to make sure that the
     * corresponding custom lease file exists. If not, 'touch' it */
174 175 176 177
    if (virFileTouch(custom_lease_file, 0644) < 0) {
        fprintf(stderr,
                _("Unable to create: %s\n errno=%d"),
                custom_lease_file, errno);
178
        goto cleanup;
179
    }
180

P
Peter Krempa 已提交
181
    switch ((enum virLeaseActionFlags) action) {
182
    case VIR_LEASE_ACTION_ADD:
P
Peter Krempa 已提交
183
    case VIR_LEASE_ACTION_OLD:
J
Ján Tomko 已提交
184
        /* Create new lease */
185
        if (virLeaseNew(&lease_new, mac, clientid, ip, hostname, iaid, server_duid) < 0)
J
Ján Tomko 已提交
186
            goto cleanup;
187 188 189 190 191 192 193 194 195 196 197 198 199
        /* Custom ipv6 leases *will not* be created if the env-var DNSMASQ_MAC
         * is not set. In the special case, when the $(interface).status file
         * is not already present and dnsmasq is (re)started, the corresponding
         * ipv6 custom lease will be created only when the guest sends the
         * 'old' action for its existing ipv6 interfaces.
         *
         * According to rfc3315, the combination of DUID and IAID can be used
         * to uniquely identify each ipv6 guest interface. So, in future, if
         * we introduce virNetworkGetDHCPLeaseBy(IAID|DUID|IAID+DUID) for ipv6
         * interfaces, then, the following if condition won't be required, as
         * the new lease will be created irrespective of whether the MACID is
         * known or not.
         */
J
Ján Tomko 已提交
200
        if (!lease_new)
P
Peter Krempa 已提交
201 202
            break;

203
        G_GNUC_FALLTHROUGH;
P
Peter Krempa 已提交
204
    case VIR_LEASE_ACTION_DEL:
205
        /* Delete the corresponding lease, if it already exists */
P
Peter Krempa 已提交
206 207 208 209 210 211
        delete = true;
        break;

    case VIR_LEASE_ACTION_INIT:
    case VIR_LEASE_ACTION_LAST:
        break;
212
    }
213

214
    leases_array_new = virJSONValueNewArray();
215

216 217 218
    if (virLeaseReadCustomLeaseFile(leases_array_new, custom_lease_file,
                                    delete ? ip : NULL, &server_duid) < 0)
        goto cleanup;
219

220 221
    switch ((enum virLeaseActionFlags) action) {
    case VIR_LEASE_ACTION_INIT:
222 223
        if (virLeasePrintLeases(leases_array_new, server_duid) < 0)
            goto cleanup;
224 225 226 227 228

        break;

    case VIR_LEASE_ACTION_OLD:
    case VIR_LEASE_ACTION_ADD:
229
        if (lease_new && virJSONValueArrayAppend(leases_array_new, lease_new) < 0) {
230 231 232 233
            virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                           _("failed to create json"));
            goto cleanup;
        }
P
Pavel Hrdina 已提交
234
        lease_new = NULL;
235

236
        G_GNUC_FALLTHROUGH;
P
Peter Krempa 已提交
237
    case VIR_LEASE_ACTION_DEL:
238 239 240 241 242
        if (!(leases_str = virJSONValueToString(leases_array_new, true))) {
            virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                           _("empty json array"));
            goto cleanup;
        }
243

244
        /* Write to file */
245
        if (virFileRewriteStr(custom_lease_file, 0644, leases_str) < 0)
246
            goto cleanup;
P
Peter Krempa 已提交
247 248 249 250
        break;

    case VIR_LEASE_ACTION_LAST:
        break;
251
    }
252 253 254 255 256 257 258 259

    rv = EXIT_SUCCESS;

 cleanup:
    if (pid_file_fd != -1)
        virPidFileReleasePath(pid_file, pid_file_fd);

    VIR_FREE(pid_file);
260
    VIR_FREE(server_duid);
261 262 263 264 265 266
    VIR_FREE(custom_lease_file);
    virJSONValueFree(lease_new);
    virJSONValueFree(leases_array_new);

    return rv;
}