pub extern crate serde_json;
extern crate serde;
extern crate regex;
mod json_gettext_value;
pub use json_gettext_value::JSONGetTextValue;
use std::collections::HashMap;
use std::path::Path;
use std::fs::File;
use std::io;
use regex::Regex;
pub use serde_json::Value;
type Context<'a> = HashMap<String, HashMap<String, JSONGetTextValue<'a>>>;
#[derive(Debug, Clone)]
pub struct JSONGetTextBuilder<'a> {
default_key: String,
context: Context<'a>,
}
#[derive(Debug)]
pub enum JSONGetTextBuilderError {
KeyRepeat(String),
IOError(io::Error),
SerdeError(serde_json::Error),
}
impl<'a, 'e> JSONGetTextBuilder<'a> {
pub fn new<S: Into<String>>(default_key: S) -> JSONGetTextBuilder<'a> {
JSONGetTextBuilder {
default_key: default_key.into(),
context: HashMap::new(),
}
}
#[deprecated(since = "2.0.2", note = "please use `new` instead")]
pub fn from_default_key_str<S: AsRef<str>>(default_key: S) -> JSONGetTextBuilder<'a> {
Self::new(default_key.as_ref().to_string())
}
#[deprecated(since = "2.0.2", note = "please use `new` instead")]
pub fn from_default_key_string(default_key: String) -> JSONGetTextBuilder<'a> {
Self::new(default_key)
}
pub fn add_json_string_to_context<K: Into<String>, J: AsRef<str>>(&mut self, key: K, json: J) -> Result<&Self, JSONGetTextBuilderError> {
let key = key.into();
if self.context.contains_key(&key) {
return Err(JSONGetTextBuilderError::KeyRepeat(key));
}
let data: HashMap<String, serde_json::Value> = serde_json::from_str(json.as_ref()).map_err(|err| JSONGetTextBuilderError::SerdeError(err))?;
let mut map = HashMap::new();
for (k, v) in data {
map.insert(k, JSONGetTextValue::JSONValue(v));
}
self.context.insert(key, map);
Ok(self)
}
pub fn add_json_bytes_to_context<K: Into<String> + 'e, J: ?Sized + AsRef<[u8]>>(&mut self, key: K, json: &J) -> Result<&Self, JSONGetTextBuilderError> {
let key = key.into();
if self.context.contains_key(&key) {
return Err(JSONGetTextBuilderError::KeyRepeat(key));
}
let data: HashMap<String, serde_json::Value> = serde_json::from_slice(json.as_ref()).map_err(|err| JSONGetTextBuilderError::SerdeError(err))?;
let mut map = HashMap::new();
for (k, v) in data {
map.insert(k, JSONGetTextValue::JSONValue(v));
}
self.context.insert(key, map);
Ok(self)
}
pub fn add_json_file_to_context<K: Into<String> + 'e, P: AsRef<Path>>(&mut self, key: K, path: P) -> Result<&Self, JSONGetTextBuilderError> {
let key = key.into();
if self.context.contains_key(&key) {
return Err(JSONGetTextBuilderError::KeyRepeat(key));
}
let file = File::open(path).map_err(|err| JSONGetTextBuilderError::IOError(err))?;
let data: HashMap<String, serde_json::Value> = serde_json::from_reader(&file).map_err(|err| JSONGetTextBuilderError::SerdeError(err))?;
let mut map = HashMap::new();
for (k, v) in data {
map.insert(k, JSONGetTextValue::JSONValue(v));
}
self.context.insert(key, map);
Ok(self)
}
pub fn add_map_to_context<K: Into<String> + 'e>(&mut self, key: K, map: HashMap<String, JSONGetTextValue<'a>>) -> Result<&Self, JSONGetTextBuilderError> {
let key = key.into();
if self.context.contains_key(&key) {
return Err(JSONGetTextBuilderError::KeyRepeat(key));
}
self.context.insert(key, map);
Ok(self)
}
pub fn build(self) -> Result<JSONGetText<'a>, JSONGetTextError> {
JSONGetText::from_context_inner(self.default_key, self.context)
}
}
#[derive(Debug)]
pub struct JSONGetText<'a> {
default_key: String,
context: Context<'a>,
}
#[derive(Debug)]
pub enum JSONGetTextError {
DefaultKeyNotFound,
TextInKeyNotInDefaultKey {
key: String,
text: String,
},
}
impl<'a> JSONGetText<'a> {
pub fn build<S: Into<String>>(default_key: S) -> JSONGetTextBuilder<'a> {
JSONGetTextBuilder::new(default_key)
}
#[deprecated(since = "2.0.2", note = "please use `build` instead")]
pub fn build_with_default_key_str<S: AsRef<str>>(default_key: S) -> JSONGetTextBuilder<'a> {
JSONGetTextBuilder::new(default_key.as_ref())
}
#[deprecated(since = "2.0.2", note = "please use `build` instead")]
pub fn build_with_default_key_string(default_key: String) -> JSONGetTextBuilder<'a> {
JSONGetTextBuilder::new(default_key)
}
pub fn from_context_with_default_key<S: Into<String>>(default_key: S, context: Context<'a>) -> Result<JSONGetText<'a>, JSONGetTextError> {
JSONGetText::from_context_inner(default_key.into(), context)
}
#[deprecated(since = "2.0.2", note = "please use `from_context_with_default_key` instead")]
pub fn from_context_with_default_key_str<S: AsRef<str>>(default_key: S, context: Context<'a>) -> Result<JSONGetText<'a>, JSONGetTextError> {
JSONGetText::from_context_inner(default_key.as_ref().to_string(), context)
}
#[deprecated(since = "2.0.2", note = "please use `from_context_with_default_key` instead")]
pub fn from_context_with_default_key_string(default_key: String, context: Context<'a>) -> Result<JSONGetText<'a>, JSONGetTextError> {
JSONGetText::from_context_inner(default_key, context)
}
fn from_context_inner(default_key: String, mut context: Context<'a>) -> Result<JSONGetText<'a>, JSONGetTextError> {
if !context.contains_key(&default_key) {
return Err(JSONGetTextError::DefaultKeyNotFound);
}
let default_map = context.remove(&default_key).unwrap();
let mut inner_context = HashMap::new();
{
for (key, mut map) in context {
{
for map_key in map.keys() {
if !default_map.contains_key(map_key) {
return Err(JSONGetTextError::TextInKeyNotInDefaultKey {
key,
text: map_key.clone(),
});
}
}
}
{
for map_key in default_map.keys() {
if !map.contains_key(map_key) {
map.insert(map_key.clone(), default_map.get(map_key).unwrap().clone());
}
}
}
inner_context.insert(key, map);
}
inner_context.insert(default_key.clone(), default_map);
}
Ok(JSONGetText {
default_key,
context: inner_context,
})
}
pub fn get_keys(&self) -> Vec<&str> {
let mut vec = Vec::new();
for key in self.context.keys() {
vec.push(key.as_str());
}
vec
}
pub fn get_default_key(&self) -> &str {
&self.default_key
}
pub fn get<K: AsRef<str>>(&self, key: K) -> &HashMap<String, JSONGetTextValue> {
match self.context.get(key.as_ref()) {
Some(m) => m,
None => self.context.get(&self.default_key).unwrap()
}
}
pub fn get_text<T: AsRef<str>>(&self, text: T) -> Option<JSONGetTextValue> {
let map = self.context.get(&self.default_key).unwrap();
map.get(text.as_ref()).map(|s| match s {
JSONGetTextValue::JSONValue(v) => JSONGetTextValue::JSONValueRef(v),
_ => JSONGetTextValue::Str("")
})
}
pub fn get_text_with_key<K: AsRef<str>, T: AsRef<str>>(&self, key: K, text: T) -> Option<JSONGetTextValue> {
let map = match self.context.get(key.as_ref()) {
Some(m) => m,
None => self.context.get(&self.default_key).unwrap()
};
map.get(text.as_ref()).map(|s| match s {
JSONGetTextValue::JSONValue(v) => JSONGetTextValue::JSONValueRef(v),
_ => JSONGetTextValue::Str("")
})
}
pub fn get_multiple_text<'b, T: AsRef<str>>(&self, text_array: &[&'b T]) -> Option<HashMap<&'b str, JSONGetTextValue>> {
let map = self.context.get(&self.default_key).unwrap();
let mut new_map = HashMap::new();
for &text in text_array.iter() {
let text = text.as_ref();
let value = map.get(text)?;
new_map.insert(text, JSONGetTextValue::JSONValueRef(match value {
JSONGetTextValue::JSONValue(v) => v,
_ => return None
}));
}
Some(new_map)
}
pub fn get_multiple_text_with_key<'b, K: AsRef<str>, T: ?Sized + AsRef<str>>(&self, key: K, text_array: &[&'b T]) -> Option<HashMap<&'b str, JSONGetTextValue>> {
let map = match self.context.get(key.as_ref()) {
Some(m) => m,
None => self.context.get(&self.default_key).unwrap()
};
let mut new_map = HashMap::new();
for &text in text_array.iter() {
let text = text.as_ref();
let value = map.get(text)?;
new_map.insert(text, JSONGetTextValue::JSONValueRef(match value {
JSONGetTextValue::JSONValue(v) => v,
_ => return None
}));
}
Some(new_map)
}
pub fn get_filtered_text(&self, regex: &Regex) -> Option<HashMap<&str, JSONGetTextValue>> {
let map = self.context.get(&self.default_key).unwrap();
let mut new_map = HashMap::new();
for (key, value) in map.iter() {
if !regex.is_match(key) {
continue;
}
new_map.insert(key.as_str(), JSONGetTextValue::JSONValueRef(match value {
JSONGetTextValue::JSONValue(v) => v,
_ => return None
}));
}
Some(new_map)
}
pub fn get_filtered_text_with_key<K: AsRef<str>>(&self, key: K, regex: &Regex) -> Option<HashMap<&str, JSONGetTextValue>> {
let map = match self.context.get(key.as_ref()) {
Some(m) => m,
None => self.context.get(&self.default_key).unwrap()
};
let mut new_map = HashMap::new();
for (key, value) in map.iter() {
if !regex.is_match(key) {
continue;
}
new_map.insert(key.as_str(), JSONGetTextValue::JSONValueRef(match value {
JSONGetTextValue::JSONValue(v) => v,
_ => return None
}));
}
Some(new_map)
}
}
mod macros;