mirror of
				https://github.com/astral-sh/ruff.git
				synced 2025-10-26 09:58:17 +00:00 
			
		
		
		
	[ty] Remap Jupyter notebook cell indices in ruff_db (#19698)
	
		
			
	
		
	
	
		
	
		
			Some checks are pending
		
		
	
	
		
			
				
	
				CI / Determine changes (push) Waiting to run
				
			
		
			
				
	
				CI / cargo fmt (push) Waiting to run
				
			
		
			
				
	
				CI / cargo clippy (push) Blocked by required conditions
				
			
		
			
				
	
				CI / cargo test (linux) (push) Blocked by required conditions
				
			
		
			
				
	
				CI / cargo test (linux, release) (push) Blocked by required conditions
				
			
		
			
				
	
				CI / cargo test (windows) (push) Blocked by required conditions
				
			
		
			
				
	
				CI / cargo test (wasm) (push) Blocked by required conditions
				
			
		
			
				
	
				CI / cargo build (release) (push) Waiting to run
				
			
		
			
				
	
				CI / cargo build (msrv) (push) Blocked by required conditions
				
			
		
			
				
	
				CI / cargo fuzz build (push) Blocked by required conditions
				
			
		
			
				
	
				CI / fuzz parser (push) Blocked by required conditions
				
			
		
			
				
	
				CI / test scripts (push) Blocked by required conditions
				
			
		
			
				
	
				CI / ecosystem (push) Blocked by required conditions
				
			
		
			
				
	
				CI / Fuzz for new ty panics (push) Blocked by required conditions
				
			
		
			
				
	
				CI / cargo shear (push) Blocked by required conditions
				
			
		
			
				
	
				CI / python package (push) Waiting to run
				
			
		
			
				
	
				CI / pre-commit (push) Waiting to run
				
			
		
			
				
	
				CI / mkdocs (push) Waiting to run
				
			
		
			
				
	
				CI / formatter instabilities and black similarity (push) Blocked by required conditions
				
			
		
			
				
	
				CI / test ruff-lsp (push) Blocked by required conditions
				
			
		
			
				
	
				CI / check playground (push) Blocked by required conditions
				
			
		
			
				
	
				CI / benchmarks-instrumented (push) Blocked by required conditions
				
			
		
			
				
	
				CI / benchmarks-walltime (push) Blocked by required conditions
				
			
		
			
				
	
				[ty Playground] Release / publish (push) Waiting to run
				
			
		
		
	
	
				
					
				
			
		
			Some checks are pending
		
		
	
	CI / Determine changes (push) Waiting to run
				
			CI / cargo fmt (push) Waiting to run
				
			CI / cargo clippy (push) Blocked by required conditions
				
			CI / cargo test (linux) (push) Blocked by required conditions
				
			CI / cargo test (linux, release) (push) Blocked by required conditions
				
			CI / cargo test (windows) (push) Blocked by required conditions
				
			CI / cargo test (wasm) (push) Blocked by required conditions
				
			CI / cargo build (release) (push) Waiting to run
				
			CI / cargo build (msrv) (push) Blocked by required conditions
				
			CI / cargo fuzz build (push) Blocked by required conditions
				
			CI / fuzz parser (push) Blocked by required conditions
				
			CI / test scripts (push) Blocked by required conditions
				
			CI / ecosystem (push) Blocked by required conditions
				
			CI / Fuzz for new ty panics (push) Blocked by required conditions
				
			CI / cargo shear (push) Blocked by required conditions
				
			CI / python package (push) Waiting to run
				
			CI / pre-commit (push) Waiting to run
				
			CI / mkdocs (push) Waiting to run
				
			CI / formatter instabilities and black similarity (push) Blocked by required conditions
				
			CI / test ruff-lsp (push) Blocked by required conditions
				
			CI / check playground (push) Blocked by required conditions
				
			CI / benchmarks-instrumented (push) Blocked by required conditions
				
			CI / benchmarks-walltime (push) Blocked by required conditions
				
			[ty Playground] Release / publish (push) Waiting to run
				
			## Summary
This PR remaps ranges in Jupyter notebooks from simple `row:column`
indices in the concatenated source code to `cell:row:col` to match
Ruff's output. This is probably not a likely change to land upstream in
`annotate-snippets`, but I didn't see a good way around it.
The remapping logic is taken nearly verbatim from here:
cd6bf1457d/crates/ruff_linter/src/message/text.rs (L212-L222)
## Test Plan
New `full` rendering test for a notebook
I was mainly focused on Ruff, but in local tests this also works for ty:
```
error[invalid-assignment]: Object of type `Literal[1]` is not assignable to `str`
 --> Untitled.ipynb:cell 1:3:1
  |
1 | import math
2 |
3 | x: str = 1
  | ^
  |
info: rule `invalid-assignment` is enabled by default
error[invalid-assignment]: Object of type `Literal[1]` is not assignable to `str`
 --> Untitled.ipynb:cell 2:3:1
  |
1 | import math
2 |
3 | x: str = 1
  | ^
  |
info: rule `invalid-assignment` is enabled by default
```
This isn't a duplicate diagnostic, just an unimaginative example:
```py
# cell 1
import math
x: str = 1
# cell 2
import math
x: str = 1
```
			
			
This commit is contained in:
		
							parent
							
								
									b324ae1be3
								
							
						
					
					
						commit
						5bfffe1aa7
					
				
					 4 changed files with 280 additions and 41 deletions
				
			
		|  | @ -263,7 +263,11 @@ impl DisplaySet<'_> { | |||
|             if annotation.is_fixable { | ||||
|                 buffer.append(line_offset, "[", stylesheet.none); | ||||
|                 buffer.append(line_offset, "*", stylesheet.help); | ||||
|                 buffer.append(line_offset, "] ", stylesheet.none); | ||||
|                 buffer.append(line_offset, "]", stylesheet.none); | ||||
|                 // In the hide-severity case, we need a space instead of the colon and space below.
 | ||||
|                 if hide_severity { | ||||
|                     buffer.append(line_offset, " ", stylesheet.none); | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             if !is_annotation_empty(annotation) { | ||||
|  | @ -298,11 +302,15 @@ impl DisplaySet<'_> { | |||
|                 let lineno_color = stylesheet.line_no(); | ||||
|                 buffer.puts(line_offset, lineno_width, header_sigil, *lineno_color); | ||||
|                 buffer.puts(line_offset, lineno_width + 4, path, stylesheet.none); | ||||
|                 if let Some((col, row)) = pos { | ||||
|                     buffer.append(line_offset, ":", stylesheet.none); | ||||
|                     buffer.append(line_offset, col.to_string().as_str(), stylesheet.none); | ||||
|                 if let Some(Position { row, col, cell }) = pos { | ||||
|                     if let Some(cell) = cell { | ||||
|                         buffer.append(line_offset, ":", stylesheet.none); | ||||
|                         buffer.append(line_offset, &format!("cell {cell}"), stylesheet.none); | ||||
|                     } | ||||
|                     buffer.append(line_offset, ":", stylesheet.none); | ||||
|                     buffer.append(line_offset, row.to_string().as_str(), stylesheet.none); | ||||
|                     buffer.append(line_offset, ":", stylesheet.none); | ||||
|                     buffer.append(line_offset, col.to_string().as_str(), stylesheet.none); | ||||
|                 } | ||||
|                 Ok(()) | ||||
|             } | ||||
|  | @ -883,6 +891,13 @@ impl DisplaySourceAnnotation<'_> { | |||
|     } | ||||
| } | ||||
| 
 | ||||
| #[derive(Debug, PartialEq)] | ||||
| pub(crate) struct Position { | ||||
|     row: usize, | ||||
|     col: usize, | ||||
|     cell: Option<usize>, | ||||
| } | ||||
| 
 | ||||
| /// Raw line - a line which does not have the `lineno` part and is not considered
 | ||||
| /// a part of the snippet.
 | ||||
| #[derive(Debug, PartialEq)] | ||||
|  | @ -891,7 +906,7 @@ pub(crate) enum DisplayRawLine<'a> { | |||
|     /// slice in the project structure.
 | ||||
|     Origin { | ||||
|         path: &'a str, | ||||
|         pos: Option<(usize, usize)>, | ||||
|         pos: Option<Position>, | ||||
|         header_type: DisplayHeaderType, | ||||
|     }, | ||||
| 
 | ||||
|  | @ -1191,13 +1206,15 @@ fn format_snippet<'m>( | |||
|             "Non-empty file-level snippet that won't be rendered: {:?}", | ||||
|             snippet.source | ||||
|         ); | ||||
|         let header = format_header(origin, main_range, &[], is_first); | ||||
|         let header = format_header(origin, main_range, &[], is_first, snippet.cell_index); | ||||
|         return DisplaySet { | ||||
|             display_lines: header.map_or_else(Vec::new, |header| vec![header]), | ||||
|             margin: Margin::new(0, 0, 0, 0, term_width, 0), | ||||
|         }; | ||||
|     } | ||||
| 
 | ||||
|     let cell_index = snippet.cell_index; | ||||
| 
 | ||||
|     let mut body = format_body( | ||||
|         snippet, | ||||
|         need_empty_header, | ||||
|  | @ -1206,7 +1223,13 @@ fn format_snippet<'m>( | |||
|         anonymized_line_numbers, | ||||
|         cut_indicator, | ||||
|     ); | ||||
|     let header = format_header(origin, main_range, &body.display_lines, is_first); | ||||
|     let header = format_header( | ||||
|         origin, | ||||
|         main_range, | ||||
|         &body.display_lines, | ||||
|         is_first, | ||||
|         cell_index, | ||||
|     ); | ||||
| 
 | ||||
|     if let Some(header) = header { | ||||
|         body.display_lines.insert(0, header); | ||||
|  | @ -1226,6 +1249,7 @@ fn format_header<'a>( | |||
|     main_range: Option<usize>, | ||||
|     body: &[DisplayLine<'_>], | ||||
|     is_first: bool, | ||||
|     cell_index: Option<usize>, | ||||
| ) -> Option<DisplayLine<'a>> { | ||||
|     let display_header = if is_first { | ||||
|         DisplayHeaderType::Initial | ||||
|  | @ -1262,7 +1286,11 @@ fn format_header<'a>( | |||
| 
 | ||||
|         return Some(DisplayLine::Raw(DisplayRawLine::Origin { | ||||
|             path, | ||||
|             pos: Some((line_offset, col)), | ||||
|             pos: Some(Position { | ||||
|                 row: line_offset, | ||||
|                 col, | ||||
|                 cell: cell_index, | ||||
|             }), | ||||
|             header_type: display_header, | ||||
|         })); | ||||
|     } | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Brent Westbrook
						Brent Westbrook