Merge changes from topic "aconfig-part-2" am: 80bff11aef am: 2fc13df5d7 am: 08492141ab am: 0abf095ac5 am: c1397bd11c am: 359f95ab50

Original change: https://android-review.googlesource.com/c/platform/build/+/2578555

Change-Id: I4dfaa141c77973149e0b51563bdae6526c61b32d
Signed-off-by: Automerger Merge Worker <android-build-automerger-merge-worker@system.gserviceaccount.com>
This commit is contained in:
Mårten Kongstad 2023-05-08 15:56:09 +00:00 committed by Automerger Merge Worker
commit 2c8f3bdd57
5 changed files with 200 additions and 76 deletions

View file

@ -20,9 +20,20 @@ syntax = "proto2";
package android.aconfig; package android.aconfig;
enum flag_state {
ENABLED = 1;
DISABLED = 2;
}
enum permission {
READ_ONLY = 1;
READ_WRITE = 2;
}
message value { message value {
required bool value = 1; required flag_state state = 1;
optional uint32 since = 2; required permission permission = 2;
optional uint32 since = 3;
} }
message flag { message flag {
@ -37,7 +48,8 @@ message android_config {
message override { message override {
required string id = 1; required string id = 1;
required bool value = 2; required flag_state state = 2;
required permission permission = 3;
}; };
message override_config { message override_config {

View file

@ -15,25 +15,65 @@
*/ */
use anyhow::{anyhow, Context, Error, Result}; use anyhow::{anyhow, Context, Error, Result};
use protobuf::{Enum, EnumOrUnknown};
use serde::{Deserialize, Serialize};
use crate::protos::{ use crate::protos::{
ProtoAndroidConfig, ProtoFlag, ProtoOverride, ProtoOverrideConfig, ProtoValue, ProtoAndroidConfig, ProtoFlag, ProtoFlagState, ProtoOverride, ProtoOverrideConfig,
ProtoPermission, ProtoValue,
}; };
#[derive(Debug, PartialEq, Eq, Serialize, Deserialize, Clone, Copy)]
pub enum FlagState {
Enabled,
Disabled,
}
impl TryFrom<EnumOrUnknown<ProtoFlagState>> for FlagState {
type Error = Error;
fn try_from(proto: EnumOrUnknown<ProtoFlagState>) -> Result<Self, Self::Error> {
match ProtoFlagState::from_i32(proto.value()) {
Some(ProtoFlagState::ENABLED) => Ok(FlagState::Enabled),
Some(ProtoFlagState::DISABLED) => Ok(FlagState::Disabled),
None => Err(anyhow!("unknown flag state enum value {}", proto.value())),
}
}
}
#[derive(Debug, PartialEq, Eq, Serialize, Deserialize, Clone, Copy)]
pub enum Permission {
ReadOnly,
ReadWrite,
}
impl TryFrom<EnumOrUnknown<ProtoPermission>> for Permission {
type Error = Error;
fn try_from(proto: EnumOrUnknown<ProtoPermission>) -> Result<Self, Self::Error> {
match ProtoPermission::from_i32(proto.value()) {
Some(ProtoPermission::READ_ONLY) => Ok(Permission::ReadOnly),
Some(ProtoPermission::READ_WRITE) => Ok(Permission::ReadWrite),
None => Err(anyhow!("unknown permission enum value {}", proto.value())),
}
}
}
#[derive(Debug, PartialEq, Eq)] #[derive(Debug, PartialEq, Eq)]
pub struct Value { pub struct Value {
value: bool, state: FlagState,
permission: Permission,
since: Option<u32>, since: Option<u32>,
} }
#[allow(dead_code)] // only used in unit tests #[allow(dead_code)] // only used in unit tests
impl Value { impl Value {
pub fn new(value: bool, since: u32) -> Value { pub fn new(state: FlagState, permission: Permission, since: u32) -> Value {
Value { value, since: Some(since) } Value { state, permission, since: Some(since) }
} }
pub fn default(value: bool) -> Value { pub fn default(state: FlagState, permission: Permission) -> Value {
Value { value, since: None } Value { state, permission, since: None }
} }
} }
@ -41,10 +81,15 @@ impl TryFrom<ProtoValue> for Value {
type Error = Error; type Error = Error;
fn try_from(proto: ProtoValue) -> Result<Self, Self::Error> { fn try_from(proto: ProtoValue) -> Result<Self, Self::Error> {
let Some(value) = proto.value else { let Some(proto_state) = proto.state else {
return Err(anyhow!("missing 'value' field")); return Err(anyhow!("missing 'state' field"));
}; };
Ok(Value { value, since: proto.since }) let state = proto_state.try_into()?;
let Some(proto_permission) = proto.permission else {
return Err(anyhow!("missing 'permission' field"));
};
let permission = proto_permission.try_into()?;
Ok(Value { state, permission, since: proto.since })
} }
} }
@ -72,15 +117,17 @@ impl Flag {
proto.flag.into_iter().map(|proto_flag| proto_flag.try_into()).collect() proto.flag.into_iter().map(|proto_flag| proto_flag.try_into()).collect()
} }
pub fn resolve_value(&self, build_id: u32) -> bool { pub fn resolve(&self, build_id: u32) -> (FlagState, Permission) {
let mut value = self.values[0].value; let mut state = self.values[0].state;
let mut permission = self.values[0].permission;
for candidate in self.values.iter().skip(1) { for candidate in self.values.iter().skip(1) {
let since = candidate.since.expect("invariant: non-defaults values have Some(since)"); let since = candidate.since.expect("invariant: non-defaults values have Some(since)");
if since <= build_id { if since <= build_id {
value = candidate.value; state = candidate.state;
permission = candidate.permission;
} }
} }
value (state, permission)
} }
} }
@ -119,7 +166,8 @@ impl TryFrom<ProtoFlag> for Flag {
#[derive(Debug, PartialEq, Eq)] #[derive(Debug, PartialEq, Eq)]
pub struct Override { pub struct Override {
pub id: String, pub id: String,
pub value: bool, pub state: FlagState,
pub permission: Permission,
} }
impl Override { impl Override {
@ -142,10 +190,15 @@ impl TryFrom<ProtoOverride> for Override {
let Some(id) = proto.id else { let Some(id) = proto.id else {
return Err(anyhow!("missing 'id' field")); return Err(anyhow!("missing 'id' field"));
}; };
let Some(value) = proto.value else { let Some(proto_state) = proto.state else {
return Err(anyhow!("missing 'value' field")); return Err(anyhow!("missing 'state' field"));
}; };
Ok(Override { id, value }) let state = proto_state.try_into()?;
let Some(proto_permission) = proto.permission else {
return Err(anyhow!("missing 'permission' field"));
};
let permission = proto_permission.try_into()?;
Ok(Override { id, state, permission })
} }
} }
@ -158,17 +211,22 @@ mod tests {
let expected = Flag { let expected = Flag {
id: "1234".to_owned(), id: "1234".to_owned(),
description: "Description of the flag".to_owned(), description: "Description of the flag".to_owned(),
values: vec![Value::default(false), Value::new(true, 8)], values: vec![
Value::default(FlagState::Disabled, Permission::ReadOnly),
Value::new(FlagState::Enabled, Permission::ReadWrite, 8),
],
}; };
let s = r#" let s = r#"
id: "1234" id: "1234"
description: "Description of the flag" description: "Description of the flag"
value { value {
value: false state: DISABLED
permission: READ_ONLY
} }
value { value {
value: true state: ENABLED
permission: READ_WRITE
since: 8 since: 8
} }
"#; "#;
@ -189,7 +247,8 @@ mod tests {
let s = r#" let s = r#"
description: "Description of the flag" description: "Description of the flag"
value { value {
value: true state: ENABLED
permission: READ_ONLY
} }
"#; "#;
let error = Flag::try_from_text_proto(s).unwrap_err(); let error = Flag::try_from_text_proto(s).unwrap_err();
@ -199,10 +258,12 @@ mod tests {
id: "a" id: "a"
description: "Description of the flag" description: "Description of the flag"
value { value {
value: true state: ENABLED
permission: READ_ONLY
} }
value { value {
value: true state: ENABLED
permission: READ_ONLY
} }
"#; "#;
let error = Flag::try_from_text_proto(s).unwrap_err(); let error = Flag::try_from_text_proto(s).unwrap_err();
@ -215,12 +276,12 @@ mod tests {
Flag { Flag {
id: "a".to_owned(), id: "a".to_owned(),
description: "A".to_owned(), description: "A".to_owned(),
values: vec![Value::default(true)], values: vec![Value::default(FlagState::Enabled, Permission::ReadOnly)],
}, },
Flag { Flag {
id: "b".to_owned(), id: "b".to_owned(),
description: "B".to_owned(), description: "B".to_owned(),
values: vec![Value::default(false)], values: vec![Value::default(FlagState::Disabled, Permission::ReadWrite)],
}, },
]; ];
@ -229,14 +290,16 @@ mod tests {
id: "a" id: "a"
description: "A" description: "A"
value { value {
value: true state: ENABLED
permission: READ_ONLY
} }
} }
flag { flag {
id: "b" id: "b"
description: "B" description: "B"
value { value {
value: false state: DISABLED
permission: READ_WRITE
} }
} }
"#; "#;
@ -247,11 +310,16 @@ mod tests {
#[test] #[test]
fn test_override_try_from_text_proto_list() { fn test_override_try_from_text_proto_list() {
let expected = Override { id: "1234".to_owned(), value: true }; let expected = Override {
id: "1234".to_owned(),
state: FlagState::Enabled,
permission: Permission::ReadOnly,
};
let s = r#" let s = r#"
id: "1234" id: "1234"
value: true state: ENABLED
permission: READ_ONLY
"#; "#;
let actual = Override::try_from_text_proto(s).unwrap(); let actual = Override::try_from_text_proto(s).unwrap();
@ -259,26 +327,26 @@ mod tests {
} }
#[test] #[test]
fn test_resolve_value() { fn test_flag_resolve() {
let flag = Flag { let flag = Flag {
id: "a".to_owned(), id: "a".to_owned(),
description: "A".to_owned(), description: "A".to_owned(),
values: vec![ values: vec![
Value::default(true), Value::default(FlagState::Disabled, Permission::ReadOnly),
Value::new(false, 10), Value::new(FlagState::Disabled, Permission::ReadWrite, 10),
Value::new(true, 20), Value::new(FlagState::Enabled, Permission::ReadOnly, 20),
Value::new(false, 30), Value::new(FlagState::Enabled, Permission::ReadWrite, 30),
], ],
}; };
assert!(flag.resolve_value(0)); assert_eq!((FlagState::Disabled, Permission::ReadOnly), flag.resolve(0));
assert!(flag.resolve_value(9)); assert_eq!((FlagState::Disabled, Permission::ReadOnly), flag.resolve(9));
assert!(!flag.resolve_value(10)); assert_eq!((FlagState::Disabled, Permission::ReadWrite), flag.resolve(10));
assert!(!flag.resolve_value(11)); assert_eq!((FlagState::Disabled, Permission::ReadWrite), flag.resolve(11));
assert!(!flag.resolve_value(19)); assert_eq!((FlagState::Disabled, Permission::ReadWrite), flag.resolve(19));
assert!(flag.resolve_value(20)); assert_eq!((FlagState::Enabled, Permission::ReadOnly), flag.resolve(20));
assert!(flag.resolve_value(21)); assert_eq!((FlagState::Enabled, Permission::ReadOnly), flag.resolve(21));
assert!(flag.resolve_value(29)); assert_eq!((FlagState::Enabled, Permission::ReadOnly), flag.resolve(29));
assert!(!flag.resolve_value(30)); assert_eq!((FlagState::Enabled, Permission::ReadWrite), flag.resolve(30));
assert!(!flag.resolve_value(10_000)); assert_eq!((FlagState::Enabled, Permission::ReadWrite), flag.resolve(10_000));
} }
} }

View file

@ -18,18 +18,19 @@ use anyhow::{anyhow, Result};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use std::io::{Read, Write}; use std::io::{Read, Write};
use crate::aconfig::{Flag, Override}; use crate::aconfig::{Flag, FlagState, Override, Permission};
use crate::commands::Source; use crate::commands::Source;
#[derive(Serialize, Deserialize)] #[derive(Serialize, Deserialize, Debug)]
pub struct Item { pub struct Item {
pub id: String, pub id: String,
pub description: String, pub description: String,
pub value: bool, pub state: FlagState,
pub permission: Permission,
pub debug: Vec<String>, pub debug: Vec<String>,
} }
#[derive(Serialize, Deserialize)] #[derive(Serialize, Deserialize, Debug)]
pub struct Cache { pub struct Cache {
build_id: u32, build_id: u32,
items: Vec<Item>, items: Vec<Item>,
@ -56,12 +57,13 @@ impl Cache {
source, source,
)); ));
} }
let value = flag.resolve_value(self.build_id); let (state, permission) = flag.resolve(self.build_id);
self.items.push(Item { self.items.push(Item {
id: flag.id.clone(), id: flag.id.clone(),
description: flag.description, description: flag.description,
value, state,
debug: vec![format!("{}:{}", source, value)], permission,
debug: vec![format!("{}:{:?} {:?}", source, state, permission)],
}); });
Ok(()) Ok(())
} }
@ -70,8 +72,11 @@ impl Cache {
let Some(existing_item) = self.items.iter_mut().find(|item| item.id == override_.id) else { let Some(existing_item) = self.items.iter_mut().find(|item| item.id == override_.id) else {
return Err(anyhow!("failed to override flag {}: unknown flag", override_.id)); return Err(anyhow!("failed to override flag {}: unknown flag", override_.id));
}; };
existing_item.value = override_.value; existing_item.state = override_.state;
existing_item.debug.push(format!("{}:{}", source, override_.value)); existing_item.permission = override_.permission;
existing_item
.debug
.push(format!("{}:{:?} {:?}", source, override_.state, override_.permission));
Ok(()) Ok(())
} }
@ -85,7 +90,7 @@ impl Item {}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;
use crate::aconfig::Value; use crate::aconfig::{FlagState, Permission, Value};
#[test] #[test]
fn test_add_flag() { fn test_add_flag() {
@ -96,7 +101,7 @@ mod tests {
Flag { Flag {
id: "foo".to_string(), id: "foo".to_string(),
description: "desc".to_string(), description: "desc".to_string(),
values: vec![Value::default(true)], values: vec![Value::default(FlagState::Enabled, Permission::ReadOnly)],
}, },
) )
.unwrap(); .unwrap();
@ -106,7 +111,7 @@ mod tests {
Flag { Flag {
id: "foo".to_string(), id: "foo".to_string(),
description: "desc".to_string(), description: "desc".to_string(),
values: vec![Value::default(false)], values: vec![Value::default(FlagState::Disabled, Permission::ReadOnly)],
}, },
) )
.unwrap_err(); .unwrap_err();
@ -118,13 +123,21 @@ mod tests {
#[test] #[test]
fn test_add_override() { fn test_add_override() {
fn check_value(cache: &Cache, id: &str, expected: bool) -> bool { fn check(cache: &Cache, id: &str, expected: (FlagState, Permission)) -> bool {
cache.iter().find(|&item| item.id == id).unwrap().value == expected let item = cache.iter().find(|&item| item.id == id).unwrap();
item.state == expected.0 && item.permission == expected.1
} }
let mut cache = Cache::new(1); let mut cache = Cache::new(1);
let error = cache let error = cache
.add_override(Source::Memory, Override { id: "foo".to_string(), value: true }) .add_override(
Source::Memory,
Override {
id: "foo".to_string(),
state: FlagState::Enabled,
permission: Permission::ReadOnly,
},
)
.unwrap_err(); .unwrap_err();
assert_eq!(&format!("{:?}", error), "failed to override flag foo: unknown flag"); assert_eq!(&format!("{:?}", error), "failed to override flag foo: unknown flag");
@ -134,20 +147,36 @@ mod tests {
Flag { Flag {
id: "foo".to_string(), id: "foo".to_string(),
description: "desc".to_string(), description: "desc".to_string(),
values: vec![Value::default(true)], values: vec![Value::default(FlagState::Enabled, Permission::ReadOnly)],
}, },
) )
.unwrap(); .unwrap();
assert!(check_value(&cache, "foo", true)); dbg!(&cache);
assert!(check(&cache, "foo", (FlagState::Enabled, Permission::ReadOnly)));
cache cache
.add_override(Source::Memory, Override { id: "foo".to_string(), value: false }) .add_override(
Source::Memory,
Override {
id: "foo".to_string(),
state: FlagState::Disabled,
permission: Permission::ReadWrite,
},
)
.unwrap(); .unwrap();
assert!(check_value(&cache, "foo", false)); dbg!(&cache);
assert!(check(&cache, "foo", (FlagState::Disabled, Permission::ReadWrite)));
cache cache
.add_override(Source::Memory, Override { id: "foo".to_string(), value: true }) .add_override(
Source::Memory,
Override {
id: "foo".to_string(),
state: FlagState::Enabled,
permission: Permission::ReadWrite,
},
)
.unwrap(); .unwrap();
assert!(check_value(&cache, "foo", true)); assert!(check(&cache, "foo", (FlagState::Enabled, Permission::ReadWrite)));
} }
} }

View file

@ -16,14 +16,13 @@
use anyhow::{Context, Result}; use anyhow::{Context, Result};
use clap::ValueEnum; use clap::ValueEnum;
use serde::{Deserialize, Serialize};
use std::fmt; use std::fmt;
use std::io::Read; use std::io::Read;
use crate::aconfig::{Flag, Override}; use crate::aconfig::{Flag, Override};
use crate::cache::Cache; use crate::cache::Cache;
#[derive(Clone, Serialize, Deserialize)] #[derive(Clone)]
pub enum Source { pub enum Source {
#[allow(dead_code)] // only used in unit tests #[allow(dead_code)] // only used in unit tests
Memory, Memory,
@ -80,12 +79,12 @@ pub fn dump_cache(cache: Cache, format: Format) -> Result<()> {
match format { match format {
Format::Text => { Format::Text => {
for item in cache.iter() { for item in cache.iter() {
println!("{}: {}", item.id, item.value); println!("{}: {:?}", item.id, item.state);
} }
} }
Format::Debug => { Format::Debug => {
for item in cache.iter() { for item in cache.iter() {
println!("{}: {} ({:?})", item.id, item.value, item.debug); println!("{:?}", item);
} }
} }
} }
@ -95,6 +94,7 @@ pub fn dump_cache(cache: Cache, format: Format) -> Result<()> {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;
use crate::aconfig::{FlagState, Permission};
#[test] #[test]
fn test_create_cache() { fn test_create_cache() {
@ -103,7 +103,8 @@ mod tests {
id: "a" id: "a"
description: "Description of a" description: "Description of a"
value { value {
value: true state: ENABLED
permission: READ_WRITE
} }
} }
"#; "#;
@ -111,12 +112,14 @@ mod tests {
let o = r#" let o = r#"
override { override {
id: "a" id: "a"
value: false state: DISABLED
permission: READ_ONLY
} }
"#; "#;
let overrides = vec![Input { source: Source::Memory, reader: Box::new(o.as_bytes()) }]; let overrides = vec![Input { source: Source::Memory, reader: Box::new(o.as_bytes()) }];
let cache = create_cache(1, aconfigs, overrides).unwrap(); let cache = create_cache(1, aconfigs, overrides).unwrap();
let value = cache.iter().find(|&item| item.id == "a").unwrap().value; let item = cache.iter().find(|&item| item.id == "a").unwrap();
assert!(!value); assert_eq!(FlagState::Disabled, item.state);
assert_eq!(Permission::ReadOnly, item.permission);
} }
} }

View file

@ -42,6 +42,12 @@ pub use aconfig_protos::aconfig::Override_config as ProtoOverrideConfig;
#[cfg(not(feature = "cargo"))] #[cfg(not(feature = "cargo"))]
pub use aconfig_protos::aconfig::Override as ProtoOverride; pub use aconfig_protos::aconfig::Override as ProtoOverride;
#[cfg(not(feature = "cargo"))]
pub use aconfig_protos::aconfig::Permission as ProtoPermission;
#[cfg(not(feature = "cargo"))]
pub use aconfig_protos::aconfig::Flag_state as ProtoFlagState;
// ---- When building with cargo ---- // ---- When building with cargo ----
#[cfg(feature = "cargo")] #[cfg(feature = "cargo")]
include!(concat!(env!("OUT_DIR"), "/aconfig_proto/mod.rs")); include!(concat!(env!("OUT_DIR"), "/aconfig_proto/mod.rs"));
@ -61,6 +67,12 @@ pub use aconfig::Override_config as ProtoOverrideConfig;
#[cfg(feature = "cargo")] #[cfg(feature = "cargo")]
pub use aconfig::Override as ProtoOverride; pub use aconfig::Override as ProtoOverride;
#[cfg(feature = "cargo")]
pub use aconfig::Permission as ProtoPermission;
#[cfg(feature = "cargo")]
pub use aconfig::Flag_state as ProtoFlagState;
// ---- Common for both the Android tool-chain and cargo ---- // ---- Common for both the Android tool-chain and cargo ----
use anyhow::Result; use anyhow::Result;