secret_driver.c 13.0 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

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

165 166 167
    if (virSecretLookupByUUIDEnsureACL(conn, secret->def) < 0)
        goto cleanup;

168 169 170
    ret = virGetSecret(conn,
                       secret->def->uuid,
                       secret->def->usage_type,
171
                       virSecretUsageIDForDef(secret->def));
172

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


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

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

194 195 196
    if (virSecretLookupByUsageEnsureACL(conn, secret->def) < 0)
        goto cleanup;

197 198 199
    ret = virGetSecret(conn,
                       secret->def->uuid,
                       secret->def->usage_type,
200
                       virSecretUsageIDForDef(secret->def));
201

202
 cleanup:
203
    virSecretObjEndAPI(&secret);
204 205 206 207 208
    return ret;
}


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

218 219
    virCheckFlags(0, NULL);

220
    if (!(new_attrs = virSecretDefParseString(xml)))
221 222
        return NULL;

223 224 225
    if (virSecretDefineXMLEnsureACL(conn, new_attrs) < 0)
        goto cleanup;

226 227 228
    if (!(secret = virSecretObjListAdd(driver->secrets, new_attrs,
                                       driver->configDir, &backup)))
        goto cleanup;
229 230

    if (!new_attrs->ephemeral) {
231 232 233
        if (secretEnsureDirectory() < 0)
            goto cleanup;

234
        if (backup && backup->ephemeral) {
235
            if (virSecretObjSaveData(secret) < 0)
236 237
                goto restore_backup;
        }
238 239

        if (virSecretObjSaveConfig(secret) < 0) {
240
            if (backup && backup->ephemeral) {
241 242
                /* Undo the virSecretObjSaveData() above; ignore errors */
                virSecretObjDeleteData(secret);
243 244 245
            }
            goto restore_backup;
        }
246
    } else if (backup && !backup->ephemeral) {
247
        if (virSecretObjDeleteConfig(secret) < 0)
248
            goto restore_backup;
249 250

        virSecretObjDeleteData(secret);
251
    }
252
    /* Saved successfully - drop old values */
253 254 255
    new_attrs = NULL;
    virSecretDefFree(backup);

256 257 258
    ret = virGetSecret(conn,
                       secret->def->uuid,
                       secret->def->usage_type,
259
                       virSecretUsageIDForDef(secret->def));
260 261
    goto cleanup;

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

272
 cleanup:
273
    virSecretDefFree(new_attrs);
274
    virSecretObjEndAPI(&secret);
275 276 277 278 279

    return ret;
}

static char *
280 281
secretGetXMLDesc(virSecretPtr obj,
                 unsigned int flags)
282 283
{
    char *ret = NULL;
J
John Ferlan 已提交
284
    virSecretObjPtr secret;
285

286 287
    virCheckFlags(0, NULL);

J
John Ferlan 已提交
288
    if (!(secret = secretObjFromSecret(obj)))
289 290
        goto cleanup;

291 292 293
    if (virSecretGetXMLDescEnsureACL(obj->conn, secret->def) < 0)
        goto cleanup;

294
    ret = virSecretDefFormat(secret->def);
295

296
 cleanup:
297
    virSecretObjEndAPI(&secret);
298 299 300 301 302

    return ret;
}

static int
303 304 305 306
secretSetValue(virSecretPtr obj,
               const unsigned char *value,
               size_t value_size,
               unsigned int flags)
307 308 309 310
{
    int ret = -1;
    unsigned char *old_value, *new_value;
    size_t old_value_size;
J
John Ferlan 已提交
311
    virSecretObjPtr secret;
312

313 314
    virCheckFlags(0, -1);

315
    if (VIR_ALLOC_N(new_value, value_size) < 0)
316 317
        return -1;

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

321 322 323
    if (virSecretSetValueEnsureACL(obj->conn, secret->def) < 0)
        goto cleanup;

324 325 326 327 328 329 330
    old_value = secret->value;
    old_value_size = secret->value_size;

    memcpy(new_value, value, value_size);
    secret->value = new_value;
    secret->value_size = value_size;
    if (!secret->def->ephemeral) {
331 332 333 334
        if (secretEnsureDirectory() < 0)
            goto cleanup;

        if (virSecretObjSaveData(secret) < 0)
335 336
            goto restore_backup;
    }
337
    /* Saved successfully - drop old value */
338 339 340 341 342 343 344 345 346
    if (old_value != NULL) {
        memset(old_value, 0, old_value_size);
        VIR_FREE(old_value);
    }
    new_value = NULL;

    ret = 0;
    goto cleanup;

347
 restore_backup:
348 349 350 351 352
    /* Error - restore previous state and free new value */
    secret->value = old_value;
    secret->value_size = old_value_size;
    memset(new_value, 0, value_size);

353
 cleanup:
354
    virSecretObjEndAPI(&secret);
355 356 357 358 359 360 361

    VIR_FREE(new_value);

    return ret;
}

static unsigned char *
362 363 364
secretGetValue(virSecretPtr obj,
               size_t *value_size,
               unsigned int flags,
365
               unsigned int internalFlags)
366 367
{
    unsigned char *ret = NULL;
J
John Ferlan 已提交
368
    virSecretObjPtr secret;
369

370 371
    virCheckFlags(0, NULL);

J
John Ferlan 已提交
372
    if (!(secret = secretObjFromSecret(obj)))
373
        goto cleanup;
374

375 376 377
    if (virSecretGetValueEnsureACL(obj->conn, secret->def) < 0)
        goto cleanup;

378
    if (secret->value == NULL) {
379 380
        char uuidstr[VIR_UUID_STRING_BUFLEN];
        virUUIDFormat(obj->uuid, uuidstr);
381 382
        virReportError(VIR_ERR_NO_SECRET,
                       _("secret '%s' does not have a value"), uuidstr);
383 384 385
        goto cleanup;
    }

386
    if ((internalFlags & VIR_SECRET_GET_VALUE_INTERNAL_CALL) == 0 &&
387
        secret->def->private) {
388
        virReportError(VIR_ERR_INVALID_SECRET, "%s",
389
                       _("secret is private"));
390 391 392
        goto cleanup;
    }

393
    if (VIR_ALLOC_N(ret, secret->value_size) < 0)
394 395 396 397
        goto cleanup;
    memcpy(ret, secret->value, secret->value_size);
    *value_size = secret->value_size;

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

    return ret;
}

static int
secretUndefine(virSecretPtr obj)
{
    int ret = -1;
J
John Ferlan 已提交
408
    virSecretObjPtr secret;
409

J
John Ferlan 已提交
410
    if (!(secret = secretObjFromSecret(obj)))
411 412
        goto cleanup;

413 414 415
    if (virSecretUndefineEnsureACL(obj->conn, secret->def) < 0)
        goto cleanup;

416
    if (virSecretObjDeleteConfig(secret) < 0)
417 418
        goto cleanup;

419 420
    virSecretObjDeleteData(secret);

421
    virSecretObjListRemove(driver->secrets, secret);
422 423 424

    ret = 0;

425
 cleanup:
426
    virSecretObjEndAPI(&secret);
427 428 429 430 431

    return ret;
}

static int
432
secretStateCleanup(void)
433
{
434
    if (!driver)
435 436
        return -1;

437
    secretDriverLock();
438

439
    virObjectUnref(driver->secrets);
440
    VIR_FREE(driver->configDir);
441

442 443 444
    secretDriverUnlock();
    virMutexDestroy(&driver->lock);
    VIR_FREE(driver);
445 446 447 448 449

    return 0;
}

static int
450 451 452
secretStateInitialize(bool privileged,
                      virStateInhibitCallback callback ATTRIBUTE_UNUSED,
                      void *opaque ATTRIBUTE_UNUSED)
453 454 455
{
    char *base = NULL;

456
    if (VIR_ALLOC(driver) < 0)
457 458
        return -1;

459 460
    if (virMutexInit(&driver->lock) < 0) {
        VIR_FREE(driver);
461 462
        return -1;
    }
463
    secretDriverLock();
464 465

    if (privileged) {
466 467
        if (VIR_STRDUP(base, SYSCONFDIR "/libvirt") < 0)
            goto error;
468
    } else {
469
        if (!(base = virGetUserConfigDirectory()))
470 471
            goto error;
    }
472
    if (virAsprintf(&driver->configDir, "%s/secrets", base) < 0)
473
        goto error;
474 475
    VIR_FREE(base);

476 477 478
    if (!(driver->secrets = virSecretObjListNew()))
        goto error;

479
    if (virSecretLoadAllConfigs(driver->secrets, driver->configDir) < 0)
480 481
        goto error;

482
    secretDriverUnlock();
483 484 485 486
    return 0;

 error:
    VIR_FREE(base);
487
    secretDriverUnlock();
488
    secretStateCleanup();
489 490 491 492
    return -1;
}

static int
493
secretStateReload(void)
494
{
495
    if (!driver)
496 497
        return -1;

498
    secretDriverLock();
499

500
    ignore_value(virSecretLoadAllConfigs(driver->secrets, driver->configDir));
501

502
    secretDriverUnlock();
503 504 505 506 507
    return 0;
}

static virSecretDriver secretDriver = {
    .name = "secret",
508 509 510
    .connectNumOfSecrets = secretConnectNumOfSecrets, /* 0.7.1 */
    .connectListSecrets = secretConnectListSecrets, /* 0.7.1 */
    .connectListAllSecrets = secretConnectListAllSecrets, /* 0.10.2 */
511 512 513 514 515 516 517
    .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 */
518 519 520
};

static virStateDriver stateDriver = {
521
    .name = "secret",
522 523 524
    .stateInitialize = secretStateInitialize,
    .stateCleanup = secretStateCleanup,
    .stateReload = secretStateReload,
525 526 527 528 529
};

int
secretRegister(void)
{
530
    if (virSetSharedSecretDriver(&secretDriver) < 0)
531 532 533
        return -1;
    if (virRegisterStateDriver(&stateDriver) < 0)
        return -1;
534 535
    return 0;
}