cairo-lang-doc 2.18.0

A collection of documentation processing utilities for the Cairo programming language.
Documentation
use cairo_lang_diagnostics::{DiagnosticAdded, DiagnosticsBuilder, Maybe};
use cairo_lang_filesystem::db::FilesGroup;
use cairo_lang_filesystem::ids::{FileKind, FileLongId, SmolStrId, VirtualFile};
use cairo_lang_formatter::{FormatterConfig, get_formatted_file};
use cairo_lang_parser::parser::Parser;
use cairo_lang_syntax::node::green::GreenNodeDetails;
use cairo_lang_syntax::node::kind::SyntaxKind;
use cairo_lang_syntax::node::{SyntaxNode, TypedSyntaxNode};
use cairo_lang_utils::Intern;
use salsa::Database;

use crate::documentable_item::DocumentableItemId;

/// A helper struct to map parts of item signature to the respective documentable items.
#[derive(Clone, Debug, PartialEq, Eq, salsa::Update)]
pub struct LocationLink<'db> {
    /// Link's start offset in documentable item's signature.
    pub start: usize,
    /// Link's end offset in documentable item's signature.
    pub end: usize,
    /// Linked item identifier.
    pub item_id: DocumentableItemId<'db>,
}

impl<'db> LocationLink<'db> {
    /// Creates a new instance of `LocationLink` moving `start` and `end` by given `offset`.
    pub fn new(start: usize, end: usize, item_id: DocumentableItemId<'db>, offset: usize) -> Self {
        Self { start: (start + offset), end: (end + offset), item_id }
    }
}

/// Extends `green_nodes` with all
/// [`cairo_lang_syntax::node::green::GreenNode`]s for a [`SyntaxNode`],
/// storing each node's [`SyntaxKind`] and text.
fn extend_green_nodes<'db>(
    db: &dyn Database,
    syntax_node: &SyntaxNode<'db>,
    green_nodes: &mut Vec<(SyntaxKind, String)>,
) {
    let green_node = syntax_node.green_node(db);

    match &green_node.details {
        GreenNodeDetails::Token(text) => green_nodes.push((green_node.kind, text.to_string(db))),
        GreenNodeDetails::Node { .. } => {
            let syntax_node_children = syntax_node.get_children(db);
            syntax_node_children.iter().for_each(|child| {
                extend_green_nodes(db, child, green_nodes);
            });
        }
    }
}

/// Creates a virtual file for further signature syntax processing.
fn get_virtual_syntax_file_signature<'db>(
    sig_db: &'db dyn Database,
    signature: &str,
) -> Maybe<SyntaxNode<'db>> {
    let virtual_file = FileLongId::Virtual(VirtualFile {
        parent: None,
        name: SmolStrId::from(sig_db, "string_to_format"),
        content: SmolStrId::from(sig_db, signature),
        code_mappings: [].into(),
        kind: FileKind::Module,
        original_item_removed: false,
    })
    .intern(sig_db);

    let mut diagnostics_builder = DiagnosticsBuilder::default();
    let content = sig_db.file_content(virtual_file).expect("File was just ensured to be in db");
    let syntax_file: SyntaxNode<'_> =
        Parser::parse_file(sig_db, &mut diagnostics_builder, virtual_file, content)
            .as_syntax_node();

    // Allow a single "Missing token '{'..." error
    if diagnostics_builder.error_count <= 1 { Ok(syntax_file) } else { Err(DiagnosticAdded) }
}

/// Calculates offsets for original and formatted signature,
/// thanks to that [`LocationLink`]s can be adjusted.
fn get_offsets(
    signature1: Vec<(SyntaxKind, String)>,
    signature2: Vec<(SyntaxKind, String)>,
) -> Vec<(usize, i32)> {
    let mut offset_vector: Vec<(usize, i32)> = Vec::new();
    let mut original_length_tracker = 0;
    let mut signature1_index = 0;
    let mut signature2_index = 0;

    while signature1_index < signature1.len() && signature2_index < signature2.len() {
        let (kind1, ref string1) = signature1[signature1_index];
        let (kind2, ref string2) = signature2[signature2_index];

        if kind1 == kind2 {
            signature1_index += 1;

            if string1.len() != string2.len() {
                offset_vector
                    .push((original_length_tracker, string2.len() as i32 - string1.len() as i32));
            }
            original_length_tracker += string1.len();
        } else {
            offset_vector.push((original_length_tracker, string2.len() as i32));
        }
        signature2_index += 1;
    }
    offset_vector
}

/// Adjusts [`LocationLink`]s based on differences created in signature syntax formatting.
pub fn move_location_links(
    mut location_links: Vec<LocationLink<'_>>,
    offset_vector: Vec<(usize, i32)>,
) -> Vec<LocationLink<'_>> {
    for link in &mut location_links {
        let mut new_start = link.start as i32;
        let mut new_end = link.end as i32;

        for (location, length) in offset_vector.iter() {
            if link.end < *location {
                break;
            }
            if link.start >= *location {
                new_start += length;
            }
            if link.end > *location {
                new_end += length;
            } else {
                break;
            }
        }
        link.start = new_start as usize;
        link.end = new_end as usize;
    }
    location_links
}

/// Performs a set of actions to return formatted signature with [`LocationLink`]s adjusted.
pub fn format_signature<'db>(
    db: &'db dyn Database,
    signature: String,
    location_links: Vec<LocationLink<'db>>,
) -> (String, Vec<LocationLink<'db>>) {
    match get_virtual_syntax_file_signature(db, &signature) {
        Ok(syntax_file) => {
            let formatted_file = get_formatted_file(db, &syntax_file, FormatterConfig::default());

            if !location_links.is_empty() {
                match get_virtual_syntax_file_signature(db, &formatted_file) {
                    Ok(syntax_file_formatted) => {
                        let mut nodes_original = Vec::new();
                        extend_green_nodes(db, &syntax_file, &mut nodes_original);
                        let mut nodes_formatted = Vec::new();
                        extend_green_nodes(db, &syntax_file_formatted, &mut nodes_formatted);

                        let offsets = get_offsets(nodes_original, nodes_formatted);
                        (
                            formatted_file.trim_end().to_string(),
                            move_location_links(location_links, offsets),
                        )
                    }
                    Err(_) => (formatted_file.trim_end().to_string(), location_links),
                }
            } else {
                (formatted_file.trim_end().to_string(), location_links)
            }
        }
        Err(_) => (signature, location_links),
    }
}