blob: ab5b0f2e25021930470ce9d9a8fad77d10d4d67c [file] [log] [blame]
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +02001/*
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
Mårten Kongstad403658f2023-06-14 09:51:56 +020017use anyhow::{bail, ensure, Context, Result};
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +020018use clap::ValueEnum;
Mårten Kongstada1029092023-05-08 11:51:59 +020019use protobuf::Message;
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +020020use std::io::Read;
Mårten Kongstadd42eeeb2023-05-12 10:01:00 +020021use std::path::PathBuf;
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +020022
Dennis Shen1dc9ad42023-05-12 00:21:55 +000023use crate::codegen_cpp::generate_cpp_code;
Mårten Kongstadd42eeeb2023-05-12 10:01:00 +020024use crate::codegen_java::generate_java_code;
Mårten Kongstadf73b9632023-05-24 15:43:47 +020025use crate::codegen_rust::generate_rust_code;
Mårten Kongstad403658f2023-06-14 09:51:56 +020026use crate::protos::{
27 ProtoFlagPermission, ProtoFlagState, ProtoParsedFlag, ProtoParsedFlags, ProtoTracepoint,
28};
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +020029
30pub struct Input {
Mårten Kongstad403658f2023-06-14 09:51:56 +020031 pub source: String,
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +020032 pub reader: Box<dyn Read>,
33}
34
Mårten Kongstad403658f2023-06-14 09:51:56 +020035impl Input {
36 fn try_parse_flags(&mut self) -> Result<ProtoParsedFlags> {
37 let mut buffer = Vec::new();
Mårten Kongstadcd414d4c2023-07-27 14:25:33 +020038 self.reader
39 .read_to_end(&mut buffer)
40 .with_context(|| format!("failed to read {}", self.source))?;
Mårten Kongstad403658f2023-06-14 09:51:56 +020041 crate::protos::parsed_flags::try_from_binary_proto(&buffer)
Mårten Kongstadcd414d4c2023-07-27 14:25:33 +020042 .with_context(|| self.error_context())
43 }
44
45 fn error_context(&self) -> String {
46 format!("failed to parse {}", self.source)
Mårten Kongstad403658f2023-06-14 09:51:56 +020047 }
48}
49
Mårten Kongstadd42eeeb2023-05-12 10:01:00 +020050pub struct OutputFile {
51 pub path: PathBuf, // relative to some root directory only main knows about
52 pub contents: Vec<u8>,
53}
54
Zhi Dou24a0b6a2023-08-10 21:39:59 +000055pub const DEFAULT_FLAG_STATE: ProtoFlagState = ProtoFlagState::DISABLED;
56pub const DEFAULT_FLAG_PERMISSION: ProtoFlagPermission = ProtoFlagPermission::READ_WRITE;
Mårten Kongstad403658f2023-06-14 09:51:56 +020057
Zhi Dou24a0b6a2023-08-10 21:39:59 +000058pub fn parse_flags(
59 package: &str,
60 declarations: Vec<Input>,
61 values: Vec<Input>,
62 default_permission: ProtoFlagPermission,
63) -> Result<Vec<u8>> {
Mårten Kongstad403658f2023-06-14 09:51:56 +020064 let mut parsed_flags = ProtoParsedFlags::new();
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +020065
Mårten Kongstadfa23d292023-05-11 14:47:02 +020066 for mut input in declarations {
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +020067 let mut contents = String::new();
Mårten Kongstadcd414d4c2023-07-27 14:25:33 +020068 input
69 .reader
70 .read_to_string(&mut contents)
71 .with_context(|| format!("failed to read {}", input.source))?;
Mårten Kongstad403658f2023-06-14 09:51:56 +020072
73 let flag_declarations = crate::protos::flag_declarations::try_from_text_proto(&contents)
Mårten Kongstadcd414d4c2023-07-27 14:25:33 +020074 .with_context(|| input.error_context())?;
Mårten Kongstad30950782023-05-09 13:31:29 +020075 ensure!(
Mårten Kongstad403658f2023-06-14 09:51:56 +020076 package == flag_declarations.package(),
Mårten Kongstadcd414d4c2023-07-27 14:25:33 +020077 "failed to parse {}: expected package {}, got {}",
Mårten Kongstad30950782023-05-09 13:31:29 +020078 input.source,
Mårten Kongstad9fb58962023-05-31 13:02:13 +020079 package,
Mårten Kongstad403658f2023-06-14 09:51:56 +020080 flag_declarations.package()
Mårten Kongstad30950782023-05-09 13:31:29 +020081 );
Mårten Kongstad403658f2023-06-14 09:51:56 +020082 for mut flag_declaration in flag_declarations.flag.into_iter() {
83 crate::protos::flag_declaration::verify_fields(&flag_declaration)
Mårten Kongstadcd414d4c2023-07-27 14:25:33 +020084 .with_context(|| input.error_context())?;
Mårten Kongstad403658f2023-06-14 09:51:56 +020085
86 // create ParsedFlag using FlagDeclaration and default values
87 let mut parsed_flag = ProtoParsedFlag::new();
88 parsed_flag.set_package(package.to_string());
89 parsed_flag.set_name(flag_declaration.take_name());
90 parsed_flag.set_namespace(flag_declaration.take_namespace());
91 parsed_flag.set_description(flag_declaration.take_description());
Mårten Kongstad1b8636b2023-06-22 10:12:24 +020092 parsed_flag.bug.append(&mut flag_declaration.bug);
Mårten Kongstad403658f2023-06-14 09:51:56 +020093 parsed_flag.set_state(DEFAULT_FLAG_STATE);
Zhi Dou24a0b6a2023-08-10 21:39:59 +000094 parsed_flag.set_permission(default_permission);
Mårten Kongstad403658f2023-06-14 09:51:56 +020095 let mut tracepoint = ProtoTracepoint::new();
96 tracepoint.set_source(input.source.clone());
97 tracepoint.set_state(DEFAULT_FLAG_STATE);
Zhi Dou24a0b6a2023-08-10 21:39:59 +000098 tracepoint.set_permission(default_permission);
Mårten Kongstad403658f2023-06-14 09:51:56 +020099 parsed_flag.trace.push(tracepoint);
100
101 // verify ParsedFlag looks reasonable
102 crate::protos::parsed_flag::verify_fields(&parsed_flag)?;
103
104 // verify ParsedFlag can be added
105 ensure!(
106 parsed_flags.parsed_flag.iter().all(|other| other.name() != parsed_flag.name()),
107 "failed to declare flag {} from {}: flag already declared",
108 parsed_flag.name(),
109 input.source
110 );
111
112 // add ParsedFlag to ParsedFlags
113 parsed_flags.parsed_flag.push(parsed_flag);
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200114 }
115 }
116
Mårten Kongstadfa23d292023-05-11 14:47:02 +0200117 for mut input in values {
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200118 let mut contents = String::new();
Mårten Kongstadcd414d4c2023-07-27 14:25:33 +0200119 input
120 .reader
121 .read_to_string(&mut contents)
122 .with_context(|| format!("failed to read {}", input.source))?;
Mårten Kongstad403658f2023-06-14 09:51:56 +0200123 let flag_values = crate::protos::flag_values::try_from_text_proto(&contents)
Mårten Kongstadcd414d4c2023-07-27 14:25:33 +0200124 .with_context(|| input.error_context())?;
Mårten Kongstad403658f2023-06-14 09:51:56 +0200125 for flag_value in flag_values.flag_value.into_iter() {
126 crate::protos::flag_value::verify_fields(&flag_value)
Mårten Kongstadcd414d4c2023-07-27 14:25:33 +0200127 .with_context(|| input.error_context())?;
Mårten Kongstad403658f2023-06-14 09:51:56 +0200128
Dennis Shen3cfbcf52023-07-17 14:57:23 +0000129 let Some(parsed_flag) = parsed_flags
130 .parsed_flag
131 .iter_mut()
132 .find(|pf| pf.package() == flag_value.package() && pf.name() == flag_value.name())
133 else {
Mårten Kongstad403658f2023-06-14 09:51:56 +0200134 // (silently) skip unknown flags
135 continue;
136 };
137
138 parsed_flag.set_state(flag_value.state());
139 parsed_flag.set_permission(flag_value.permission());
140 let mut tracepoint = ProtoTracepoint::new();
141 tracepoint.set_source(input.source.clone());
142 tracepoint.set_state(flag_value.state());
143 tracepoint.set_permission(flag_value.permission());
144 parsed_flag.trace.push(tracepoint);
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200145 }
146 }
147
Zhi Dou92cf0ec2023-07-19 19:29:22 +0000148 // Create a sorted parsed_flags
149 crate::protos::parsed_flags::sort_parsed_flags(&mut parsed_flags);
Mårten Kongstad403658f2023-06-14 09:51:56 +0200150 crate::protos::parsed_flags::verify_fields(&parsed_flags)?;
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200151 let mut output = Vec::new();
Mårten Kongstad403658f2023-06-14 09:51:56 +0200152 parsed_flags.write_to_vec(&mut output)?;
153 Ok(output)
154}
155
Zhi Dou8ba6aa72023-06-26 21:03:40 +0000156#[derive(Copy, Clone, Debug, PartialEq, Eq, ValueEnum)]
157pub enum CodegenMode {
158 Production,
159 Test,
160}
161
162pub fn create_java_lib(mut input: Input, codegen_mode: CodegenMode) -> Result<Vec<OutputFile>> {
Mårten Kongstad403658f2023-06-14 09:51:56 +0200163 let parsed_flags = input.try_parse_flags()?;
164 let Some(package) = find_unique_package(&parsed_flags) else {
165 bail!("no parsed flags, or the parsed flags use different packages");
166 };
Zhi Dou8ba6aa72023-06-26 21:03:40 +0000167 generate_java_code(package, parsed_flags.parsed_flag.iter(), codegen_mode)
Mårten Kongstad403658f2023-06-14 09:51:56 +0200168}
169
Dennis Shen8d544f72023-06-29 00:45:42 +0000170pub fn create_cpp_lib(mut input: Input, codegen_mode: CodegenMode) -> Result<Vec<OutputFile>> {
Mårten Kongstad403658f2023-06-14 09:51:56 +0200171 let parsed_flags = input.try_parse_flags()?;
172 let Some(package) = find_unique_package(&parsed_flags) else {
173 bail!("no parsed flags, or the parsed flags use different packages");
174 };
Dennis Shen8d544f72023-06-29 00:45:42 +0000175 generate_cpp_code(package, parsed_flags.parsed_flag.iter(), codegen_mode)
Mårten Kongstad403658f2023-06-14 09:51:56 +0200176}
177
Dennis Shen3cfbcf52023-07-17 14:57:23 +0000178pub fn create_rust_lib(mut input: Input, codegen_mode: CodegenMode) -> Result<OutputFile> {
Mårten Kongstad403658f2023-06-14 09:51:56 +0200179 let parsed_flags = input.try_parse_flags()?;
180 let Some(package) = find_unique_package(&parsed_flags) else {
181 bail!("no parsed flags, or the parsed flags use different packages");
182 };
Dennis Shen3cfbcf52023-07-17 14:57:23 +0000183 generate_rust_code(package, parsed_flags.parsed_flag.iter(), codegen_mode)
Mårten Kongstad403658f2023-06-14 09:51:56 +0200184}
185
186pub fn create_device_config_defaults(mut input: Input) -> Result<Vec<u8>> {
187 let parsed_flags = input.try_parse_flags()?;
188 let mut output = Vec::new();
189 for parsed_flag in parsed_flags
190 .parsed_flag
191 .into_iter()
192 .filter(|pf| pf.permission() == ProtoFlagPermission::READ_WRITE)
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200193 {
194 let line = format!(
Mårten Kongstad202102f2023-06-08 11:22:44 +0200195 "{}:{}.{}={}\n",
Mårten Kongstad403658f2023-06-14 09:51:56 +0200196 parsed_flag.namespace(),
197 parsed_flag.package(),
198 parsed_flag.name(),
199 match parsed_flag.state() {
200 ProtoFlagState::ENABLED => "enabled",
201 ProtoFlagState::DISABLED => "disabled",
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200202 }
203 );
204 output.extend_from_slice(line.as_bytes());
205 }
206 Ok(output)
207}
208
Mårten Kongstad403658f2023-06-14 09:51:56 +0200209pub fn create_device_config_sysprops(mut input: Input) -> Result<Vec<u8>> {
210 let parsed_flags = input.try_parse_flags()?;
Mårten Kongstadc31a6ff2023-06-02 11:54:36 +0200211 let mut output = Vec::new();
Mårten Kongstad403658f2023-06-14 09:51:56 +0200212 for parsed_flag in parsed_flags
213 .parsed_flag
214 .into_iter()
215 .filter(|pf| pf.permission() == ProtoFlagPermission::READ_WRITE)
Mårten Kongstadc31a6ff2023-06-02 11:54:36 +0200216 {
217 let line = format!(
218 "persist.device_config.{}.{}={}\n",
Mårten Kongstad403658f2023-06-14 09:51:56 +0200219 parsed_flag.package(),
220 parsed_flag.name(),
221 match parsed_flag.state() {
222 ProtoFlagState::ENABLED => "true",
223 ProtoFlagState::DISABLED => "false",
Mårten Kongstadc31a6ff2023-06-02 11:54:36 +0200224 }
225 );
226 output.extend_from_slice(line.as_bytes());
227 }
228 Ok(output)
229}
230
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200231#[derive(Copy, Clone, Debug, PartialEq, Eq, ValueEnum)]
Mårten Kongstadba94e6a2023-05-16 11:00:16 +0200232pub enum DumpFormat {
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200233 Text,
Mårten Kongstadea498142023-07-20 11:07:35 +0200234 Verbose,
Mårten Kongstada1029092023-05-08 11:51:59 +0200235 Protobuf,
Mårten Kongstad3228b292023-06-26 10:17:42 +0200236 Textproto,
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200237}
238
Mårten Kongstad403658f2023-06-14 09:51:56 +0200239pub fn dump_parsed_flags(mut input: Vec<Input>, format: DumpFormat) -> Result<Vec<u8>> {
240 let individually_parsed_flags: Result<Vec<ProtoParsedFlags>> =
241 input.iter_mut().map(|i| i.try_parse_flags()).collect();
242 let parsed_flags: ProtoParsedFlags =
243 crate::protos::parsed_flags::merge(individually_parsed_flags?)?;
244
Mårten Kongstadaf677032023-05-17 16:18:25 +0200245 let mut output = Vec::new();
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200246 match format {
247 DumpFormat::Text => {
Mårten Kongstad403658f2023-06-14 09:51:56 +0200248 for parsed_flag in parsed_flags.parsed_flag.into_iter() {
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200249 let line = format!(
Mårten Kongstad3fa2f072023-07-20 09:35:05 +0200250 "{}/{}: {:?} + {:?}\n",
Mårten Kongstad403658f2023-06-14 09:51:56 +0200251 parsed_flag.package(),
252 parsed_flag.name(),
Mårten Kongstad3fa2f072023-07-20 09:35:05 +0200253 parsed_flag.permission(),
254 parsed_flag.state()
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200255 );
256 output.extend_from_slice(line.as_bytes());
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200257 }
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200258 }
Mårten Kongstadea498142023-07-20 11:07:35 +0200259 DumpFormat::Verbose => {
260 for parsed_flag in parsed_flags.parsed_flag.into_iter() {
261 let sources: Vec<_> =
262 parsed_flag.trace.iter().map(|tracepoint| tracepoint.source()).collect();
263 let line = format!(
264 "{}/{}: {:?} + {:?} ({})\n",
265 parsed_flag.package(),
266 parsed_flag.name(),
267 parsed_flag.permission(),
268 parsed_flag.state(),
269 sources.join(", ")
270 );
271 output.extend_from_slice(line.as_bytes());
272 }
273 }
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200274 DumpFormat::Protobuf => {
Mårten Kongstad403658f2023-06-14 09:51:56 +0200275 parsed_flags.write_to_vec(&mut output)?;
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200276 }
Mårten Kongstad3228b292023-06-26 10:17:42 +0200277 DumpFormat::Textproto => {
278 let s = protobuf::text_format::print_to_string_pretty(&parsed_flags);
279 output.extend_from_slice(s.as_bytes());
280 }
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200281 }
Mårten Kongstadaf677032023-05-17 16:18:25 +0200282 Ok(output)
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200283}
284
Mårten Kongstad403658f2023-06-14 09:51:56 +0200285fn find_unique_package(parsed_flags: &ProtoParsedFlags) -> Option<&str> {
286 let Some(package) = parsed_flags.parsed_flag.first().map(|pf| pf.package()) else {
287 return None;
288 };
289 if parsed_flags.parsed_flag.iter().any(|pf| pf.package() != package) {
290 return None;
291 }
292 Some(package)
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200293}
294
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200295#[cfg(test)]
296mod tests {
297 use super::*;
Mårten Kongstada1029092023-05-08 11:51:59 +0200298
299 #[test]
Mårten Kongstad403658f2023-06-14 09:51:56 +0200300 fn test_parse_flags() {
301 let parsed_flags = crate::test::parse_test_flags(); // calls parse_flags
302 crate::protos::parsed_flags::verify_fields(&parsed_flags).unwrap();
303
304 let enabled_ro =
305 parsed_flags.parsed_flag.iter().find(|pf| pf.name() == "enabled_ro").unwrap();
306 assert!(crate::protos::parsed_flag::verify_fields(enabled_ro).is_ok());
307 assert_eq!("com.android.aconfig.test", enabled_ro.package());
308 assert_eq!("enabled_ro", enabled_ro.name());
309 assert_eq!("This flag is ENABLED + READ_ONLY", enabled_ro.description());
310 assert_eq!(ProtoFlagState::ENABLED, enabled_ro.state());
311 assert_eq!(ProtoFlagPermission::READ_ONLY, enabled_ro.permission());
312 assert_eq!(3, enabled_ro.trace.len());
313 assert_eq!("tests/test.aconfig", enabled_ro.trace[0].source());
314 assert_eq!(ProtoFlagState::DISABLED, enabled_ro.trace[0].state());
315 assert_eq!(ProtoFlagPermission::READ_WRITE, enabled_ro.trace[0].permission());
316 assert_eq!("tests/first.values", enabled_ro.trace[1].source());
317 assert_eq!(ProtoFlagState::DISABLED, enabled_ro.trace[1].state());
318 assert_eq!(ProtoFlagPermission::READ_WRITE, enabled_ro.trace[1].permission());
319 assert_eq!("tests/second.values", enabled_ro.trace[2].source());
320 assert_eq!(ProtoFlagState::ENABLED, enabled_ro.trace[2].state());
321 assert_eq!(ProtoFlagPermission::READ_ONLY, enabled_ro.trace[2].permission());
322
323 assert_eq!(4, parsed_flags.parsed_flag.len());
324 for pf in parsed_flags.parsed_flag.iter() {
325 let first = pf.trace.first().unwrap();
326 assert_eq!(DEFAULT_FLAG_STATE, first.state());
327 assert_eq!(DEFAULT_FLAG_PERMISSION, first.permission());
328
329 let last = pf.trace.last().unwrap();
330 assert_eq!(pf.state(), last.state());
331 assert_eq!(pf.permission(), last.permission());
332 }
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200333 }
Mårten Kongstada1029092023-05-08 11:51:59 +0200334
335 #[test]
Zhi Dou24a0b6a2023-08-10 21:39:59 +0000336 fn test_parse_flags_setting_default() {
337 let first_flag = r#"
338 package: "com.first"
339 flag {
340 name: "first"
341 namespace: "first_ns"
342 description: "This is the description of the first flag."
343 bug: "123"
344 }
345 "#;
346 let declaration =
347 vec![Input { source: "momery".to_string(), reader: Box::new(first_flag.as_bytes()) }];
348 let value: Vec<Input> = vec![];
349
350 let flags_bytes = crate::commands::parse_flags(
351 "com.first",
352 declaration,
353 value,
354 ProtoFlagPermission::READ_ONLY,
355 )
356 .unwrap();
357 let parsed_flags =
358 crate::protos::parsed_flags::try_from_binary_proto(&flags_bytes).unwrap();
359 assert_eq!(1, parsed_flags.parsed_flag.len());
360 let parsed_flag = parsed_flags.parsed_flag.first().unwrap();
361 assert_eq!(ProtoFlagState::DISABLED, parsed_flag.state());
362 assert_eq!(ProtoFlagPermission::READ_ONLY, parsed_flag.permission());
363 }
364
365 #[test]
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200366 fn test_create_device_config_defaults() {
Mårten Kongstad403658f2023-06-14 09:51:56 +0200367 let input = parse_test_flags_as_input();
368 let bytes = create_device_config_defaults(input).unwrap();
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200369 let text = std::str::from_utf8(&bytes).unwrap();
Mårten Kongstad202102f2023-06-08 11:22:44 +0200370 assert_eq!("aconfig_test:com.android.aconfig.test.disabled_rw=disabled\naconfig_test:com.android.aconfig.test.enabled_rw=enabled\n", text);
Mårten Kongstadf02734e2023-06-02 11:34:24 +0200371 }
372
373 #[test]
Mårten Kongstadc31a6ff2023-06-02 11:54:36 +0200374 fn test_create_device_config_sysprops() {
Mårten Kongstad403658f2023-06-14 09:51:56 +0200375 let input = parse_test_flags_as_input();
376 let bytes = create_device_config_sysprops(input).unwrap();
Mårten Kongstadc31a6ff2023-06-02 11:54:36 +0200377 let text = std::str::from_utf8(&bytes).unwrap();
Mårten Kongstadfbd71e22023-05-31 13:29:35 +0200378 assert_eq!("persist.device_config.com.android.aconfig.test.disabled_rw=false\npersist.device_config.com.android.aconfig.test.enabled_rw=true\n", text);
Mårten Kongstadc31a6ff2023-06-02 11:54:36 +0200379 }
380
381 #[test]
Mårten Kongstada1029092023-05-08 11:51:59 +0200382 fn test_dump_text_format() {
Mårten Kongstad403658f2023-06-14 09:51:56 +0200383 let input = parse_test_flags_as_input();
384 let bytes = dump_parsed_flags(vec![input], DumpFormat::Text).unwrap();
Mårten Kongstada1029092023-05-08 11:51:59 +0200385 let text = std::str::from_utf8(&bytes).unwrap();
Mårten Kongstad3fa2f072023-07-20 09:35:05 +0200386 assert!(text.contains("com.android.aconfig.test/disabled_ro: READ_ONLY + DISABLED"));
Mårten Kongstada1029092023-05-08 11:51:59 +0200387 }
388
Mårten Kongstad1b8636b2023-06-22 10:12:24 +0200389 #[test]
390 fn test_dump_protobuf_format() {
Mårten Kongstad3228b292023-06-26 10:17:42 +0200391 let expected = protobuf::text_format::parse_from_str::<ProtoParsedFlags>(
392 crate::test::TEST_FLAGS_TEXTPROTO,
393 )
394 .unwrap()
395 .write_to_bytes()
396 .unwrap();
Mårten Kongstad1b8636b2023-06-22 10:12:24 +0200397
398 let input = parse_test_flags_as_input();
399 let actual = dump_parsed_flags(vec![input], DumpFormat::Protobuf).unwrap();
400
401 assert_eq!(expected, actual);
402 }
403
Mårten Kongstad3228b292023-06-26 10:17:42 +0200404 #[test]
405 fn test_dump_textproto_format() {
406 let input = parse_test_flags_as_input();
407 let bytes = dump_parsed_flags(vec![input], DumpFormat::Textproto).unwrap();
408 let text = std::str::from_utf8(&bytes).unwrap();
409 assert_eq!(crate::test::TEST_FLAGS_TEXTPROTO.trim(), text.trim());
410 }
411
Mårten Kongstad403658f2023-06-14 09:51:56 +0200412 fn parse_test_flags_as_input() -> Input {
413 let parsed_flags = crate::test::parse_test_flags();
414 let binary_proto = parsed_flags.write_to_bytes().unwrap();
415 let cursor = std::io::Cursor::new(binary_proto);
416 let reader = Box::new(cursor);
417 Input { source: "test.data".to_string(), reader }
Mårten Kongstadaf677032023-05-17 16:18:25 +0200418 }
Mårten Kongstad4d2b4b02023-04-27 16:05:58 +0200419}