mirror of
https://github.com/nushell/nushell.git
synced 2025-04-19 10:48:21 +02:00
This commit extracts five new crates: - nu-source, which contains the core source-code handling logic in Nu, including Text, Span, and also the pretty.rs-based debug logic - nu-parser, which is the parser and expander logic - nu-protocol, which is the bulk of the types and basic conveniences used by plugins - nu-errors, which contains ShellError, ParseError and error handling conveniences - nu-textview, which is the textview plugin extracted into a crate One of the major consequences of this refactor is that it's no longer possible to `impl X for Spanned<Y>` outside of the `nu-source` crate, so a lot of types became more concrete (Value became a concrete type instead of Spanned<Value>, for example). This also turned a number of inherent methods in the main nu crate into plain functions (impl Value {} became a bunch of functions in the `value` namespace in `crate::data::value`).
102 lines
2.7 KiB
Rust
102 lines
2.7 KiB
Rust
use derive_new::new;
|
|
use language_reporting::{FileName, Location};
|
|
use log::trace;
|
|
use nu_source::Span;
|
|
|
|
#[derive(new, Debug, Clone)]
|
|
pub struct Files {
|
|
snippet: String,
|
|
}
|
|
|
|
impl language_reporting::ReportingFiles for Files {
|
|
type Span = Span;
|
|
type FileId = usize;
|
|
|
|
fn byte_span(
|
|
&self,
|
|
_file: Self::FileId,
|
|
from_index: usize,
|
|
to_index: usize,
|
|
) -> Option<Self::Span> {
|
|
Some(Span::new(from_index, to_index))
|
|
}
|
|
|
|
fn file_id(&self, _tag: Self::Span) -> Self::FileId {
|
|
0
|
|
}
|
|
|
|
fn file_name(&self, _file: Self::FileId) -> FileName {
|
|
FileName::Verbatim(format!("shell"))
|
|
}
|
|
|
|
fn byte_index(&self, _file: Self::FileId, _line: usize, _column: usize) -> Option<usize> {
|
|
unimplemented!("byte_index")
|
|
}
|
|
|
|
fn location(&self, _file: Self::FileId, byte_index: usize) -> Option<Location> {
|
|
let source = &self.snippet;
|
|
let mut seen_lines = 0;
|
|
let mut seen_bytes = 0;
|
|
|
|
for (pos, slice) in source.match_indices('\n') {
|
|
trace!(
|
|
"SEARCH={} SEEN={} POS={} SLICE={:?} LEN={} ALL={:?}",
|
|
byte_index,
|
|
seen_bytes,
|
|
pos,
|
|
slice,
|
|
source.len(),
|
|
source
|
|
);
|
|
|
|
if pos >= byte_index {
|
|
return Some(language_reporting::Location::new(
|
|
seen_lines,
|
|
byte_index - seen_bytes,
|
|
));
|
|
} else {
|
|
seen_lines += 1;
|
|
seen_bytes = pos;
|
|
}
|
|
}
|
|
|
|
if seen_lines == 0 {
|
|
Some(language_reporting::Location::new(0, byte_index))
|
|
} else {
|
|
panic!("byte index {} wasn't valid", byte_index);
|
|
}
|
|
}
|
|
|
|
fn line_span(&self, _file: Self::FileId, lineno: usize) -> Option<Self::Span> {
|
|
let source = &self.snippet;
|
|
let mut seen_lines = 0;
|
|
let mut seen_bytes = 0;
|
|
|
|
for (pos, _) in source.match_indices('\n') {
|
|
if seen_lines == lineno {
|
|
return Some(Span::new(seen_bytes, pos + 1));
|
|
} else {
|
|
seen_lines += 1;
|
|
seen_bytes = pos + 1;
|
|
}
|
|
}
|
|
|
|
if seen_lines == 0 {
|
|
Some(Span::new(0, self.snippet.len() - 1))
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
fn source(&self, span: Self::Span) -> Option<String> {
|
|
trace!("source(tag={:?}) snippet={:?}", span, self.snippet);
|
|
|
|
if span.start() > span.end() {
|
|
return None;
|
|
} else if span.end() > self.snippet.len() {
|
|
return None;
|
|
}
|
|
Some(span.slice(&self.snippet).to_string())
|
|
}
|
|
}
|