Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2023 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
| 16 | |
| 17 | use anyhow::{anyhow, Context, Error, Result}; |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 18 | use protobuf::{Enum, EnumOrUnknown}; |
| 19 | use serde::{Deserialize, Serialize}; |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 20 | |
Mårten Kongstad | a102909 | 2023-05-08 11:51:59 +0200 | [diff] [blame^] | 21 | use crate::cache::{Cache, Item, TracePoint}; |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 22 | use crate::protos::{ |
Mårten Kongstad | a102909 | 2023-05-08 11:51:59 +0200 | [diff] [blame^] | 23 | ProtoAndroidConfig, ProtoDump, ProtoDumpItem, ProtoDumpTracePoint, ProtoFlag, ProtoFlagState, |
| 24 | ProtoOverride, ProtoOverrideConfig, ProtoPermission, ProtoValue, |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 25 | }; |
| 26 | |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 27 | #[derive(Debug, PartialEq, Eq, Serialize, Deserialize, Clone, Copy)] |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 28 | pub enum FlagState { |
| 29 | Enabled, |
| 30 | Disabled, |
| 31 | } |
| 32 | |
| 33 | impl TryFrom<EnumOrUnknown<ProtoFlagState>> for FlagState { |
| 34 | type Error = Error; |
| 35 | |
| 36 | fn try_from(proto: EnumOrUnknown<ProtoFlagState>) -> Result<Self, Self::Error> { |
| 37 | match ProtoFlagState::from_i32(proto.value()) { |
| 38 | Some(ProtoFlagState::ENABLED) => Ok(FlagState::Enabled), |
| 39 | Some(ProtoFlagState::DISABLED) => Ok(FlagState::Disabled), |
| 40 | None => Err(anyhow!("unknown flag state enum value {}", proto.value())), |
| 41 | } |
| 42 | } |
| 43 | } |
| 44 | |
Mårten Kongstad | a102909 | 2023-05-08 11:51:59 +0200 | [diff] [blame^] | 45 | impl From<FlagState> for ProtoFlagState { |
| 46 | fn from(state: FlagState) -> Self { |
| 47 | match state { |
| 48 | FlagState::Enabled => ProtoFlagState::ENABLED, |
| 49 | FlagState::Disabled => ProtoFlagState::DISABLED, |
| 50 | } |
| 51 | } |
| 52 | } |
| 53 | |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 54 | #[derive(Debug, PartialEq, Eq, Serialize, Deserialize, Clone, Copy)] |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 55 | pub enum Permission { |
| 56 | ReadOnly, |
| 57 | ReadWrite, |
| 58 | } |
| 59 | |
| 60 | impl TryFrom<EnumOrUnknown<ProtoPermission>> for Permission { |
| 61 | type Error = Error; |
| 62 | |
| 63 | fn try_from(proto: EnumOrUnknown<ProtoPermission>) -> Result<Self, Self::Error> { |
| 64 | match ProtoPermission::from_i32(proto.value()) { |
| 65 | Some(ProtoPermission::READ_ONLY) => Ok(Permission::ReadOnly), |
| 66 | Some(ProtoPermission::READ_WRITE) => Ok(Permission::ReadWrite), |
| 67 | None => Err(anyhow!("unknown permission enum value {}", proto.value())), |
| 68 | } |
| 69 | } |
| 70 | } |
| 71 | |
Mårten Kongstad | a102909 | 2023-05-08 11:51:59 +0200 | [diff] [blame^] | 72 | impl From<Permission> for ProtoPermission { |
| 73 | fn from(permission: Permission) -> Self { |
| 74 | match permission { |
| 75 | Permission::ReadOnly => ProtoPermission::READ_ONLY, |
| 76 | Permission::ReadWrite => ProtoPermission::READ_WRITE, |
| 77 | } |
| 78 | } |
| 79 | } |
| 80 | |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 81 | #[derive(Debug, PartialEq, Eq)] |
| 82 | pub struct Value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 83 | state: FlagState, |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 84 | permission: Permission, |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 85 | since: Option<u32>, |
| 86 | } |
| 87 | |
| 88 | #[allow(dead_code)] // only used in unit tests |
| 89 | impl Value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 90 | pub fn new(state: FlagState, permission: Permission, since: u32) -> Value { |
| 91 | Value { state, permission, since: Some(since) } |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 92 | } |
| 93 | |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 94 | pub fn default(state: FlagState, permission: Permission) -> Value { |
| 95 | Value { state, permission, since: None } |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 96 | } |
| 97 | } |
| 98 | |
| 99 | impl TryFrom<ProtoValue> for Value { |
| 100 | type Error = Error; |
| 101 | |
| 102 | fn try_from(proto: ProtoValue) -> Result<Self, Self::Error> { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 103 | let Some(proto_state) = proto.state else { |
| 104 | return Err(anyhow!("missing 'state' field")); |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 105 | }; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 106 | let state = proto_state.try_into()?; |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 107 | let Some(proto_permission) = proto.permission else { |
| 108 | return Err(anyhow!("missing 'permission' field")); |
| 109 | }; |
| 110 | let permission = proto_permission.try_into()?; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 111 | Ok(Value { state, permission, since: proto.since }) |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 112 | } |
| 113 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 114 | |
| 115 | #[derive(Debug, PartialEq, Eq)] |
| 116 | pub struct Flag { |
| 117 | pub id: String, |
| 118 | pub description: String, |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 119 | |
| 120 | // ordered by Value.since; guaranteed to contain at least one item (the default value, with |
| 121 | // since == None) |
| 122 | pub values: Vec<Value>, |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 123 | } |
| 124 | |
| 125 | impl Flag { |
Mårten Kongstad | 4d2b4b0 | 2023-04-27 16:05:58 +0200 | [diff] [blame] | 126 | #[allow(dead_code)] // only used in unit tests |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 127 | pub fn try_from_text_proto(text_proto: &str) -> Result<Flag> { |
| 128 | let proto: ProtoFlag = crate::protos::try_from_text_proto(text_proto) |
| 129 | .with_context(|| text_proto.to_owned())?; |
| 130 | proto.try_into() |
| 131 | } |
| 132 | |
| 133 | pub fn try_from_text_proto_list(text_proto: &str) -> Result<Vec<Flag>> { |
| 134 | let proto: ProtoAndroidConfig = crate::protos::try_from_text_proto(text_proto) |
| 135 | .with_context(|| text_proto.to_owned())?; |
| 136 | proto.flag.into_iter().map(|proto_flag| proto_flag.try_into()).collect() |
| 137 | } |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 138 | |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 139 | pub fn resolve(&self, build_id: u32) -> (FlagState, Permission) { |
| 140 | let mut state = self.values[0].state; |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 141 | let mut permission = self.values[0].permission; |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 142 | for candidate in self.values.iter().skip(1) { |
| 143 | let since = candidate.since.expect("invariant: non-defaults values have Some(since)"); |
| 144 | if since <= build_id { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 145 | state = candidate.state; |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 146 | permission = candidate.permission; |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 147 | } |
| 148 | } |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 149 | (state, permission) |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 150 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 151 | } |
| 152 | |
| 153 | impl TryFrom<ProtoFlag> for Flag { |
| 154 | type Error = Error; |
| 155 | |
| 156 | fn try_from(proto: ProtoFlag) -> Result<Self, Self::Error> { |
| 157 | let Some(id) = proto.id else { |
| 158 | return Err(anyhow!("missing 'id' field")); |
| 159 | }; |
| 160 | let Some(description) = proto.description else { |
| 161 | return Err(anyhow!("missing 'description' field")); |
| 162 | }; |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 163 | if proto.value.is_empty() { |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 164 | return Err(anyhow!("missing 'value' field")); |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 165 | } |
| 166 | |
| 167 | let mut values: Vec<Value> = vec![]; |
| 168 | for proto_value in proto.value.into_iter() { |
| 169 | let v: Value = proto_value.try_into()?; |
| 170 | if values.iter().any(|w| v.since == w.since) { |
| 171 | let msg = match v.since { |
| 172 | None => format!("flag {}: multiple default values", id), |
| 173 | Some(x) => format!("flag {}: multiple values for since={}", id, x), |
| 174 | }; |
| 175 | return Err(anyhow!(msg)); |
| 176 | } |
| 177 | values.push(v); |
| 178 | } |
| 179 | values.sort_by_key(|v| v.since); |
| 180 | |
| 181 | Ok(Flag { id, description, values }) |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 182 | } |
| 183 | } |
| 184 | |
| 185 | #[derive(Debug, PartialEq, Eq)] |
| 186 | pub struct Override { |
| 187 | pub id: String, |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 188 | pub state: FlagState, |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 189 | pub permission: Permission, |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 190 | } |
| 191 | |
| 192 | impl Override { |
Mårten Kongstad | 4d2b4b0 | 2023-04-27 16:05:58 +0200 | [diff] [blame] | 193 | #[allow(dead_code)] // only used in unit tests |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 194 | pub fn try_from_text_proto(text_proto: &str) -> Result<Override> { |
| 195 | let proto: ProtoOverride = crate::protos::try_from_text_proto(text_proto)?; |
| 196 | proto.try_into() |
| 197 | } |
| 198 | |
| 199 | pub fn try_from_text_proto_list(text_proto: &str) -> Result<Vec<Override>> { |
| 200 | let proto: ProtoOverrideConfig = crate::protos::try_from_text_proto(text_proto)?; |
| 201 | proto.override_.into_iter().map(|proto_flag| proto_flag.try_into()).collect() |
| 202 | } |
| 203 | } |
| 204 | |
| 205 | impl TryFrom<ProtoOverride> for Override { |
| 206 | type Error = Error; |
| 207 | |
| 208 | fn try_from(proto: ProtoOverride) -> Result<Self, Self::Error> { |
| 209 | let Some(id) = proto.id else { |
| 210 | return Err(anyhow!("missing 'id' field")); |
| 211 | }; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 212 | let Some(proto_state) = proto.state else { |
| 213 | return Err(anyhow!("missing 'state' field")); |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 214 | }; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 215 | let state = proto_state.try_into()?; |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 216 | let Some(proto_permission) = proto.permission else { |
| 217 | return Err(anyhow!("missing 'permission' field")); |
| 218 | }; |
| 219 | let permission = proto_permission.try_into()?; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 220 | Ok(Override { id, state, permission }) |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 221 | } |
| 222 | } |
| 223 | |
Mårten Kongstad | a102909 | 2023-05-08 11:51:59 +0200 | [diff] [blame^] | 224 | impl From<Cache> for ProtoDump { |
| 225 | fn from(cache: Cache) -> Self { |
| 226 | let mut dump = ProtoDump::new(); |
| 227 | for item in cache.into_iter() { |
| 228 | dump.item.push(item.into()); |
| 229 | } |
| 230 | dump |
| 231 | } |
| 232 | } |
| 233 | |
| 234 | impl From<Item> for ProtoDumpItem { |
| 235 | fn from(item: Item) -> Self { |
| 236 | let mut dump_item = crate::protos::ProtoDumpItem::new(); |
| 237 | dump_item.set_id(item.id.clone()); |
| 238 | dump_item.set_description(item.description.clone()); |
| 239 | dump_item.set_state(item.state.into()); |
| 240 | dump_item.set_permission(item.permission.into()); |
| 241 | for trace in item.trace.into_iter() { |
| 242 | dump_item.trace.push(trace.into()); |
| 243 | } |
| 244 | dump_item |
| 245 | } |
| 246 | } |
| 247 | |
| 248 | impl From<TracePoint> for ProtoDumpTracePoint { |
| 249 | fn from(tracepoint: TracePoint) -> Self { |
| 250 | let mut dump_tracepoint = ProtoDumpTracePoint::new(); |
| 251 | dump_tracepoint.set_source(format!("{}", tracepoint.source)); |
| 252 | dump_tracepoint.set_state(tracepoint.state.into()); |
| 253 | dump_tracepoint.set_permission(tracepoint.permission.into()); |
| 254 | dump_tracepoint |
| 255 | } |
| 256 | } |
| 257 | |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 258 | #[cfg(test)] |
| 259 | mod tests { |
| 260 | use super::*; |
| 261 | |
| 262 | #[test] |
| 263 | fn test_flag_try_from_text_proto() { |
| 264 | let expected = Flag { |
| 265 | id: "1234".to_owned(), |
| 266 | description: "Description of the flag".to_owned(), |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 267 | values: vec![ |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 268 | Value::default(FlagState::Disabled, Permission::ReadOnly), |
| 269 | Value::new(FlagState::Enabled, Permission::ReadWrite, 8), |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 270 | ], |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 271 | }; |
| 272 | |
| 273 | let s = r#" |
| 274 | id: "1234" |
| 275 | description: "Description of the flag" |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 276 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 277 | state: DISABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 278 | permission: READ_ONLY |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 279 | } |
| 280 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 281 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 282 | permission: READ_WRITE |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 283 | since: 8 |
| 284 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 285 | "#; |
| 286 | let actual = Flag::try_from_text_proto(s).unwrap(); |
| 287 | |
| 288 | assert_eq!(expected, actual); |
| 289 | } |
| 290 | |
| 291 | #[test] |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 292 | fn test_flag_try_from_text_proto_bad_input() { |
| 293 | let s = r#" |
| 294 | id: "a" |
| 295 | description: "Description of the flag" |
| 296 | "#; |
| 297 | let error = Flag::try_from_text_proto(s).unwrap_err(); |
| 298 | assert_eq!(format!("{:?}", error), "missing 'value' field"); |
| 299 | |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 300 | let s = r#" |
| 301 | description: "Description of the flag" |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 302 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 303 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 304 | permission: READ_ONLY |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 305 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 306 | "#; |
| 307 | let error = Flag::try_from_text_proto(s).unwrap_err(); |
| 308 | assert!(format!("{:?}", error).contains("Message not initialized")); |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 309 | |
| 310 | let s = r#" |
| 311 | id: "a" |
| 312 | description: "Description of the flag" |
| 313 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 314 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 315 | permission: READ_ONLY |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 316 | } |
| 317 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 318 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 319 | permission: READ_ONLY |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 320 | } |
| 321 | "#; |
| 322 | let error = Flag::try_from_text_proto(s).unwrap_err(); |
| 323 | assert_eq!(format!("{:?}", error), "flag a: multiple default values"); |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 324 | } |
| 325 | |
| 326 | #[test] |
| 327 | fn test_flag_try_from_text_proto_list() { |
| 328 | let expected = vec![ |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 329 | Flag { |
| 330 | id: "a".to_owned(), |
| 331 | description: "A".to_owned(), |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 332 | values: vec![Value::default(FlagState::Enabled, Permission::ReadOnly)], |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 333 | }, |
| 334 | Flag { |
| 335 | id: "b".to_owned(), |
| 336 | description: "B".to_owned(), |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 337 | values: vec![Value::default(FlagState::Disabled, Permission::ReadWrite)], |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 338 | }, |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 339 | ]; |
| 340 | |
| 341 | let s = r#" |
| 342 | flag { |
| 343 | id: "a" |
| 344 | description: "A" |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 345 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 346 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 347 | permission: READ_ONLY |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 348 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 349 | } |
| 350 | flag { |
| 351 | id: "b" |
| 352 | description: "B" |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 353 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 354 | state: DISABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 355 | permission: READ_WRITE |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 356 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 357 | } |
| 358 | "#; |
| 359 | let actual = Flag::try_from_text_proto_list(s).unwrap(); |
| 360 | |
| 361 | assert_eq!(expected, actual); |
| 362 | } |
| 363 | |
| 364 | #[test] |
| 365 | fn test_override_try_from_text_proto_list() { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 366 | let expected = Override { |
| 367 | id: "1234".to_owned(), |
| 368 | state: FlagState::Enabled, |
| 369 | permission: Permission::ReadOnly, |
| 370 | }; |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 371 | |
| 372 | let s = r#" |
| 373 | id: "1234" |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 374 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 375 | permission: READ_ONLY |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 376 | "#; |
| 377 | let actual = Override::try_from_text_proto(s).unwrap(); |
| 378 | |
| 379 | assert_eq!(expected, actual); |
| 380 | } |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 381 | |
| 382 | #[test] |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 383 | fn test_flag_resolve() { |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 384 | let flag = Flag { |
| 385 | id: "a".to_owned(), |
| 386 | description: "A".to_owned(), |
| 387 | values: vec![ |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 388 | Value::default(FlagState::Disabled, Permission::ReadOnly), |
| 389 | Value::new(FlagState::Disabled, Permission::ReadWrite, 10), |
| 390 | Value::new(FlagState::Enabled, Permission::ReadOnly, 20), |
| 391 | Value::new(FlagState::Enabled, Permission::ReadWrite, 30), |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 392 | ], |
| 393 | }; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 394 | assert_eq!((FlagState::Disabled, Permission::ReadOnly), flag.resolve(0)); |
| 395 | assert_eq!((FlagState::Disabled, Permission::ReadOnly), flag.resolve(9)); |
| 396 | assert_eq!((FlagState::Disabled, Permission::ReadWrite), flag.resolve(10)); |
| 397 | assert_eq!((FlagState::Disabled, Permission::ReadWrite), flag.resolve(11)); |
| 398 | assert_eq!((FlagState::Disabled, Permission::ReadWrite), flag.resolve(19)); |
| 399 | assert_eq!((FlagState::Enabled, Permission::ReadOnly), flag.resolve(20)); |
| 400 | assert_eq!((FlagState::Enabled, Permission::ReadOnly), flag.resolve(21)); |
| 401 | assert_eq!((FlagState::Enabled, Permission::ReadOnly), flag.resolve(29)); |
| 402 | assert_eq!((FlagState::Enabled, Permission::ReadWrite), flag.resolve(30)); |
| 403 | assert_eq!((FlagState::Enabled, Permission::ReadWrite), flag.resolve(10_000)); |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 404 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 405 | } |