use std::collections::btree_map::BTreeMap;
use std::env;
use std::fs::File;
use std::io::Read;
use std::path::Path;
use std::path::PathBuf;
use toml;
use toml::Parser as TomlParser;
pub struct Config {
pub sdk_path: PathBuf,
pub ndk_path: PathBuf,
pub ant_command: String,
pub package_name: String,
pub project_name: String,
pub package_label: String,
pub package_icon: Option<String>,
pub build_targets: Vec<String>,
pub android_version: u32,
pub assets_path: Option<PathBuf>,
pub res_path: Option<PathBuf>,
pub release: bool,
pub fullscreen: bool,
pub application_attributes: Option<String>,
pub activity_attributes: Option<String>,
pub target: Option<String>,
}
pub fn load(manifest_path: &Path) -> Config {
let (package_name, manifest_content) = {
let content = {
let mut file = File::open(manifest_path).unwrap();
let mut content = String::new();
file.read_to_string(&mut content).unwrap();
content
};
let toml = TomlParser::new(&content).parse().unwrap();
let decoded: TomlPackage = toml::decode(toml["package"].clone()).unwrap();
let package_name = decoded.name.clone();
(package_name, decoded.metadata.and_then(|m| m.android))
};
let ndk_path = env::var("NDK_HOME").expect("Please set the path to the Android NDK with the \
$NDK_HOME environment variable.");
let sdk_path = {
let mut try = env::var("ANDROID_SDK_HOME").ok();
if try.is_none() {
try = env::var("ANDROID_HOME").ok();
}
try.expect("Please set the path to the Android SDK with either the $ANDROID_SDK_HOME or \
the $ANDROID_HOME environment variable.")
};
Config {
sdk_path: Path::new(&sdk_path).to_owned(),
ndk_path: Path::new(&ndk_path).to_owned(),
ant_command: if cfg!(target_os = "windows") { "ant.bat" } else { "ant" }.to_owned(),
package_name: manifest_content.as_ref().and_then(|a| a.package_name.clone())
.unwrap_or_else(|| format!("rust.{}", package_name)),
project_name: package_name.clone(),
package_label: manifest_content.as_ref().and_then(|a| a.label.clone())
.unwrap_or_else(|| package_name.clone()),
package_icon: manifest_content.as_ref().and_then(|a| a.icon.clone()),
build_targets: manifest_content.as_ref().and_then(|a| a.build_targets.clone())
.unwrap_or(vec!["arm-linux-androideabi".to_owned()]),
android_version: manifest_content.as_ref().and_then(|a| a.android_version).unwrap_or(18),
assets_path: manifest_content.as_ref().and_then(|a| a.assets.as_ref())
.map(|p| manifest_path.parent().unwrap().join(p)),
res_path: manifest_content.as_ref().and_then(|a| a.res.as_ref())
.map(|p| manifest_path.parent().unwrap().join(p)),
release: false,
fullscreen: manifest_content.as_ref().and_then(|a| a.fullscreen.clone()).unwrap_or(false),
application_attributes: manifest_content.as_ref().and_then(|a| map_to_string(a.application_attributes.clone())),
activity_attributes: manifest_content.as_ref().and_then(|a| map_to_string(a.activity_attributes.clone())),
target: None,
}
}
fn map_to_string(input_map: Option<BTreeMap<String, String>>) -> Option<String> {
if let Some(map) = input_map {
let mut result = String::new();
for (key, val) in map {
result.push_str(&format!("\n{}=\"{}\"", key, val))
}
Some(result)
} else {
None
}
}
#[derive(Debug, Clone, RustcDecodable)]
struct TomlPackage {
name: String,
metadata: Option<TomlMetadata>,
}
#[derive(Debug, Clone, RustcDecodable)]
struct TomlMetadata {
android: Option<TomlAndroid>,
}
#[derive(Debug, Clone, RustcDecodable)]
struct TomlAndroid {
package_name: Option<String>,
label: Option<String>,
icon: Option<String>,
assets: Option<String>,
res: Option<String>,
android_version: Option<u32>,
fullscreen: Option<bool>,
application_attributes: Option<BTreeMap<String, String>>,
activity_attributes: Option<BTreeMap<String, String>>,
build_targets: Option<Vec<String>>,
}