blob: 70afda156d1a6534083b4cbeb15a3d7c6b013271 [file] [log] [blame]
// Copyright 2015 The Weave Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "src/macaroon.h"
#include <string.h>
#include "src/crypto_utils.h"
#include "src/macaroon_caveat.h"
#include "src/macaroon_encoding.h"
static bool create_mac_tag_(const uint8_t* key, size_t key_len,
const UwMacaroonCaveat* caveats, size_t num_caveats,
uint8_t mac_tag[UW_MACAROON_MAC_LEN]) {
if (key == NULL || key_len == 0 || caveats == NULL || num_caveats == 0 ||
mac_tag == NULL) {
return false;
}
// Store the intermediate MAC tags in an internal buffer before we finish the
// whole computation.
// If we use the output buffer mac_tag directly and certain errors happen in
// the middle of this computation, mac_tag will probably contain a valid
// macaroon tag with large scope than expected.
uint8_t mac_tag_buff[UW_MACAROON_MAC_LEN];
// Compute the first tag by using the key
if (!uw_macaroon_caveat_sign_(key, key_len, &(caveats[0]), mac_tag_buff,
UW_MACAROON_MAC_LEN)) {
return false;
}
// Compute the rest of the tags by using the tag as the key
for (size_t i = 1; i < num_caveats; i++) {
if (!uw_macaroon_caveat_sign_(mac_tag_buff, UW_MACAROON_MAC_LEN,
&(caveats[i]), mac_tag_buff,
UW_MACAROON_MAC_LEN)) {
return false;
}
}
memcpy(mac_tag, mac_tag_buff, UW_MACAROON_MAC_LEN);
return true;
}
bool uw_macaroon_new_from_mac_tag_(UwMacaroon* new_macaroon,
const uint8_t mac_tag[UW_MACAROON_MAC_LEN],
const UwMacaroonCaveat* caveats,
size_t num_caveats) {
if (new_macaroon == NULL || mac_tag == NULL || caveats == NULL ||
num_caveats == 0) {
return false;
}
memcpy(new_macaroon->mac_tag, mac_tag, UW_MACAROON_MAC_LEN);
new_macaroon->num_caveats = num_caveats;
new_macaroon->caveats = caveats;
return true;
}
bool uw_macaroon_new_from_root_key_(UwMacaroon* new_macaroon,
const uint8_t* root_key,
size_t root_key_len,
const UwMacaroonCaveat* caveats,
size_t num_caveats) {
if (new_macaroon == NULL || root_key == NULL || root_key_len == 0 ||
caveats == NULL || num_caveats == 0) {
return false;
}
if (!create_mac_tag_(root_key, root_key_len, caveats, num_caveats,
new_macaroon->mac_tag)) {
return false;
}
new_macaroon->num_caveats = num_caveats;
new_macaroon->caveats = caveats;
return true;
}
bool uw_macaroon_verify_(const UwMacaroon* macaroon,
const uint8_t* root_key,
size_t root_key_len) {
if (macaroon == NULL || root_key == NULL) {
return false;
}
uint8_t mac_tag[UW_MACAROON_MAC_LEN] = {0};
if (!create_mac_tag_(root_key, root_key_len, macaroon->caveats,
macaroon->num_caveats, mac_tag)) {
return false;
}
return uw_crypto_utils_equal_(mac_tag, macaroon->mac_tag,
UW_MACAROON_MAC_LEN);
}
bool uw_macaroon_extend_(const UwMacaroon* old_macaroon,
UwMacaroon* new_macaroon,
const UwMacaroonCaveat* additional_caveat,
uint8_t* buffer, size_t buffer_size) {
if (old_macaroon == NULL || new_macaroon == NULL ||
additional_caveat == NULL || buffer == NULL || buffer_size == 0) {
return false;
}
new_macaroon->num_caveats = old_macaroon->num_caveats + 1;
// Extend the caveat list
if ((new_macaroon->num_caveats) * sizeof(UwMacaroonCaveat) > buffer_size) {
// Not enough memory to store the extended caveat list
return false;
}
UwMacaroonCaveat* extended_list = (UwMacaroonCaveat*)buffer;
if (old_macaroon->caveats != NULL && extended_list != old_macaroon->caveats) {
memcpy(extended_list, old_macaroon->caveats,
(old_macaroon->num_caveats) * sizeof(UwMacaroonCaveat));
}
extended_list[old_macaroon->num_caveats] = *additional_caveat;
new_macaroon->caveats = extended_list;
// Compute the new MAC tag
return create_mac_tag_(old_macaroon->mac_tag, UW_MACAROON_MAC_LEN,
additional_caveat, 1, new_macaroon->mac_tag);
}
// Encode a Macaroon to a byte string
bool uw_macaroon_dump_(const UwMacaroon* macaroon,
uint8_t* out,
size_t out_len,
size_t* resulting_str_len) {
if (macaroon == NULL || out == NULL || out_len == 0 ||
resulting_str_len == NULL) {
return false;
}
size_t offset = 0, item_len;
if (!uw_macaroon_encoding_encode_byte_str_(
macaroon->mac_tag, UW_MACAROON_MAC_LEN, out, out_len, &item_len)) {
return false;
}
offset += item_len;
if (!uw_macaroon_encoding_encode_array_len_(
(uint32_t)(macaroon->num_caveats), out + offset, out_len - offset, &item_len)) {
return false;
}
offset += item_len;
for (size_t i = 0; i < macaroon->num_caveats; i++) {
if (!uw_macaroon_encoding_encode_byte_str_(
macaroon->caveats[i].bytes, macaroon->caveats[i].num_bytes,
out + offset, out_len - offset, &item_len)) {
return false;
}
offset += item_len;
}
*resulting_str_len = offset;
return true;
}
// Decode a byte string to a Macaroon
bool uw_macaroon_load_(const uint8_t* in,
size_t in_len,
uint8_t* caveats_buffer,
size_t caveats_buffer_size,
UwMacaroon* macaroon) {
if (in == NULL || in_len == 0 || caveats_buffer == NULL ||
caveats_buffer_size == 0 || macaroon == NULL) {
return false;
}
const uint8_t* tag;
size_t tag_len;
if (!uw_macaroon_encoding_decode_byte_str_(in, in_len, &tag, &tag_len) ||
tag_len != UW_MACAROON_MAC_LEN) {
return false;
}
memcpy(macaroon->mac_tag, tag, UW_MACAROON_MAC_LEN);
size_t offset = 0, cbor_item_len;
if (!uw_macaroon_encoding_get_item_len_(in, in_len, &cbor_item_len)) {
return false;
}
offset += cbor_item_len;
uint32_t array_len;
if (!uw_macaroon_encoding_decode_array_len_(in + offset, in_len - offset,
&array_len)) {
return false;
}
macaroon->num_caveats = (size_t)array_len;
if (caveats_buffer_size < array_len * sizeof(UwMacaroonCaveat)) {
return false;
}
UwMacaroonCaveat* caveats = (UwMacaroonCaveat*)caveats_buffer;
for (size_t i = 0; i < array_len; i++) {
if (!uw_macaroon_encoding_get_item_len_(in + offset, in_len - offset,
&cbor_item_len)) {
return false;
}
offset += cbor_item_len;
if (!uw_macaroon_encoding_decode_byte_str_(in + offset, in_len - offset,
&(caveats[i].bytes),
&(caveats[i].num_bytes))) {
return false;
}
}
macaroon->caveats = caveats;
return true;
}