mirror of
https://github.com/danbulant/oxc
synced 2026-05-24 20:32:10 +00:00
138 lines
4.3 KiB
Rust
138 lines
4.3 KiB
Rust
use crate::{token::TokenChunk, SourceMap, Token};
|
|
use std::sync::Arc;
|
|
|
|
/// The `ConcatSourceMapBuilder` is a helper to concat sourcemaps.
|
|
#[derive(Debug, Default)]
|
|
pub struct ConcatSourceMapBuilder {
|
|
pub(crate) names: Vec<Arc<str>>,
|
|
pub(crate) sources: Vec<Arc<str>>,
|
|
pub(crate) source_contents: Vec<Arc<str>>,
|
|
pub(crate) tokens: Vec<Token>,
|
|
/// The `token_chunks` is used for encode tokens to vlq mappings at parallel.
|
|
pub(crate) token_chunks: Vec<TokenChunk>,
|
|
pub(crate) token_chunk_prev_name_id: u32,
|
|
}
|
|
|
|
#[allow(clippy::cast_possible_truncation)]
|
|
impl ConcatSourceMapBuilder {
|
|
pub fn add_sourcemap(&mut self, sourcemap: &SourceMap, line_offset: u32) {
|
|
let source_offset = self.sources.len() as u32;
|
|
let name_offset = self.names.len() as u32;
|
|
|
|
// Add `token_chunks`, See `TokenChunk`.
|
|
if let Some(last_token) = self.tokens.last() {
|
|
self.token_chunks.push(TokenChunk::new(
|
|
self.tokens.len() as u32,
|
|
self.tokens.len() as u32 + sourcemap.tokens.len() as u32,
|
|
last_token.get_dst_line(),
|
|
last_token.get_dst_col(),
|
|
last_token.get_src_line(),
|
|
last_token.get_src_col(),
|
|
self.token_chunk_prev_name_id,
|
|
source_offset - 1,
|
|
));
|
|
} else {
|
|
self.token_chunks.push(TokenChunk::new(
|
|
0,
|
|
sourcemap.tokens.len() as u32,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
));
|
|
}
|
|
|
|
// Extend `sources` and `source_contents`.
|
|
self.sources.reserve(sourcemap.sources.len());
|
|
for (index, source) in sourcemap.get_sources().enumerate() {
|
|
let source_content = sourcemap.get_source_content(index as u32).unwrap_or_default();
|
|
self.sources.push(source.into());
|
|
self.source_contents.push(source_content.into());
|
|
}
|
|
|
|
// Extend `names`.
|
|
self.names.reserve(sourcemap.names.len());
|
|
self.names.extend(sourcemap.get_names().map(Into::into));
|
|
|
|
// Extend `tokens`.
|
|
self.tokens.reserve(sourcemap.tokens.len());
|
|
let tokens = sourcemap.get_tokens().map(|token| {
|
|
Token::new(
|
|
token.get_dst_line() + line_offset,
|
|
token.get_dst_col(),
|
|
token.get_src_line(),
|
|
token.get_src_col(),
|
|
token.get_source_id().map(|x| x + source_offset),
|
|
token.get_name_id().map(|x| {
|
|
self.token_chunk_prev_name_id = x + name_offset;
|
|
self.token_chunk_prev_name_id
|
|
}),
|
|
)
|
|
});
|
|
self.tokens.extend(tokens);
|
|
}
|
|
|
|
pub fn into_sourcemap(self) -> SourceMap {
|
|
SourceMap::new(
|
|
None,
|
|
self.names,
|
|
None,
|
|
self.sources,
|
|
Some(self.source_contents),
|
|
self.tokens,
|
|
Some(self.token_chunks),
|
|
)
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_concat_sourcemap_builder() {
|
|
let sm1 = SourceMap::new(
|
|
None,
|
|
vec!["foo".into(), "foo2".into()],
|
|
None,
|
|
vec!["foo.js".into()],
|
|
None,
|
|
vec![Token::new(1, 1, 1, 1, Some(0), Some(0))],
|
|
None,
|
|
);
|
|
let sm2 = SourceMap::new(
|
|
None,
|
|
vec!["bar".into()],
|
|
None,
|
|
vec!["bar.js".into()],
|
|
None,
|
|
vec![Token::new(1, 1, 1, 1, Some(0), Some(0))],
|
|
None,
|
|
);
|
|
|
|
let mut builder = ConcatSourceMapBuilder::default();
|
|
builder.add_sourcemap(&sm1, 0);
|
|
builder.add_sourcemap(&sm2, 2);
|
|
|
|
let sm = SourceMap::new(
|
|
None,
|
|
vec!["foo".into(), "foo2".into(), "bar".into()],
|
|
None,
|
|
vec!["foo.js".into(), "bar.js".into()],
|
|
None,
|
|
vec![Token::new(1, 1, 1, 1, Some(0), Some(0)), Token::new(3, 1, 1, 1, Some(1), Some(2))],
|
|
None,
|
|
);
|
|
let concat_sm = builder.into_sourcemap();
|
|
|
|
assert_eq!(concat_sm.tokens, sm.tokens);
|
|
assert_eq!(concat_sm.sources, sm.sources);
|
|
assert_eq!(concat_sm.names, sm.names);
|
|
assert_eq!(
|
|
concat_sm.token_chunks,
|
|
Some(vec![
|
|
TokenChunk::new(0, 1, 0, 0, 0, 0, 0, 0,),
|
|
TokenChunk::new(1, 2, 1, 1, 1, 1, 0, 0,)
|
|
])
|
|
);
|
|
|
|
assert_eq!(sm.to_json_string().unwrap(), sm.to_json_string().unwrap());
|
|
}
|