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