ucg/src/convert/flags.rs

147 lines
4.5 KiB
Rust
Raw Normal View History

// Copyright 2017 Jeremy Wall <jeremy@marzhillstudios.com>
//
// 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.
//! Contains code for converting a UCG Val into the command line flag output target.
2018-05-14 21:34:38 -05:00
use std::io::Write;
use std::rc::Rc;
2018-12-06 12:23:52 -06:00
use crate::build::Val;
use crate::convert::traits::{ConvertResult, Converter};
/// FlagConverter implements the conversion logic for converting a Val into a set
/// of command line flags.
pub struct FlagConverter {
sep: &'static str,
}
impl FlagConverter {
pub fn new() -> Self {
FlagConverter { sep: "." }
}
pub fn with_sep(mut self, sep: &'static str) -> Self {
self.sep = sep;
self
}
fn write_flag_name(&self, pfx: &str, name: &str, w: &mut Write) -> ConvertResult {
if name.chars().count() > 1 || pfx.chars().count() > 0 {
2018-12-06 12:46:47 -06:00
write!(w, "--{}{} ", pfx, name)?;
} else {
2018-12-06 12:46:47 -06:00
write!(w, "-{} ", name)?;
}
return Ok(());
}
fn write_list_flag(
&self,
pfx: &str,
name: &str,
def: &Vec<Rc<Val>>,
w: &mut Write,
) -> ConvertResult {
// first of all we need to make sure that each &Val is only a primitive type.
for v in def.iter() {
let vref = v.as_ref();
if vref.is_list() || vref.is_tuple() || vref.is_func() {
eprintln!(
"Skipping non primitive val in list for flag {}{}",
pfx, name
);
} else {
2018-12-06 12:46:47 -06:00
self.write_flag_name(pfx, name, w)?;
self.write(pfx, vref, w)?;
}
}
return Ok(());
}
fn write(&self, pfx: &str, v: &Val, w: &mut Write) -> ConvertResult {
match v {
&Val::Empty => {
// Empty is a noop.
return Ok(());
}
2018-03-22 20:09:38 -05:00
&Val::Boolean(b) => {
2018-12-06 12:46:47 -06:00
write!(w, "{} ", if b { "true" } else { "false" })?;
2018-03-22 20:09:38 -05:00
}
&Val::Float(ref f) => {
2018-12-06 12:46:47 -06:00
write!(w, "{} ", f)?;
}
&Val::Int(ref i) => {
2018-12-06 12:46:47 -06:00
write!(w, "{} ", i)?;
}
&Val::Str(ref s) => {
2018-12-06 12:46:47 -06:00
write!(w, "'{}' ", s)?;
}
&Val::List(ref _def) => {
eprintln!("Skipping List...");
}
2018-12-10 21:27:44 -06:00
&Val::Tuple(ref flds) => {
for &(ref name, ref val) in flds.iter() {
if let &Val::Empty = val.as_ref() {
self.write_flag_name(pfx, name, w)?;
2018-12-10 21:27:44 -06:00
continue;
}
match val.as_ref() {
&Val::Tuple(_) => {
let new_pfx = format!("{}{}{}", pfx, name, self.sep);
2018-12-10 21:27:44 -06:00
self.write(&new_pfx, val, w)?;
}
&Val::List(ref def) => {
self.write_list_flag(pfx, name, def, w)?;
2018-12-10 21:27:44 -06:00
}
_ => {
self.write_flag_name(pfx, name, w)?;
2018-12-10 21:27:44 -06:00
self.write(pfx, &val, w)?;
}
}
}
2018-12-10 21:27:44 -06:00
}
&Val::Func(ref _def) => {
// This is ignored
eprintln!("Skipping macro...");
}
&Val::Env(ref _fs) => {
// This is ignored
eprintln!("Skipping env...");
}
&Val::Module(ref _def) => {
// This is ignored
eprintln!("Skipping module...");
}
}
Ok(())
}
}
impl Converter for FlagConverter {
fn convert(&self, v: Rc<Val>, mut w: &mut Write) -> ConvertResult {
self.write("", &v, &mut w)
}
fn file_ext(&self) -> String {
String::from("txt")
}
fn description(&self) -> String {
"Convert ucg Vals into command line flags.".to_string()
}
2018-03-11 15:11:27 -05:00
#[allow(unused_must_use)]
fn help(&self) -> String {
include_str!("flags_help.txt").to_string()
}
}