secret_driver.c 12.3 KB
Newer Older
1 2 3
/*
 * secret_driver.c: local driver for secret manipulation API
 *
4
 * Copyright (C) 2009-2016 Red Hat, Inc.
5 6 7 8 9 10 11 12 13 14 15 16
 *
 * 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
17
 * License along with this library.  If not, see
O
Osier Yang 已提交
18
 * <http://www.gnu.org/licenses/>.
19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
 *
 * Red Hat Author: Miloslav Trmač <mitr@redhat.com>
 */

#include <config.h>

#include <fcntl.h>
#include <string.h>
#include <sys/stat.h>
#include <unistd.h>

#include "internal.h"
#include "base64.h"
#include "datatypes.h"
#include "driver.h"
34
#include "virlog.h"
35
#include "viralloc.h"
36
#include "secret_conf.h"
37
#include "virsecretobj.h"
38
#include "secret_driver.h"
39
#include "virthread.h"
40
#include "viruuid.h"
41
#include "virerror.h"
E
Eric Blake 已提交
42
#include "virfile.h"
43
#include "configmake.h"
44
#include "virstring.h"
45
#include "viraccessapicheck.h"
46 47 48

#define VIR_FROM_THIS VIR_FROM_SECRET

49 50
VIR_LOG_INIT("secret.secret_driver");

51 52
enum { SECRET_MAX_XML_FILE = 10*1024*1024 };

53
/* Internal driver state */
54 55 56 57 58

typedef struct _virSecretDriverState virSecretDriverState;
typedef virSecretDriverState *virSecretDriverStatePtr;
struct _virSecretDriverState {
    virMutex lock;
59
    virSecretObjListPtr secrets;
60
    char *configDir;
61 62
};

63
static virSecretDriverStatePtr driver;
64 65

static void
66
secretDriverLock(void)
67 68 69 70 71
{
    virMutexLock(&driver->lock);
}

static void
72
secretDriverUnlock(void)
73 74 75 76
{
    virMutexUnlock(&driver->lock);
}

J
John Ferlan 已提交
77

J
John Ferlan 已提交
78 79 80 81 82 83 84

static virSecretObjPtr
secretObjFromSecret(virSecretPtr secret)
{
    virSecretObjPtr obj;
    char uuidstr[VIR_UUID_STRING_BUFLEN];

85
    if (!(obj = virSecretObjListFindByUUID(driver->secrets, secret->uuid))) {
J
John Ferlan 已提交
86 87 88 89 90 91 92 93 94
        virUUIDFormat(secret->uuid, uuidstr);
        virReportError(VIR_ERR_NO_SECRET,
                       _("no secret with matching uuid '%s'"), uuidstr);
        return NULL;
    }
    return obj;
}


95
static int
96
secretEnsureDirectory(void)
97
{
98
    if (mkdir(driver->configDir, S_IRWXU) < 0 && errno != EEXIST) {
99
        virReportSystemError(errno, _("cannot create '%s'"),
100
                             driver->configDir);
101 102 103 104 105
        return -1;
    }
    return 0;
}

106
/* Driver functions */
107 108

static int
109
secretConnectNumOfSecrets(virConnectPtr conn)
110
{
111 112 113
    if (virConnectNumOfSecretsEnsureACL(conn) < 0)
        return -1;

114 115 116
    return virSecretObjListNumOfSecrets(driver->secrets,
                                        virConnectNumOfSecretsCheckACL,
                                        conn);
117 118 119
}

static int
120 121 122
secretConnectListSecrets(virConnectPtr conn,
                         char **uuids,
                         int maxuuids)
123 124 125
{
    memset(uuids, 0, maxuuids * sizeof(*uuids));

126 127 128
    if (virConnectListSecretsEnsureACL(conn) < 0)
        return -1;

129 130
    return virSecretObjListGetUUIDs(driver->secrets, uuids, maxuuids,
                                    virConnectListSecretsCheckACL, conn);
131 132
}

133

O
Osier Yang 已提交
134
static int
135 136
secretConnectListAllSecrets(virConnectPtr conn,
                            virSecretPtr **secrets,
137 138
                            unsigned int flags)
{
O
Osier Yang 已提交
139 140
    virCheckFlags(VIR_CONNECT_LIST_SECRETS_FILTERS_ALL, -1);

141 142 143
    if (virConnectListAllSecretsEnsureACL(conn) < 0)
        return -1;

144 145 146
    return virSecretObjListExport(conn, driver->secrets, secrets,
                                  virConnectListAllSecretsCheckACL,
                                  flags);
O
Osier Yang 已提交
147 148 149
}


150
static virSecretPtr
151 152
secretLookupByUUID(virConnectPtr conn,
                   const unsigned char *uuid)
153 154
{
    virSecretPtr ret = NULL;
J
John Ferlan 已提交
155
    virSecretObjPtr secret;
156
    virSecretDefPtr def;
157

158
    if (!(secret = virSecretObjListFindByUUID(driver->secrets, uuid))) {
159 160
        char uuidstr[VIR_UUID_STRING_BUFLEN];
        virUUIDFormat(uuid, uuidstr);
161 162
        virReportError(VIR_ERR_NO_SECRET,
                       _("no secret with matching uuid '%s'"), uuidstr);
163 164 165
        goto cleanup;
    }

166 167
    def = virSecretObjGetDef(secret);
    if (virSecretLookupByUUIDEnsureACL(conn, def) < 0)
168 169
        goto cleanup;

170
    ret = virGetSecret(conn,
171 172 173
                       def->uuid,
                       def->usage_type,
                       virSecretUsageIDForDef(def));
174

175
 cleanup:
176
    virSecretObjEndAPI(&secret);
177 178 179 180 181
    return ret;
}


static virSecretPtr
182 183 184
secretLookupByUsage(virConnectPtr conn,
                    int usageType,
                    const char *usageID)
185 186
{
    virSecretPtr ret = NULL;
J
John Ferlan 已提交
187
    virSecretObjPtr secret;
188
    virSecretDefPtr def;
189

190 191
    if (!(secret = virSecretObjListFindByUsage(driver->secrets,
                                               usageType, usageID))) {
192 193
        virReportError(VIR_ERR_NO_SECRET,
                       _("no secret with matching usage '%s'"), usageID);
194 195 196
        goto cleanup;
    }

197 198
    def = virSecretObjGetDef(secret);
    if (virSecretLookupByUsageEnsureACL(conn, def) < 0)
199 200
        goto cleanup;

201
    ret = virGetSecret(conn,
202 203 204
                       def->uuid,
                       def->usage_type,
                       virSecretUsageIDForDef(def));
205

206
 cleanup:
207
    virSecretObjEndAPI(&secret);
208 209 210 211 212
    return ret;
}


static virSecretPtr
213 214
secretDefineXML(virConnectPtr conn,
                const char *xml,
215
                unsigned int flags)
216 217
{
    virSecretPtr ret = NULL;
J
John Ferlan 已提交
218
    virSecretObjPtr secret = NULL;
219 220
    virSecretDefPtr backup = NULL;
    virSecretDefPtr new_attrs;
221

222 223
    virCheckFlags(0, NULL);

224
    if (!(new_attrs = virSecretDefParseString(xml)))
225 226
        return NULL;

227 228 229
    if (virSecretDefineXMLEnsureACL(conn, new_attrs) < 0)
        goto cleanup;

230 231 232
    if (!(secret = virSecretObjListAdd(driver->secrets, new_attrs,
                                       driver->configDir, &backup)))
        goto cleanup;
233

234
    if (!new_attrs->isephemeral) {
235 236 237
        if (secretEnsureDirectory() < 0)
            goto cleanup;

238
        if (backup && backup->isephemeral) {
239
            if (virSecretObjSaveData(secret) < 0)
240 241
                goto restore_backup;
        }
242 243

        if (virSecretObjSaveConfig(secret) < 0) {
244
            if (backup && backup->isephemeral) {
245 246
                /* Undo the virSecretObjSaveData() above; ignore errors */
                virSecretObjDeleteData(secret);
247 248 249
            }
            goto restore_backup;
        }
250
    } else if (backup && !backup->isephemeral) {
251
        if (virSecretObjDeleteConfig(secret) < 0)
252
            goto restore_backup;
253 254

        virSecretObjDeleteData(secret);
255
    }
256
    /* Saved successfully - drop old values */
257 258
    virSecretDefFree(backup);

259
    ret = virGetSecret(conn,
260 261 262 263
                       new_attrs->uuid,
                       new_attrs->usage_type,
                       virSecretUsageIDForDef(new_attrs));
    new_attrs = NULL;
264 265
    goto cleanup;

266
 restore_backup:
267
    /* If we have a backup, then secret was defined before, so just restore
268
     * the backup. The current (new_attrs) will be handled below.
269 270 271
     * Otherwise, this is a new secret, thus remove it.
     */
    if (backup)
272
        virSecretObjSetDef(secret, backup);
273 274
    else
        virSecretObjListRemove(driver->secrets, secret);
275

276
 cleanup:
277
    virSecretDefFree(new_attrs);
278
    virSecretObjEndAPI(&secret);
279 280 281 282 283

    return ret;
}

static char *
284 285
secretGetXMLDesc(virSecretPtr obj,
                 unsigned int flags)
286 287
{
    char *ret = NULL;
J
John Ferlan 已提交
288
    virSecretObjPtr secret;
289
    virSecretDefPtr def;
290

291 292
    virCheckFlags(0, NULL);

J
John Ferlan 已提交
293
    if (!(secret = secretObjFromSecret(obj)))
294 295
        goto cleanup;

296 297
    def = virSecretObjGetDef(secret);
    if (virSecretGetXMLDescEnsureACL(obj->conn, def) < 0)
298 299
        goto cleanup;

300
    ret = virSecretDefFormat(def);
301

302
 cleanup:
303
    virSecretObjEndAPI(&secret);
304 305 306 307 308

    return ret;
}

static int
309 310 311 312
secretSetValue(virSecretPtr obj,
               const unsigned char *value,
               size_t value_size,
               unsigned int flags)
313 314
{
    int ret = -1;
J
John Ferlan 已提交
315
    virSecretObjPtr secret;
316
    virSecretDefPtr def;
317

318 319
    virCheckFlags(0, -1);

J
John Ferlan 已提交
320
    if (!(secret = secretObjFromSecret(obj)))
321 322
        goto cleanup;

323 324
    def = virSecretObjGetDef(secret);
    if (virSecretSetValueEnsureACL(obj->conn, def) < 0)
325 326
        goto cleanup;

327 328
    if (secretEnsureDirectory() < 0)
        goto cleanup;
329

330 331
    if (virSecretObjSetValue(secret, value, value_size) < 0)
        goto cleanup;
332 333 334

    ret = 0;

335
 cleanup:
336
    virSecretObjEndAPI(&secret);
337 338 339 340 341

    return ret;
}

static unsigned char *
342 343 344
secretGetValue(virSecretPtr obj,
               size_t *value_size,
               unsigned int flags,
345
               unsigned int internalFlags)
346 347
{
    unsigned char *ret = NULL;
J
John Ferlan 已提交
348
    virSecretObjPtr secret;
349
    virSecretDefPtr def;
350

351 352
    virCheckFlags(0, NULL);

J
John Ferlan 已提交
353
    if (!(secret = secretObjFromSecret(obj)))
354
        goto cleanup;
355

356 357
    def = virSecretObjGetDef(secret);
    if (virSecretGetValueEnsureACL(obj->conn, def) < 0)
358 359
        goto cleanup;

360
    if ((internalFlags & VIR_SECRET_GET_VALUE_INTERNAL_CALL) == 0 &&
361
        def->isprivate) {
362
        virReportError(VIR_ERR_INVALID_SECRET, "%s",
363
                       _("secret is private"));
364 365 366
        goto cleanup;
    }

367
    if (!(ret = virSecretObjGetValue(secret)))
368
        goto cleanup;
369 370

    *value_size = virSecretObjGetValueSize(secret);
371

372
 cleanup:
373
    virSecretObjEndAPI(&secret);
374 375 376 377 378 379 380 381

    return ret;
}

static int
secretUndefine(virSecretPtr obj)
{
    int ret = -1;
J
John Ferlan 已提交
382
    virSecretObjPtr secret;
383
    virSecretDefPtr def;
384

J
John Ferlan 已提交
385
    if (!(secret = secretObjFromSecret(obj)))
386 387
        goto cleanup;

388 389
    def = virSecretObjGetDef(secret);
    if (virSecretUndefineEnsureACL(obj->conn, def) < 0)
390 391
        goto cleanup;

392
    if (virSecretObjDeleteConfig(secret) < 0)
393 394
        goto cleanup;

395 396
    virSecretObjDeleteData(secret);

397
    virSecretObjListRemove(driver->secrets, secret);
398 399 400

    ret = 0;

401
 cleanup:
402
    virSecretObjEndAPI(&secret);
403 404 405 406 407

    return ret;
}

static int
408
secretStateCleanup(void)
409
{
410
    if (!driver)
411 412
        return -1;

413
    secretDriverLock();
414

415
    virObjectUnref(driver->secrets);
416
    VIR_FREE(driver->configDir);
417

418 419 420
    secretDriverUnlock();
    virMutexDestroy(&driver->lock);
    VIR_FREE(driver);
421 422 423 424 425

    return 0;
}

static int
426 427 428
secretStateInitialize(bool privileged,
                      virStateInhibitCallback callback ATTRIBUTE_UNUSED,
                      void *opaque ATTRIBUTE_UNUSED)
429 430 431
{
    char *base = NULL;

432
    if (VIR_ALLOC(driver) < 0)
433 434
        return -1;

435 436
    if (virMutexInit(&driver->lock) < 0) {
        VIR_FREE(driver);
437 438
        return -1;
    }
439
    secretDriverLock();
440 441

    if (privileged) {
442 443
        if (VIR_STRDUP(base, SYSCONFDIR "/libvirt") < 0)
            goto error;
444
    } else {
445
        if (!(base = virGetUserConfigDirectory()))
446 447
            goto error;
    }
448
    if (virAsprintf(&driver->configDir, "%s/secrets", base) < 0)
449
        goto error;
450 451
    VIR_FREE(base);

452 453 454
    if (!(driver->secrets = virSecretObjListNew()))
        goto error;

455
    if (virSecretLoadAllConfigs(driver->secrets, driver->configDir) < 0)
456 457
        goto error;

458
    secretDriverUnlock();
459 460 461 462
    return 0;

 error:
    VIR_FREE(base);
463
    secretDriverUnlock();
464
    secretStateCleanup();
465 466 467 468
    return -1;
}

static int
469
secretStateReload(void)
470
{
471
    if (!driver)
472 473
        return -1;

474
    secretDriverLock();
475

476
    ignore_value(virSecretLoadAllConfigs(driver->secrets, driver->configDir));
477

478
    secretDriverUnlock();
479 480 481 482 483
    return 0;
}

static virSecretDriver secretDriver = {
    .name = "secret",
484 485 486
    .connectNumOfSecrets = secretConnectNumOfSecrets, /* 0.7.1 */
    .connectListSecrets = secretConnectListSecrets, /* 0.7.1 */
    .connectListAllSecrets = secretConnectListAllSecrets, /* 0.10.2 */
487 488 489 490 491 492 493
    .secretLookupByUUID = secretLookupByUUID, /* 0.7.1 */
    .secretLookupByUsage = secretLookupByUsage, /* 0.7.1 */
    .secretDefineXML = secretDefineXML, /* 0.7.1 */
    .secretGetXMLDesc = secretGetXMLDesc, /* 0.7.1 */
    .secretSetValue = secretSetValue, /* 0.7.1 */
    .secretGetValue = secretGetValue, /* 0.7.1 */
    .secretUndefine = secretUndefine, /* 0.7.1 */
494 495 496
};

static virStateDriver stateDriver = {
497
    .name = "secret",
498 499 500
    .stateInitialize = secretStateInitialize,
    .stateCleanup = secretStateCleanup,
    .stateReload = secretStateReload,
501 502 503 504 505
};

int
secretRegister(void)
{
506
    if (virSetSharedSecretDriver(&secretDriver) < 0)
507 508 509
        return -1;
    if (virRegisterStateDriver(&stateDriver) < 0)
        return -1;
510 511
    return 0;
}