mirror of
https://github.com/danbulant/oxc
synced 2026-05-20 12:48:38 +00:00
230 lines
6.7 KiB
Rust
230 lines
6.7 KiB
Rust
use std::{
|
|
fs::File,
|
|
io::Write,
|
|
path::{Path, PathBuf},
|
|
};
|
|
|
|
use oxc_allocator::Allocator;
|
|
use oxc_codegen::{Codegen, CodegenOptions};
|
|
use oxc_parser::Parser;
|
|
use oxc_span::SourceType;
|
|
use oxc_tasks_common::{project_root, TestFiles};
|
|
|
|
use crate::suite::{Case, Suite, TestResult};
|
|
|
|
static FIXTURES_PATH: &str =
|
|
"tasks/coverage/babel/packages/babel-generator/test/fixtures/sourcemaps";
|
|
|
|
pub struct SourcemapSuite<T: Case> {
|
|
test_root: PathBuf,
|
|
test_cases: Vec<T>,
|
|
}
|
|
|
|
impl<T: Case> SourcemapSuite<T> {
|
|
pub fn new() -> Self {
|
|
Self {
|
|
test_root: project_root().join(FIXTURES_PATH),
|
|
test_cases: TestFiles::new()
|
|
.files()
|
|
.iter()
|
|
.filter(|file| file.file_name.contains("react"))
|
|
.map(|file| T::new(file.file_name.clone().into(), file.source_text.clone()))
|
|
.collect::<Vec<_>>(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T: Case> Suite<T> for SourcemapSuite<T> {
|
|
fn get_test_root(&self) -> &Path {
|
|
&self.test_root
|
|
}
|
|
|
|
fn save_test_cases(&mut self, tests: Vec<T>) {
|
|
self.test_cases.extend(tests);
|
|
}
|
|
|
|
fn get_test_cases(&self) -> &Vec<T> {
|
|
&self.test_cases
|
|
}
|
|
|
|
fn get_test_cases_mut(&mut self) -> &mut Vec<T> {
|
|
&mut self.test_cases
|
|
}
|
|
|
|
fn skip_test_path(&self, path: &Path) -> bool {
|
|
let path = path.to_string_lossy();
|
|
!path.contains("input.js")
|
|
}
|
|
|
|
fn run_coverage(&self, name: &str, _args: &crate::AppArgs) {
|
|
let path = project_root().join(format!("tasks/coverage/{name}.snap"));
|
|
let mut file = File::create(path).unwrap();
|
|
|
|
let mut tests = self.get_test_cases().iter().collect::<Vec<_>>();
|
|
tests.sort_by_key(|case| case.path());
|
|
|
|
for case in tests {
|
|
let result = case.test_result();
|
|
let path = case.path().to_string_lossy();
|
|
let result = match result {
|
|
TestResult::Snapshot(snapshot) => snapshot,
|
|
TestResult::ParseError(error, _) => error,
|
|
_ => {
|
|
unreachable!()
|
|
}
|
|
};
|
|
writeln!(file, "- {path}").unwrap();
|
|
writeln!(file, "{result}\n\n").unwrap();
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct SourcemapCase {
|
|
path: PathBuf,
|
|
code: String,
|
|
source_type: SourceType,
|
|
result: TestResult,
|
|
}
|
|
|
|
impl SourcemapCase {
|
|
pub fn source_type(&self) -> SourceType {
|
|
self.source_type
|
|
}
|
|
|
|
fn generate_line_utf16_tables(content: &str) -> Vec<Vec<u16>> {
|
|
let mut tables = vec![];
|
|
let mut line_byte_offset = 0;
|
|
for (i, ch) in content.char_indices() {
|
|
match ch {
|
|
'\r' | '\n' | '\u{2028}' | '\u{2029}' => {
|
|
// Handle Windows-specific "\r\n" newlines
|
|
if ch == '\r' && content.chars().nth(i + 1) == Some('\n') {
|
|
continue;
|
|
}
|
|
tables.push(content[line_byte_offset..i].encode_utf16().collect::<Vec<_>>());
|
|
line_byte_offset = i;
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
tables.push(content[line_byte_offset..].encode_utf16().collect::<Vec<_>>());
|
|
tables
|
|
}
|
|
|
|
fn create_visualizer_text(
|
|
source: &str,
|
|
output: &str,
|
|
tokens: &[(u32, u32, u32, u32)],
|
|
) -> String {
|
|
let source_lines = Self::generate_line_utf16_tables(source);
|
|
let output_lines = Self::generate_line_utf16_tables(output);
|
|
let mut s = String::new();
|
|
|
|
tokens.iter().reduce(|pre, cur| {
|
|
s.push_str(&format!(
|
|
"({}:{}-{}:{}) {:?}",
|
|
pre.0,
|
|
pre.1,
|
|
cur.0,
|
|
cur.1,
|
|
Self::str_slice_by_token(&source_lines, (pre.0, pre.1), (cur.0, cur.1))
|
|
));
|
|
s.push_str(" --> ");
|
|
s.push_str(&format!(
|
|
"({}:{}-{}:{}) {:?}",
|
|
pre.2,
|
|
pre.3,
|
|
cur.2,
|
|
cur.3,
|
|
Self::str_slice_by_token(&output_lines, (pre.2, pre.3), (cur.2, cur.3))
|
|
));
|
|
s.push('\n');
|
|
cur
|
|
});
|
|
|
|
s
|
|
}
|
|
|
|
fn str_slice_by_token(buff: &[Vec<u16>], start: (u32, u32), end: (u32, u32)) -> String {
|
|
if start.0 == end.0 {
|
|
return String::from_utf16(&buff[start.0 as usize][start.1 as usize..end.1 as usize])
|
|
.unwrap();
|
|
}
|
|
|
|
let mut s = String::new();
|
|
|
|
for i in start.0..end.0 {
|
|
let slice = &buff[i as usize];
|
|
if i == start.0 {
|
|
s.push_str(&String::from_utf16(&slice[start.1 as usize..]).unwrap());
|
|
} else if i == end.0 {
|
|
s.push_str(&String::from_utf16(&slice[..end.1 as usize]).unwrap());
|
|
} else {
|
|
s.push_str(&String::from_utf16(slice).unwrap());
|
|
}
|
|
}
|
|
|
|
s
|
|
}
|
|
}
|
|
|
|
impl Case for SourcemapCase {
|
|
fn new(path: PathBuf, code: String) -> Self {
|
|
let source_type = SourceType::from_path(&path).unwrap();
|
|
Self { path, code, source_type, result: TestResult::ToBeRun }
|
|
}
|
|
|
|
fn code(&self) -> &str {
|
|
&self.code
|
|
}
|
|
|
|
fn path(&self) -> &Path {
|
|
&self.path
|
|
}
|
|
|
|
fn test_result(&self) -> &TestResult {
|
|
&self.result
|
|
}
|
|
|
|
fn run(&mut self) {
|
|
let source_type = self.source_type();
|
|
self.result = self.execute(source_type);
|
|
}
|
|
|
|
fn execute(&mut self, source_type: SourceType) -> TestResult {
|
|
let source_text = self.code();
|
|
let allocator = Allocator::default();
|
|
let ret = Parser::new(&allocator, source_text, source_type).parse();
|
|
|
|
if !ret.errors.is_empty() {
|
|
if let Some(error) = ret.errors.into_iter().next() {
|
|
let error = error.with_source_code(source_text.to_string());
|
|
return TestResult::ParseError(error.to_string(), false);
|
|
}
|
|
}
|
|
|
|
let codegen_options = CodegenOptions {
|
|
enable_source_map: Some(self.path.to_string_lossy().to_string()),
|
|
..CodegenOptions::default()
|
|
};
|
|
let codegen_ret = Codegen::<false>::new(source_text, codegen_options).build(&ret.program);
|
|
let content = codegen_ret.source_text;
|
|
let map = codegen_ret.source_map.unwrap();
|
|
let tokens = map
|
|
.tokens()
|
|
.map(|token| {
|
|
(
|
|
token.get_src_line(),
|
|
token.get_src_col(),
|
|
token.get_dst_line(),
|
|
token.get_dst_col(),
|
|
)
|
|
})
|
|
.collect::<Vec<_>>();
|
|
|
|
let mut result = String::new();
|
|
result.push_str(Self::create_visualizer_text(source_text, &content, &tokens).as_str());
|
|
|
|
TestResult::Snapshot(result)
|
|
}
|
|
}
|