Skip to content

Commit

Permalink
Refacto: update Policy & Keys data structure (#119)
Browse files Browse the repository at this point in the history
- Change `Axis` to `Dimension` with a clear distinction between `Ordered` and `Unordered`.
- Create a `Dictionary` data structure to store and update `Ordered Dimension` efficiently
- Change the data structure of `MasterSecretKey` and `UserSecretKey` to keep track of subkeys version.
- `Policy` does not count the attribute rotations anymore as they are stored in the subkeys.
- a `UserSecretKey` can now be refreshed without any external `Policy` information.

Co-authored-by: Théophile BRÉZOT <[email protected]>
  • Loading branch information
Hugo Rosenkranz-Costa and tbrezot authored Jan 12, 2024
1 parent 30624d3 commit 3725319
Show file tree
Hide file tree
Showing 26 changed files with 1,868 additions and 870 deletions.
10 changes: 10 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,16 @@

All notable changes to this project will be documented in this file.

## [Unreleased]

### Features

- Change `Axis` to `Dimension` with a clear distinction between `Ordered` and `Unordered`.
- Create a `Dictionary` data structure to store and update `Ordered Dimension` efficiently
- Change the data structure of `MasterSecretKey` and `UserSecretKey` to keep track of subkeys version.
- Policy does not count the attribute rotations anymore as they are stored in the subkeys.
- a `UserSecretKey` can now be refreshed without any external `Policy` information.

## [13.0.0] - 2023-11-06

### Bug Fixes
Expand Down
2 changes: 1 addition & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[package]
name = "cosmian_cover_crypt"
version = "13.0.0"
version = "13.1.0"
authors = [
"Théophile Brezot <[email protected]>",
"Bruno Grieder <[email protected]>",
Expand Down
75 changes: 39 additions & 36 deletions benches/benches.rs
Original file line number Diff line number Diff line change
Expand Up @@ -65,10 +65,10 @@ fn policy() -> Result<Policy, Error> {
fn bench_policy_editing(c: &mut Criterion) {
let cover_crypt = Covercrypt::default();
let new_dep_attr = Attribute::new("Department", "Tech");
let new_dep_name = "IT";
let new_dep_name = "IT".to_string();
let remove_dep_attr = Attribute::new("Department", "FIN");
let old_sl_attr = Attribute::new("Security Level", "Protected");
let new_sl_name = "Open";
let new_sl_name = "Open".to_string();
let disable_sl_attr = Attribute::new("Security Level", "Confidential");

let mut group = c.benchmark_group("Edit Policy");
Expand All @@ -88,11 +88,13 @@ fn bench_policy_editing(c: &mut Criterion) {
.add_attribute(new_dep_attr.clone(), EncryptionHint::Classic)
.unwrap();
policy
.rename_attribute(&new_dep_attr, new_dep_name)
.rename_attribute(&new_dep_attr, new_dep_name.clone())
.unwrap();
policy.remove_attribute(&remove_dep_attr).unwrap();

policy.rename_attribute(&old_sl_attr, new_sl_name).unwrap();
policy
.rename_attribute(&old_sl_attr, new_sl_name.clone())
.unwrap();
policy.disable_attribute(&disable_sl_attr).unwrap();

cover_crypt
Expand Down Expand Up @@ -308,57 +310,58 @@ fn bench_header_encryption(c: &mut Criterion) {
fn bench_header_decryption(c: &mut Criterion) {
let policy = policy().expect("cannot generate policy");
let authenticated_data = vec![1, 2, 3, 4, 5, 6, 7, 8, 9];
let (user_access_policy, access_policies) = get_access_policies();
let (user_access_policies, access_policies) = get_access_policies();
let cover_crypt = Covercrypt::default();
let (msk, mpk) = cover_crypt
.generate_master_keys(&policy)
.expect("cannot generate master keys");
let user_decryption_keys: Vec<_> = user_access_policy
.iter()
.map(|ap| {
cover_crypt
.generate_user_secret_key(&msk, ap, &policy)
.expect("cannot generate user private key")
})
.collect();
let mut group = c.benchmark_group("Header encryption and decryption");
for (n_partitions_usk, usk) in user_decryption_keys.iter().enumerate() {
for (n_user, user_access_policy) in user_access_policies.iter().enumerate() {
for (n_partition_ct, access_policy) in access_policies.iter().enumerate() {
group.bench_function(
&format!(
"ciphertexts with {} partition(s), usk with {} partitions",
n_partition_ct + 1,
n_partitions_usk + 1
n_user + 1
),
|b| {
b.iter(|| {
let (_, encrypted_header) = EncryptedHeader::generate(
&cover_crypt,
&policy,
&mpk,
access_policy,
None,
Some(&authenticated_data),
)
.unwrap_or_else(|_| {
panic!(
"cannot encrypt header for {} ciphertext partition(s), {} usk \
partition(s)",
n_partition_ct + 1,
n_partitions_usk
b.iter_batched(
|| {
let usk = cover_crypt
.generate_user_secret_key(&msk, user_access_policy, &policy)
.expect("cannot generate user private key");
let (_, encrypted_header) = EncryptedHeader::generate(
&cover_crypt,
&policy,
&mpk,
access_policy,
None,
Some(&authenticated_data),
)
});
encrypted_header
.decrypt(&cover_crypt, usk, Some(&authenticated_data))
.unwrap_or_else(|_| {
panic!(
"cannot decrypt header for {} ciphertext partition(s), {} usk \
"cannot encrypt header for {} ciphertext partition(s), {} usk \
partition(s)",
n_partition_ct + 1,
n_partitions_usk
n_user
)
});
});
(usk, encrypted_header)
},
|(usk, encrypted_header)| {
encrypted_header
.decrypt(&cover_crypt, &usk, Some(&authenticated_data))
.unwrap_or_else(|_| {
panic!(
"cannot decrypt header for {} ciphertext partition(s), {} \
usk partition(s)",
n_partition_ct + 1,
n_user
)
});
},
BatchSize::SmallInput,
);
},
);
}
Expand Down
12 changes: 4 additions & 8 deletions examples/runme.rs
Original file line number Diff line number Diff line change
Expand Up @@ -61,14 +61,10 @@ fn main() {
assert!(encrypted_header.decrypt(&cover_crypt, &usk, None).is_ok());

//
// Rotate the `Security Level::Top Secret` attribute
policy
.rotate(&Attribute::from(("Security Level", "Top Secret")))
.unwrap();

// Master keys need to be updated to take into account the policy rotation
// Rekey all keys using the `Security Level::Top Secret` attribute
let rekey_access_policy = AccessPolicy::Attr(Attribute::from(("Security Level", "Top Secret")));
cover_crypt
.update_master_keys(&policy, &mut msk, &mut mpk)
.rekey_master_keys(&rekey_access_policy, &policy, &mut msk, &mut mpk)
.unwrap();

// Encrypt with rotated attribute
Expand All @@ -82,7 +78,7 @@ fn main() {

// refresh user secret key, do not grant old encryption access
cover_crypt
.refresh_user_secret_key(&mut usk, &access_policy, &msk, &policy, false)
.refresh_user_secret_key(&mut usk, &msk, false)
.unwrap();

// The user with refreshed key is able to decrypt the newly encrypted header.
Expand Down
16 changes: 9 additions & 7 deletions src/abe_policy/access_policy.rs
Original file line number Diff line number Diff line change
Expand Up @@ -346,7 +346,8 @@ impl AccessPolicy {
/// given access policy. It is an OR expression of AND expressions.
///
/// - `policy` : global policy
/// - `include_lower_attributes_from_dim` : set to `true` to combine lower attributes
/// - `include_lower_attributes_from_dim` : set to `true` to combine lower
/// attributes
/// from dimension with hierarchical order
pub fn to_attribute_combinations(
&self,
Expand All @@ -360,12 +361,13 @@ impl AccessPolicy {
.get(&attr.dimension)
.ok_or_else(|| Error::DimensionNotFound(attr.dimension.to_string()))?;
let mut res = vec![vec![attr.clone()]];
if let Some(order) = dim_parameters.order.as_deref() {
if include_lower_attributes_from_dim {
// add attribute values for all attributes below the given one
for name in order.iter().take_while(|&name| name != &attr.name) {
res.push(vec![Attribute::new(&attr.dimension, name)]);
}
if include_lower_attributes_from_dim && dim_parameters.is_ordered() {
// add attribute values for all attributes below the given one
for name in dim_parameters
.get_attributes_name()
.take_while(|&name| name != &attr.name)
{
res.push(vec![Attribute::new(&attr.dimension, name)]);
}
}
Ok(res)
Expand Down
7 changes: 7 additions & 0 deletions src/abe_policy/attribute.rs
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ impl BitOr for EncryptionHint {
}

impl EncryptionHint {
#[must_use]
pub fn new(is_hybridized: bool) -> Self {
if is_hybridized {
Self::Hybridized
Expand All @@ -39,6 +40,12 @@ impl EncryptionHint {
}
}

impl From<EncryptionHint> for bool {
fn from(val: EncryptionHint) -> Self {
val == EncryptionHint::Hybridized
}
}

/// Whether to provide an encryption key in the master public key for this
/// attribute.
#[derive(Copy, Clone, Debug, Serialize, Deserialize, PartialEq, Eq)]
Expand Down
Loading

0 comments on commit 3725319

Please sign in to comment.