/* * Copyright (C) 2023 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ use anyhow::Result; use serde::Serialize; use tinytemplate::TinyTemplate; use aconfig_protos::{ProtoFlagPermission, ProtoFlagState, ProtoParsedFlag}; use std::collections::HashMap; use crate::codegen; use crate::codegen::CodegenMode; use crate::commands::{should_include_flag, OutputFile}; pub fn generate_rust_code( package: &str, flag_ids: HashMap, parsed_flags_iter: I, codegen_mode: CodegenMode, ) -> Result where I: Iterator, { let template_flags: Vec = parsed_flags_iter .map(|pf| TemplateParsedFlag::new(package, flag_ids.clone(), &pf)) .collect(); let has_readwrite = template_flags.iter().any(|item| item.readwrite); let container = (template_flags.first().expect("zero template flags").container).to_string(); let context = TemplateContext { package: package.to_string(), template_flags, modules: package.split('.').map(|s| s.to_string()).collect::>(), has_readwrite, container, }; let mut template = TinyTemplate::new(); template.add_template( "rust_code_gen", match codegen_mode { CodegenMode::Test => include_str!("../../templates/rust_test.template"), CodegenMode::Exported | CodegenMode::ForceReadOnly | CodegenMode::Production => { include_str!("../../templates/rust.template") } }, )?; let contents = template.render("rust_code_gen", &context)?; let path = ["src", "lib.rs"].iter().collect(); Ok(OutputFile { contents: contents.into(), path }) } #[derive(Serialize)] struct TemplateContext { pub package: String, pub template_flags: Vec, pub modules: Vec, pub has_readwrite: bool, pub container: String, } #[derive(Serialize)] struct TemplateParsedFlag { pub readwrite: bool, pub default_value: String, pub name: String, pub container: String, pub flag_offset: u16, pub device_config_namespace: String, pub device_config_flag: String, } impl TemplateParsedFlag { #[allow(clippy::nonminimal_bool)] fn new(package: &str, flag_offsets: HashMap, pf: &ProtoParsedFlag) -> Self { let flag_offset = match flag_offsets.get(pf.name()) { Some(offset) => offset, None => { // System/vendor/product RO+disabled flags have no offset in storage files. // Assign placeholder value. if !should_include_flag(pf) { &0 } // All other flags _must_ have an offset. else { panic!("{}", format!("missing flag offset for {}", pf.name())); } } }; Self { readwrite: pf.permission() == ProtoFlagPermission::READ_WRITE, default_value: match pf.state() { ProtoFlagState::ENABLED => "true".to_string(), ProtoFlagState::DISABLED => "false".to_string(), }, name: pf.name().to_string(), container: pf.container().to_string(), flag_offset: *flag_offset, device_config_namespace: pf.namespace().to_string(), device_config_flag: codegen::create_device_config_ident(package, pf.name()) .expect("values checked at flag parse time"), } } } #[cfg(test)] mod tests { use super::*; const PROD_EXPECTED: &str = r#" //! codegenerated rust flag lib use aconfig_storage_read_api::{Mmap, AconfigStorageError, StorageFileType, PackageReadContext, get_mapped_storage_file, get_boolean_flag_value, get_package_read_context}; use std::path::Path; use std::io::Write; use std::sync::LazyLock; use log::{log, LevelFilter, Level}; /// flag provider pub struct FlagProvider; static PACKAGE_OFFSET: LazyLock, AconfigStorageError>> = LazyLock::new(|| unsafe { get_mapped_storage_file("system", StorageFileType::PackageMap) .and_then(|package_map| get_package_read_context(&package_map, "com.android.aconfig.test")) .map(|context| context.map(|c| c.boolean_start_index)) }); static FLAG_VAL_MAP: LazyLock> = LazyLock::new(|| unsafe { get_mapped_storage_file("system", StorageFileType::FlagVal) }); /// flag value cache for disabled_rw static CACHED_disabled_rw: LazyLock = LazyLock::new(|| { // This will be called multiple times. Subsequent calls after the first are noops. logger::init( logger::Config::default() .with_tag_on_device("aconfig_rust_codegen") .with_max_level(LevelFilter::Info)); let flag_value_result = FLAG_VAL_MAP .as_ref() .map_err(|err| format!("failed to get flag val map: {err}")) .and_then(|flag_val_map| { PACKAGE_OFFSET .as_ref() .map_err(|err| format!("failed to get package read offset: {err}")) .and_then(|package_offset| { match package_offset { Some(offset) => { get_boolean_flag_value(&flag_val_map, offset + 0) .map_err(|err| format!("failed to get flag: {err}")) }, None => { log!(Level::Error, "no context found for package com.android.aconfig.test"); Err(format!("failed to flag package com.android.aconfig.test")) } } }) }); match flag_value_result { Ok(flag_value) => { return flag_value; }, Err(err) => { log!(Level::Error, "aconfig_rust_codegen: error: {err}"); return false; } } }); /// flag value cache for disabled_rw_exported static CACHED_disabled_rw_exported: LazyLock = LazyLock::new(|| { // This will be called multiple times. Subsequent calls after the first are noops. logger::init( logger::Config::default() .with_tag_on_device("aconfig_rust_codegen") .with_max_level(LevelFilter::Info)); let flag_value_result = FLAG_VAL_MAP .as_ref() .map_err(|err| format!("failed to get flag val map: {err}")) .and_then(|flag_val_map| { PACKAGE_OFFSET .as_ref() .map_err(|err| format!("failed to get package read offset: {err}")) .and_then(|package_offset| { match package_offset { Some(offset) => { get_boolean_flag_value(&flag_val_map, offset + 1) .map_err(|err| format!("failed to get flag: {err}")) }, None => { log!(Level::Error, "no context found for package com.android.aconfig.test"); Err(format!("failed to flag package com.android.aconfig.test")) } } }) }); match flag_value_result { Ok(flag_value) => { return flag_value; }, Err(err) => { log!(Level::Error, "aconfig_rust_codegen: error: {err}"); return false; } } }); /// flag value cache for disabled_rw_in_other_namespace static CACHED_disabled_rw_in_other_namespace: LazyLock = LazyLock::new(|| { // This will be called multiple times. Subsequent calls after the first are noops. logger::init( logger::Config::default() .with_tag_on_device("aconfig_rust_codegen") .with_max_level(LevelFilter::Info)); let flag_value_result = FLAG_VAL_MAP .as_ref() .map_err(|err| format!("failed to get flag val map: {err}")) .and_then(|flag_val_map| { PACKAGE_OFFSET .as_ref() .map_err(|err| format!("failed to get package read offset: {err}")) .and_then(|package_offset| { match package_offset { Some(offset) => { get_boolean_flag_value(&flag_val_map, offset + 2) .map_err(|err| format!("failed to get flag: {err}")) }, None => { log!(Level::Error, "no context found for package com.android.aconfig.test"); Err(format!("failed to flag package com.android.aconfig.test")) } } }) }); match flag_value_result { Ok(flag_value) => { return flag_value; }, Err(err) => { log!(Level::Error, "aconfig_rust_codegen: error: {err}"); return false; } } }); /// flag value cache for enabled_rw static CACHED_enabled_rw: LazyLock = LazyLock::new(|| { // This will be called multiple times. Subsequent calls after the first are noops. logger::init( logger::Config::default() .with_tag_on_device("aconfig_rust_codegen") .with_max_level(LevelFilter::Info)); let flag_value_result = FLAG_VAL_MAP .as_ref() .map_err(|err| format!("failed to get flag val map: {err}")) .and_then(|flag_val_map| { PACKAGE_OFFSET .as_ref() .map_err(|err| format!("failed to get package read offset: {err}")) .and_then(|package_offset| { match package_offset { Some(offset) => { get_boolean_flag_value(&flag_val_map, offset + 7) .map_err(|err| format!("failed to get flag: {err}")) }, None => { log!(Level::Error, "no context found for package com.android.aconfig.test"); Err(format!("failed to flag package com.android.aconfig.test")) } } }) }); match flag_value_result { Ok(flag_value) => { return flag_value; }, Err(err) => { log!(Level::Error, "aconfig_rust_codegen: error: {err}"); return true; } } }); impl FlagProvider { /// query flag disabled_ro pub fn disabled_ro(&self) -> bool { false } /// query flag disabled_rw pub fn disabled_rw(&self) -> bool { *CACHED_disabled_rw } /// query flag disabled_rw_exported pub fn disabled_rw_exported(&self) -> bool { *CACHED_disabled_rw_exported } /// query flag disabled_rw_in_other_namespace pub fn disabled_rw_in_other_namespace(&self) -> bool { *CACHED_disabled_rw_in_other_namespace } /// query flag enabled_fixed_ro pub fn enabled_fixed_ro(&self) -> bool { true } /// query flag enabled_fixed_ro_exported pub fn enabled_fixed_ro_exported(&self) -> bool { true } /// query flag enabled_ro pub fn enabled_ro(&self) -> bool { true } /// query flag enabled_ro_exported pub fn enabled_ro_exported(&self) -> bool { true } /// query flag enabled_rw pub fn enabled_rw(&self) -> bool { *CACHED_enabled_rw } } /// flag provider pub static PROVIDER: FlagProvider = FlagProvider; /// query flag disabled_ro #[inline(always)] pub fn disabled_ro() -> bool { false } /// query flag disabled_rw #[inline(always)] pub fn disabled_rw() -> bool { PROVIDER.disabled_rw() } /// query flag disabled_rw_exported #[inline(always)] pub fn disabled_rw_exported() -> bool { PROVIDER.disabled_rw_exported() } /// query flag disabled_rw_in_other_namespace #[inline(always)] pub fn disabled_rw_in_other_namespace() -> bool { PROVIDER.disabled_rw_in_other_namespace() } /// query flag enabled_fixed_ro #[inline(always)] pub fn enabled_fixed_ro() -> bool { true } /// query flag enabled_fixed_ro_exported #[inline(always)] pub fn enabled_fixed_ro_exported() -> bool { true } /// query flag enabled_ro #[inline(always)] pub fn enabled_ro() -> bool { true } /// query flag enabled_ro_exported #[inline(always)] pub fn enabled_ro_exported() -> bool { true } /// query flag enabled_rw #[inline(always)] pub fn enabled_rw() -> bool { PROVIDER.enabled_rw() } "#; const TEST_EXPECTED: &str = r#" //! codegenerated rust flag lib use aconfig_storage_read_api::{Mmap, AconfigStorageError, StorageFileType, PackageReadContext, get_mapped_storage_file, get_boolean_flag_value, get_package_read_context}; use std::collections::BTreeMap; use std::path::Path; use std::io::Write; use std::sync::{LazyLock, Mutex}; use log::{log, LevelFilter, Level}; /// flag provider pub struct FlagProvider { overrides: BTreeMap<&'static str, bool>, } static PACKAGE_OFFSET: LazyLock, AconfigStorageError>> = LazyLock::new(|| unsafe { get_mapped_storage_file("system", StorageFileType::PackageMap) .and_then(|package_map| get_package_read_context(&package_map, "com.android.aconfig.test")) .map(|context| context.map(|c| c.boolean_start_index)) }); static FLAG_VAL_MAP: LazyLock> = LazyLock::new(|| unsafe { get_mapped_storage_file("system", StorageFileType::FlagVal) }); /// flag value cache for disabled_rw static CACHED_disabled_rw: LazyLock = LazyLock::new(|| { // This will be called multiple times. Subsequent calls after the first are noops. logger::init( logger::Config::default() .with_tag_on_device("aconfig_rust_codegen") .with_max_level(LevelFilter::Info)); let flag_value_result = FLAG_VAL_MAP .as_ref() .map_err(|err| format!("failed to get flag val map: {err}")) .and_then(|flag_val_map| { PACKAGE_OFFSET .as_ref() .map_err(|err| format!("failed to get package read offset: {err}")) .and_then(|package_offset| { match package_offset { Some(offset) => { get_boolean_flag_value(&flag_val_map, offset + 0) .map_err(|err| format!("failed to get flag: {err}")) }, None => { log!(Level::Error, "no context found for package com.android.aconfig.test"); Err(format!("failed to flag package com.android.aconfig.test")) } } }) }); match flag_value_result { Ok(flag_value) => { return flag_value; }, Err(err) => { log!(Level::Error, "aconfig_rust_codegen: error: {err}"); return false; } } }); /// flag value cache for disabled_rw_exported static CACHED_disabled_rw_exported: LazyLock = LazyLock::new(|| { // This will be called multiple times. Subsequent calls after the first are noops. logger::init( logger::Config::default() .with_tag_on_device("aconfig_rust_codegen") .with_max_level(LevelFilter::Info)); let flag_value_result = FLAG_VAL_MAP .as_ref() .map_err(|err| format!("failed to get flag val map: {err}")) .and_then(|flag_val_map| { PACKAGE_OFFSET .as_ref() .map_err(|err| format!("failed to get package read offset: {err}")) .and_then(|package_offset| { match package_offset { Some(offset) => { get_boolean_flag_value(&flag_val_map, offset + 1) .map_err(|err| format!("failed to get flag: {err}")) }, None => { log!(Level::Error, "no context found for package com.android.aconfig.test"); Err(format!("failed to flag package com.android.aconfig.test")) } } }) }); match flag_value_result { Ok(flag_value) => { return flag_value; }, Err(err) => { log!(Level::Error, "aconfig_rust_codegen: error: {err}"); return false; } } }); /// flag value cache for disabled_rw_in_other_namespace static CACHED_disabled_rw_in_other_namespace: LazyLock = LazyLock::new(|| { // This will be called multiple times. Subsequent calls after the first are noops. logger::init( logger::Config::default() .with_tag_on_device("aconfig_rust_codegen") .with_max_level(LevelFilter::Info)); let flag_value_result = FLAG_VAL_MAP .as_ref() .map_err(|err| format!("failed to get flag val map: {err}")) .and_then(|flag_val_map| { PACKAGE_OFFSET .as_ref() .map_err(|err| format!("failed to get package read offset: {err}")) .and_then(|package_offset| { match package_offset { Some(offset) => { get_boolean_flag_value(&flag_val_map, offset + 2) .map_err(|err| format!("failed to get flag: {err}")) }, None => { log!(Level::Error, "no context found for package com.android.aconfig.test"); Err(format!("failed to flag package com.android.aconfig.test")) } } }) }); match flag_value_result { Ok(flag_value) => { return flag_value; }, Err(err) => { log!(Level::Error, "aconfig_rust_codegen: error: {err}"); return false; } } }); /// flag value cache for enabled_rw static CACHED_enabled_rw: LazyLock = LazyLock::new(|| { // This will be called multiple times. Subsequent calls after the first are noops. logger::init( logger::Config::default() .with_tag_on_device("aconfig_rust_codegen") .with_max_level(LevelFilter::Info)); let flag_value_result = FLAG_VAL_MAP .as_ref() .map_err(|err| format!("failed to get flag val map: {err}")) .and_then(|flag_val_map| { PACKAGE_OFFSET .as_ref() .map_err(|err| format!("failed to get package read offset: {err}")) .and_then(|package_offset| { match package_offset { Some(offset) => { get_boolean_flag_value(&flag_val_map, offset + 7) .map_err(|err| format!("failed to get flag: {err}")) }, None => { log!(Level::Error, "no context found for package com.android.aconfig.test"); Err(format!("failed to flag package com.android.aconfig.test")) } } }) }); match flag_value_result { Ok(flag_value) => { return flag_value; }, Err(err) => { log!(Level::Error, "aconfig_rust_codegen: error: {err}"); return true; } } }); impl FlagProvider { /// query flag disabled_ro pub fn disabled_ro(&self) -> bool { self.overrides.get("disabled_ro").copied().unwrap_or( false ) } /// set flag disabled_ro pub fn set_disabled_ro(&mut self, val: bool) { self.overrides.insert("disabled_ro", val); } /// query flag disabled_rw pub fn disabled_rw(&self) -> bool { self.overrides.get("disabled_rw").copied().unwrap_or( *CACHED_disabled_rw ) } /// set flag disabled_rw pub fn set_disabled_rw(&mut self, val: bool) { self.overrides.insert("disabled_rw", val); } /// query flag disabled_rw_exported pub fn disabled_rw_exported(&self) -> bool { self.overrides.get("disabled_rw_exported").copied().unwrap_or( *CACHED_disabled_rw_exported ) } /// set flag disabled_rw_exported pub fn set_disabled_rw_exported(&mut self, val: bool) { self.overrides.insert("disabled_rw_exported", val); } /// query flag disabled_rw_in_other_namespace pub fn disabled_rw_in_other_namespace(&self) -> bool { self.overrides.get("disabled_rw_in_other_namespace").copied().unwrap_or( *CACHED_disabled_rw_in_other_namespace ) } /// set flag disabled_rw_in_other_namespace pub fn set_disabled_rw_in_other_namespace(&mut self, val: bool) { self.overrides.insert("disabled_rw_in_other_namespace", val); } /// query flag enabled_fixed_ro pub fn enabled_fixed_ro(&self) -> bool { self.overrides.get("enabled_fixed_ro").copied().unwrap_or( true ) } /// set flag enabled_fixed_ro pub fn set_enabled_fixed_ro(&mut self, val: bool) { self.overrides.insert("enabled_fixed_ro", val); } /// query flag enabled_fixed_ro_exported pub fn enabled_fixed_ro_exported(&self) -> bool { self.overrides.get("enabled_fixed_ro_exported").copied().unwrap_or( true ) } /// set flag enabled_fixed_ro_exported pub fn set_enabled_fixed_ro_exported(&mut self, val: bool) { self.overrides.insert("enabled_fixed_ro_exported", val); } /// query flag enabled_ro pub fn enabled_ro(&self) -> bool { self.overrides.get("enabled_ro").copied().unwrap_or( true ) } /// set flag enabled_ro pub fn set_enabled_ro(&mut self, val: bool) { self.overrides.insert("enabled_ro", val); } /// query flag enabled_ro_exported pub fn enabled_ro_exported(&self) -> bool { self.overrides.get("enabled_ro_exported").copied().unwrap_or( true ) } /// set flag enabled_ro_exported pub fn set_enabled_ro_exported(&mut self, val: bool) { self.overrides.insert("enabled_ro_exported", val); } /// query flag enabled_rw pub fn enabled_rw(&self) -> bool { self.overrides.get("enabled_rw").copied().unwrap_or( *CACHED_enabled_rw ) } /// set flag enabled_rw pub fn set_enabled_rw(&mut self, val: bool) { self.overrides.insert("enabled_rw", val); } /// clear all flag overrides pub fn reset_flags(&mut self) { self.overrides.clear(); } } /// flag provider pub static PROVIDER: Mutex = Mutex::new( FlagProvider {overrides: BTreeMap::new()} ); /// query flag disabled_ro #[inline(always)] pub fn disabled_ro() -> bool { PROVIDER.lock().unwrap().disabled_ro() } /// set flag disabled_ro #[inline(always)] pub fn set_disabled_ro(val: bool) { PROVIDER.lock().unwrap().set_disabled_ro(val); } /// query flag disabled_rw #[inline(always)] pub fn disabled_rw() -> bool { PROVIDER.lock().unwrap().disabled_rw() } /// set flag disabled_rw #[inline(always)] pub fn set_disabled_rw(val: bool) { PROVIDER.lock().unwrap().set_disabled_rw(val); } /// query flag disabled_rw_exported #[inline(always)] pub fn disabled_rw_exported() -> bool { PROVIDER.lock().unwrap().disabled_rw_exported() } /// set flag disabled_rw_exported #[inline(always)] pub fn set_disabled_rw_exported(val: bool) { PROVIDER.lock().unwrap().set_disabled_rw_exported(val); } /// query flag disabled_rw_in_other_namespace #[inline(always)] pub fn disabled_rw_in_other_namespace() -> bool { PROVIDER.lock().unwrap().disabled_rw_in_other_namespace() } /// set flag disabled_rw_in_other_namespace #[inline(always)] pub fn set_disabled_rw_in_other_namespace(val: bool) { PROVIDER.lock().unwrap().set_disabled_rw_in_other_namespace(val); } /// query flag enabled_fixed_ro #[inline(always)] pub fn enabled_fixed_ro() -> bool { PROVIDER.lock().unwrap().enabled_fixed_ro() } /// set flag enabled_fixed_ro #[inline(always)] pub fn set_enabled_fixed_ro(val: bool) { PROVIDER.lock().unwrap().set_enabled_fixed_ro(val); } /// query flag enabled_fixed_ro_exported #[inline(always)] pub fn enabled_fixed_ro_exported() -> bool { PROVIDER.lock().unwrap().enabled_fixed_ro_exported() } /// set flag enabled_fixed_ro_exported #[inline(always)] pub fn set_enabled_fixed_ro_exported(val: bool) { PROVIDER.lock().unwrap().set_enabled_fixed_ro_exported(val); } /// query flag enabled_ro #[inline(always)] pub fn enabled_ro() -> bool { PROVIDER.lock().unwrap().enabled_ro() } /// set flag enabled_ro #[inline(always)] pub fn set_enabled_ro(val: bool) { PROVIDER.lock().unwrap().set_enabled_ro(val); } /// query flag enabled_ro_exported #[inline(always)] pub fn enabled_ro_exported() -> bool { PROVIDER.lock().unwrap().enabled_ro_exported() } /// set flag enabled_ro_exported #[inline(always)] pub fn set_enabled_ro_exported(val: bool) { PROVIDER.lock().unwrap().set_enabled_ro_exported(val); } /// query flag enabled_rw #[inline(always)] pub fn enabled_rw() -> bool { PROVIDER.lock().unwrap().enabled_rw() } /// set flag enabled_rw #[inline(always)] pub fn set_enabled_rw(val: bool) { PROVIDER.lock().unwrap().set_enabled_rw(val); } /// clear all flag override pub fn reset_flags() { PROVIDER.lock().unwrap().reset_flags() } "#; const FORCE_READ_ONLY_EXPECTED: &str = r#" //! codegenerated rust flag lib use aconfig_storage_read_api::{Mmap, AconfigStorageError, StorageFileType, PackageReadContext, get_mapped_storage_file, get_boolean_flag_value, get_package_read_context}; use std::path::Path; use std::io::Write; use std::sync::LazyLock; use log::{log, LevelFilter, Level}; /// flag provider pub struct FlagProvider; impl FlagProvider { /// query flag disabled_ro pub fn disabled_ro(&self) -> bool { false } /// query flag disabled_rw pub fn disabled_rw(&self) -> bool { false } /// query flag disabled_rw_in_other_namespace pub fn disabled_rw_in_other_namespace(&self) -> bool { false } /// query flag enabled_fixed_ro pub fn enabled_fixed_ro(&self) -> bool { true } /// query flag enabled_ro pub fn enabled_ro(&self) -> bool { true } /// query flag enabled_rw pub fn enabled_rw(&self) -> bool { true } } /// flag provider pub static PROVIDER: FlagProvider = FlagProvider; /// query flag disabled_ro #[inline(always)] pub fn disabled_ro() -> bool { false } /// query flag disabled_rw #[inline(always)] pub fn disabled_rw() -> bool { false } /// query flag disabled_rw_in_other_namespace #[inline(always)] pub fn disabled_rw_in_other_namespace() -> bool { false } /// query flag enabled_fixed_ro #[inline(always)] pub fn enabled_fixed_ro() -> bool { true } /// query flag enabled_ro #[inline(always)] pub fn enabled_ro() -> bool { true } /// query flag enabled_rw #[inline(always)] pub fn enabled_rw() -> bool { true } "#; use crate::commands::assign_flag_ids; fn test_generate_rust_code(mode: CodegenMode, expected: &str) { let parsed_flags = crate::test::parse_test_flags(); let modified_parsed_flags = crate::commands::modify_parsed_flags_based_on_mode(parsed_flags, mode).unwrap(); let flag_ids = assign_flag_ids(crate::test::TEST_PACKAGE, modified_parsed_flags.iter()).unwrap(); let generated = generate_rust_code( crate::test::TEST_PACKAGE, flag_ids, modified_parsed_flags.into_iter(), mode, ) .unwrap(); assert_eq!("src/lib.rs", format!("{}", generated.path.display())); assert_eq!( None, crate::test::first_significant_code_diff( expected, &String::from_utf8(generated.contents).unwrap() ) ); } #[test] fn test_generate_rust_code_for_prod() { test_generate_rust_code(CodegenMode::Production, PROD_EXPECTED); } #[test] fn test_generate_rust_code_for_test() { test_generate_rust_code(CodegenMode::Test, TEST_EXPECTED); } #[test] fn test_generate_rust_code_for_force_read_only() { test_generate_rust_code(CodegenMode::ForceReadOnly, FORCE_READ_ONLY_EXPECTED); } }