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 | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 21 | use crate::protos::{ |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 22 | ProtoAndroidConfig, ProtoFlag, ProtoFlagState, ProtoOverride, ProtoOverrideConfig, |
| 23 | ProtoPermission, ProtoValue, |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 24 | }; |
| 25 | |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 26 | #[derive(Debug, PartialEq, Eq, Serialize, Deserialize, Clone, Copy)] |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 27 | pub enum FlagState { |
| 28 | Enabled, |
| 29 | Disabled, |
| 30 | } |
| 31 | |
| 32 | impl TryFrom<EnumOrUnknown<ProtoFlagState>> for FlagState { |
| 33 | type Error = Error; |
| 34 | |
| 35 | fn try_from(proto: EnumOrUnknown<ProtoFlagState>) -> Result<Self, Self::Error> { |
| 36 | match ProtoFlagState::from_i32(proto.value()) { |
| 37 | Some(ProtoFlagState::ENABLED) => Ok(FlagState::Enabled), |
| 38 | Some(ProtoFlagState::DISABLED) => Ok(FlagState::Disabled), |
| 39 | None => Err(anyhow!("unknown flag state enum value {}", proto.value())), |
| 40 | } |
| 41 | } |
| 42 | } |
| 43 | |
| 44 | #[derive(Debug, PartialEq, Eq, Serialize, Deserialize, Clone, Copy)] |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 45 | pub enum Permission { |
| 46 | ReadOnly, |
| 47 | ReadWrite, |
| 48 | } |
| 49 | |
| 50 | impl TryFrom<EnumOrUnknown<ProtoPermission>> for Permission { |
| 51 | type Error = Error; |
| 52 | |
| 53 | fn try_from(proto: EnumOrUnknown<ProtoPermission>) -> Result<Self, Self::Error> { |
| 54 | match ProtoPermission::from_i32(proto.value()) { |
| 55 | Some(ProtoPermission::READ_ONLY) => Ok(Permission::ReadOnly), |
| 56 | Some(ProtoPermission::READ_WRITE) => Ok(Permission::ReadWrite), |
| 57 | None => Err(anyhow!("unknown permission enum value {}", proto.value())), |
| 58 | } |
| 59 | } |
| 60 | } |
| 61 | |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 62 | #[derive(Debug, PartialEq, Eq)] |
| 63 | pub struct Value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 64 | state: FlagState, |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 65 | permission: Permission, |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 66 | since: Option<u32>, |
| 67 | } |
| 68 | |
| 69 | #[allow(dead_code)] // only used in unit tests |
| 70 | impl Value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 71 | pub fn new(state: FlagState, permission: Permission, since: u32) -> Value { |
| 72 | Value { state, permission, since: Some(since) } |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 73 | } |
| 74 | |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 75 | pub fn default(state: FlagState, permission: Permission) -> Value { |
| 76 | Value { state, permission, since: None } |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 77 | } |
| 78 | } |
| 79 | |
| 80 | impl TryFrom<ProtoValue> for Value { |
| 81 | type Error = Error; |
| 82 | |
| 83 | fn try_from(proto: ProtoValue) -> Result<Self, Self::Error> { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 84 | let Some(proto_state) = proto.state else { |
| 85 | return Err(anyhow!("missing 'state' field")); |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 86 | }; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 87 | let state = proto_state.try_into()?; |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 88 | let Some(proto_permission) = proto.permission else { |
| 89 | return Err(anyhow!("missing 'permission' field")); |
| 90 | }; |
| 91 | let permission = proto_permission.try_into()?; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 92 | Ok(Value { state, permission, since: proto.since }) |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 93 | } |
| 94 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 95 | |
| 96 | #[derive(Debug, PartialEq, Eq)] |
| 97 | pub struct Flag { |
| 98 | pub id: String, |
| 99 | pub description: String, |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 100 | |
| 101 | // ordered by Value.since; guaranteed to contain at least one item (the default value, with |
| 102 | // since == None) |
| 103 | pub values: Vec<Value>, |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 104 | } |
| 105 | |
| 106 | impl Flag { |
Mårten Kongstad | 4d2b4b0 | 2023-04-27 16:05:58 +0200 | [diff] [blame] | 107 | #[allow(dead_code)] // only used in unit tests |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 108 | pub fn try_from_text_proto(text_proto: &str) -> Result<Flag> { |
| 109 | let proto: ProtoFlag = crate::protos::try_from_text_proto(text_proto) |
| 110 | .with_context(|| text_proto.to_owned())?; |
| 111 | proto.try_into() |
| 112 | } |
| 113 | |
| 114 | pub fn try_from_text_proto_list(text_proto: &str) -> Result<Vec<Flag>> { |
| 115 | let proto: ProtoAndroidConfig = crate::protos::try_from_text_proto(text_proto) |
| 116 | .with_context(|| text_proto.to_owned())?; |
| 117 | proto.flag.into_iter().map(|proto_flag| proto_flag.try_into()).collect() |
| 118 | } |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 119 | |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 120 | pub fn resolve(&self, build_id: u32) -> (FlagState, Permission) { |
| 121 | let mut state = self.values[0].state; |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 122 | let mut permission = self.values[0].permission; |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 123 | for candidate in self.values.iter().skip(1) { |
| 124 | let since = candidate.since.expect("invariant: non-defaults values have Some(since)"); |
| 125 | if since <= build_id { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 126 | state = candidate.state; |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 127 | permission = candidate.permission; |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 128 | } |
| 129 | } |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 130 | (state, permission) |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 131 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 132 | } |
| 133 | |
| 134 | impl TryFrom<ProtoFlag> for Flag { |
| 135 | type Error = Error; |
| 136 | |
| 137 | fn try_from(proto: ProtoFlag) -> Result<Self, Self::Error> { |
| 138 | let Some(id) = proto.id else { |
| 139 | return Err(anyhow!("missing 'id' field")); |
| 140 | }; |
| 141 | let Some(description) = proto.description else { |
| 142 | return Err(anyhow!("missing 'description' field")); |
| 143 | }; |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 144 | if proto.value.is_empty() { |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 145 | return Err(anyhow!("missing 'value' field")); |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 146 | } |
| 147 | |
| 148 | let mut values: Vec<Value> = vec![]; |
| 149 | for proto_value in proto.value.into_iter() { |
| 150 | let v: Value = proto_value.try_into()?; |
| 151 | if values.iter().any(|w| v.since == w.since) { |
| 152 | let msg = match v.since { |
| 153 | None => format!("flag {}: multiple default values", id), |
| 154 | Some(x) => format!("flag {}: multiple values for since={}", id, x), |
| 155 | }; |
| 156 | return Err(anyhow!(msg)); |
| 157 | } |
| 158 | values.push(v); |
| 159 | } |
| 160 | values.sort_by_key(|v| v.since); |
| 161 | |
| 162 | Ok(Flag { id, description, values }) |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 163 | } |
| 164 | } |
| 165 | |
| 166 | #[derive(Debug, PartialEq, Eq)] |
| 167 | pub struct Override { |
| 168 | pub id: String, |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 169 | pub state: FlagState, |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 170 | pub permission: Permission, |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 171 | } |
| 172 | |
| 173 | impl Override { |
Mårten Kongstad | 4d2b4b0 | 2023-04-27 16:05:58 +0200 | [diff] [blame] | 174 | #[allow(dead_code)] // only used in unit tests |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 175 | pub fn try_from_text_proto(text_proto: &str) -> Result<Override> { |
| 176 | let proto: ProtoOverride = crate::protos::try_from_text_proto(text_proto)?; |
| 177 | proto.try_into() |
| 178 | } |
| 179 | |
| 180 | pub fn try_from_text_proto_list(text_proto: &str) -> Result<Vec<Override>> { |
| 181 | let proto: ProtoOverrideConfig = crate::protos::try_from_text_proto(text_proto)?; |
| 182 | proto.override_.into_iter().map(|proto_flag| proto_flag.try_into()).collect() |
| 183 | } |
| 184 | } |
| 185 | |
| 186 | impl TryFrom<ProtoOverride> for Override { |
| 187 | type Error = Error; |
| 188 | |
| 189 | fn try_from(proto: ProtoOverride) -> Result<Self, Self::Error> { |
| 190 | let Some(id) = proto.id else { |
| 191 | return Err(anyhow!("missing 'id' field")); |
| 192 | }; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 193 | let Some(proto_state) = proto.state else { |
| 194 | return Err(anyhow!("missing 'state' field")); |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 195 | }; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 196 | let state = proto_state.try_into()?; |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 197 | let Some(proto_permission) = proto.permission else { |
| 198 | return Err(anyhow!("missing 'permission' field")); |
| 199 | }; |
| 200 | let permission = proto_permission.try_into()?; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 201 | Ok(Override { id, state, permission }) |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 202 | } |
| 203 | } |
| 204 | |
| 205 | #[cfg(test)] |
| 206 | mod tests { |
| 207 | use super::*; |
| 208 | |
| 209 | #[test] |
| 210 | fn test_flag_try_from_text_proto() { |
| 211 | let expected = Flag { |
| 212 | id: "1234".to_owned(), |
| 213 | description: "Description of the flag".to_owned(), |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 214 | values: vec![ |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 215 | Value::default(FlagState::Disabled, Permission::ReadOnly), |
| 216 | Value::new(FlagState::Enabled, Permission::ReadWrite, 8), |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 217 | ], |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 218 | }; |
| 219 | |
| 220 | let s = r#" |
| 221 | id: "1234" |
| 222 | description: "Description of the flag" |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 223 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 224 | state: DISABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 225 | permission: READ_ONLY |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 226 | } |
| 227 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 228 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 229 | permission: READ_WRITE |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 230 | since: 8 |
| 231 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 232 | "#; |
| 233 | let actual = Flag::try_from_text_proto(s).unwrap(); |
| 234 | |
| 235 | assert_eq!(expected, actual); |
| 236 | } |
| 237 | |
| 238 | #[test] |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 239 | fn test_flag_try_from_text_proto_bad_input() { |
| 240 | let s = r#" |
| 241 | id: "a" |
| 242 | description: "Description of the flag" |
| 243 | "#; |
| 244 | let error = Flag::try_from_text_proto(s).unwrap_err(); |
| 245 | assert_eq!(format!("{:?}", error), "missing 'value' field"); |
| 246 | |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 247 | let s = r#" |
| 248 | description: "Description of the flag" |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 249 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 250 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 251 | permission: READ_ONLY |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 252 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 253 | "#; |
| 254 | let error = Flag::try_from_text_proto(s).unwrap_err(); |
| 255 | assert!(format!("{:?}", error).contains("Message not initialized")); |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 256 | |
| 257 | let s = r#" |
| 258 | id: "a" |
| 259 | description: "Description of the flag" |
| 260 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 261 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 262 | permission: READ_ONLY |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 263 | } |
| 264 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 265 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 266 | permission: READ_ONLY |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 267 | } |
| 268 | "#; |
| 269 | let error = Flag::try_from_text_proto(s).unwrap_err(); |
| 270 | assert_eq!(format!("{:?}", error), "flag a: multiple default values"); |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 271 | } |
| 272 | |
| 273 | #[test] |
| 274 | fn test_flag_try_from_text_proto_list() { |
| 275 | let expected = vec![ |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 276 | Flag { |
| 277 | id: "a".to_owned(), |
| 278 | description: "A".to_owned(), |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 279 | values: vec![Value::default(FlagState::Enabled, Permission::ReadOnly)], |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 280 | }, |
| 281 | Flag { |
| 282 | id: "b".to_owned(), |
| 283 | description: "B".to_owned(), |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 284 | values: vec![Value::default(FlagState::Disabled, Permission::ReadWrite)], |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 285 | }, |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 286 | ]; |
| 287 | |
| 288 | let s = r#" |
| 289 | flag { |
| 290 | id: "a" |
| 291 | description: "A" |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 292 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 293 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 294 | permission: READ_ONLY |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 295 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 296 | } |
| 297 | flag { |
| 298 | id: "b" |
| 299 | description: "B" |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 300 | value { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 301 | state: DISABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 302 | permission: READ_WRITE |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 303 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 304 | } |
| 305 | "#; |
| 306 | let actual = Flag::try_from_text_proto_list(s).unwrap(); |
| 307 | |
| 308 | assert_eq!(expected, actual); |
| 309 | } |
| 310 | |
| 311 | #[test] |
| 312 | fn test_override_try_from_text_proto_list() { |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 313 | let expected = Override { |
| 314 | id: "1234".to_owned(), |
| 315 | state: FlagState::Enabled, |
| 316 | permission: Permission::ReadOnly, |
| 317 | }; |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 318 | |
| 319 | let s = r#" |
| 320 | id: "1234" |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 321 | state: ENABLED |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 322 | permission: READ_ONLY |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 323 | "#; |
| 324 | let actual = Override::try_from_text_proto(s).unwrap(); |
| 325 | |
| 326 | assert_eq!(expected, actual); |
| 327 | } |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 328 | |
| 329 | #[test] |
Mårten Kongstad | 416330b | 2023-05-05 11:10:01 +0200 | [diff] [blame] | 330 | fn test_flag_resolve() { |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 331 | let flag = Flag { |
| 332 | id: "a".to_owned(), |
| 333 | description: "A".to_owned(), |
| 334 | values: vec![ |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 335 | Value::default(FlagState::Disabled, Permission::ReadOnly), |
| 336 | Value::new(FlagState::Disabled, Permission::ReadWrite, 10), |
| 337 | Value::new(FlagState::Enabled, Permission::ReadOnly, 20), |
| 338 | Value::new(FlagState::Enabled, Permission::ReadWrite, 30), |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 339 | ], |
| 340 | }; |
Mårten Kongstad | c68c4ea | 2023-05-05 16:20:09 +0200 | [diff] [blame] | 341 | assert_eq!((FlagState::Disabled, Permission::ReadOnly), flag.resolve(0)); |
| 342 | assert_eq!((FlagState::Disabled, Permission::ReadOnly), flag.resolve(9)); |
| 343 | assert_eq!((FlagState::Disabled, Permission::ReadWrite), flag.resolve(10)); |
| 344 | assert_eq!((FlagState::Disabled, Permission::ReadWrite), flag.resolve(11)); |
| 345 | assert_eq!((FlagState::Disabled, Permission::ReadWrite), flag.resolve(19)); |
| 346 | assert_eq!((FlagState::Enabled, Permission::ReadOnly), flag.resolve(20)); |
| 347 | assert_eq!((FlagState::Enabled, Permission::ReadOnly), flag.resolve(21)); |
| 348 | assert_eq!((FlagState::Enabled, Permission::ReadOnly), flag.resolve(29)); |
| 349 | assert_eq!((FlagState::Enabled, Permission::ReadWrite), flag.resolve(30)); |
| 350 | assert_eq!((FlagState::Enabled, Permission::ReadWrite), flag.resolve(10_000)); |
Mårten Kongstad | 09c28d1 | 2023-05-04 13:29:26 +0200 | [diff] [blame] | 351 | } |
Mårten Kongstad | bb52072 | 2023-04-26 13:16:41 +0200 | [diff] [blame] | 352 | } |