2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2019-07-31 19:16:03 +02:00
|
|
|
use crate::compilers::CompiledModule;
|
|
|
|
use crate::compilers::CompiledModuleFuture;
|
|
|
|
use crate::file_fetcher::SourceFile;
|
2019-11-17 01:17:47 +01:00
|
|
|
use crate::futures::future::FutureExt;
|
2020-01-05 11:56:18 -05:00
|
|
|
use deno_core::ErrBox;
|
2019-10-26 18:04:34 -07:00
|
|
|
use regex::Regex;
|
2019-11-17 01:17:47 +01:00
|
|
|
use std::pin::Pin;
|
2019-07-31 19:16:03 +02:00
|
|
|
use std::str;
|
|
|
|
|
2019-10-26 18:04:34 -07:00
|
|
|
// From https://github.com/mathiasbynens/mothereff.in/blob/master/js-variables/eff.js
|
|
|
|
static JS_RESERVED_WORDS: &str = r"^(?:do|if|in|for|let|new|try|var|case|else|enum|eval|false|null|this|true|void|with|await|break|catch|class|const|super|throw|while|yield|delete|export|import|public|return|static|switch|typeof|default|extends|finally|package|private|continue|debugger|function|arguments|interface|protected|implements|instanceof)$";
|
|
|
|
|
2019-07-31 19:16:03 +02:00
|
|
|
pub struct JsonCompiler {}
|
|
|
|
|
|
|
|
impl JsonCompiler {
|
|
|
|
pub fn compile_async(
|
2020-01-04 15:50:52 +05:30
|
|
|
&self,
|
2019-07-31 19:16:03 +02:00
|
|
|
source_file: &SourceFile,
|
2019-11-17 01:17:47 +01:00
|
|
|
) -> Pin<Box<CompiledModuleFuture>> {
|
2019-10-26 18:04:34 -07:00
|
|
|
let maybe_json_value: serde_json::Result<serde_json::Value> =
|
|
|
|
serde_json::from_str(&str::from_utf8(&source_file.source_code).unwrap());
|
|
|
|
if let Err(err) = maybe_json_value {
|
2019-11-17 01:17:47 +01:00
|
|
|
return futures::future::err(ErrBox::from(err)).boxed();
|
2019-10-26 18:04:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut code = format!(
|
|
|
|
"export default {};\n",
|
|
|
|
str::from_utf8(&source_file.source_code).unwrap()
|
|
|
|
);
|
|
|
|
|
|
|
|
if let serde_json::Value::Object(m) = maybe_json_value.unwrap() {
|
|
|
|
// Best effort variable name exports
|
|
|
|
// Actual all allowed JS variable names are way tricker.
|
|
|
|
// We only handle a subset of alphanumeric names.
|
|
|
|
let js_var_regex = Regex::new(r"^[a-zA-Z_$][0-9a-zA-Z_$]*$").unwrap();
|
|
|
|
// Also avoid collision with reserved words.
|
|
|
|
let reserved_words = Regex::new(JS_RESERVED_WORDS).unwrap();
|
|
|
|
for (key, value) in m.iter() {
|
|
|
|
if js_var_regex.is_match(&key) && !reserved_words.is_match(&key) {
|
|
|
|
code.push_str(&format!(
|
|
|
|
"export const {} = {};\n",
|
|
|
|
key,
|
|
|
|
value.to_string()
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-31 19:16:03 +02:00
|
|
|
let module = CompiledModule {
|
2019-10-26 18:04:34 -07:00
|
|
|
code,
|
2019-07-31 19:16:03 +02:00
|
|
|
name: source_file.url.to_string(),
|
|
|
|
};
|
|
|
|
|
2019-11-17 01:17:47 +01:00
|
|
|
futures::future::ok(module).boxed()
|
2019-07-31 19:16:03 +02:00
|
|
|
}
|
|
|
|
}
|