use crate::common::safe_slice;
use crate::graph::canonical_fqn::FqnBuilder;
use crate::ingest::{ScopeSeparator, ScopeStack, SymbolFact, SymbolKind};
use crate::references::{CallFact, ReferenceFact};
use anyhow::Result;
use std::path::PathBuf;
pub struct CppParser {
pub(crate) parser: tree_sitter::Parser,
}
impl CppParser {
pub fn new() -> Result<Self> {
let mut parser = tree_sitter::Parser::new();
parser.set_language(&tree_sitter_cpp::language())?;
Ok(Self { parser })
}
pub(crate) fn from_parser(parser: tree_sitter::Parser) -> Self {
Self { parser }
}
pub fn extract_symbols(&mut self, file_path: PathBuf, source: &[u8]) -> Vec<SymbolFact> {
let tree = match self.parser.parse(source, None) {
Some(t) => t,
None => return Vec::new(), };
let root_node = tree.root_node();
let mut facts = Vec::new();
let mut scope_stack = ScopeStack::new(ScopeSeparator::DoubleColon);
let package_name = ".";
self.walk_tree_with_scope(
&root_node,
source,
&file_path,
&mut facts,
&mut scope_stack,
package_name,
);
facts
}
fn extract_name(
&self,
node: &tree_sitter::Node,
source: &[u8],
node_kind: &str,
) -> Option<String> {
if node_kind == "namespace_definition" {
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
if child.kind() == "namespace_identifier" {
let name_bytes = safe_slice(source, child.start_byte(), child.end_byte())?;
return std::str::from_utf8(name_bytes).ok().map(|s| s.to_string());
}
}
return None;
}
self.find_name_recursive(node, source)
}
fn find_name_recursive(&self, node: &tree_sitter::Node, source: &[u8]) -> Option<String> {
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
match child.kind() {
"identifier" | "type_identifier" => {
let name_bytes = safe_slice(source, child.start_byte(), child.end_byte())?;
return std::str::from_utf8(name_bytes).ok().map(|s| s.to_string());
}
"function_declarator"
| "parameter_list"
| "field_declaration_list"
| "template_parameter_list" => {
if let Some(name) = self.find_name_recursive(&child, source) {
return Some(name);
}
}
_ => {}
}
}
None
}
pub fn extract_symbols_with_parser(
parser: &mut tree_sitter::Parser,
file_path: PathBuf,
source: &[u8],
) -> Vec<SymbolFact> {
let tree = match parser.parse(source, None) {
Some(t) => t,
None => return Vec::new(),
};
let root_node = tree.root_node();
let mut facts = Vec::new();
let mut scope_stack = ScopeStack::new(ScopeSeparator::DoubleColon);
let package_name = ".";
Self::walk_tree_with_scope_static(
&root_node,
source,
&file_path,
&mut facts,
&mut scope_stack,
package_name,
);
facts
}
fn walk_tree_with_scope_static(
node: &tree_sitter::Node,
source: &[u8],
file_path: &PathBuf,
facts: &mut Vec<SymbolFact>,
scope_stack: &mut ScopeStack,
package_name: &str,
) {
let kind = node.kind();
if kind == "template_declaration" {
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
Self::walk_tree_with_scope_static(
&child,
source,
file_path,
facts,
scope_stack,
package_name,
);
}
return;
}
if kind == "namespace_definition" {
if let Some(name) = Self::extract_name_static(node, source, kind) {
if !name.is_empty() {
if let Some(fact) = Self::extract_symbol_with_fqn_static(
node,
source,
file_path,
scope_stack,
package_name,
) {
facts.push(fact);
}
scope_stack.push(&name);
}
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
Self::walk_tree_with_scope_static(
&child,
source,
file_path,
facts,
scope_stack,
package_name,
);
}
if !name.is_empty() {
scope_stack.pop();
}
return;
}
}
if let Some(fact) =
Self::extract_symbol_with_fqn_static(node, source, file_path, scope_stack, package_name)
{
facts.push(fact);
}
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
Self::walk_tree_with_scope_static(
&child,
source,
file_path,
facts,
scope_stack,
package_name,
);
}
}
fn extract_symbol_with_fqn_static(
node: &tree_sitter::Node,
source: &[u8],
file_path: &PathBuf,
scope_stack: &ScopeStack,
package_name: &str,
) -> Option<SymbolFact> {
let kind = node.kind();
let symbol_kind = match kind {
"function_definition" => SymbolKind::Function,
"class_specifier" => SymbolKind::Class,
"struct_specifier" => SymbolKind::Class,
"namespace_definition" => SymbolKind::Namespace,
_ => return None,
};
let name = Self::extract_name_static(node, source, kind)?;
let normalized_kind = symbol_kind.normalized_key().to_string();
let fqn = scope_stack.fqn_for_symbol(&name);
let builder = FqnBuilder::new(
package_name.to_string(),
file_path.to_string_lossy().to_string(),
ScopeSeparator::DoubleColon,
);
let canonical_fqn = builder.canonical(scope_stack, symbol_kind.clone(), &name);
let display_fqn = builder.display(scope_stack, symbol_kind.clone(), &name);
Some(SymbolFact {
file_path: file_path.clone(),
kind: symbol_kind,
kind_normalized: normalized_kind,
name: Some(name.clone()),
fqn: Some(fqn),
canonical_fqn: Some(canonical_fqn),
display_fqn: Some(display_fqn),
byte_start: node.start_byte(),
byte_end: node.end_byte(),
start_line: node.start_position().row + 1,
start_col: node.start_position().column,
end_line: node.end_position().row + 1,
end_col: node.end_position().column,
})
}
fn extract_name_static(
node: &tree_sitter::Node,
source: &[u8],
node_kind: &str,
) -> Option<String> {
if node_kind == "namespace_definition" {
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
if child.kind() == "namespace_identifier" {
let name_bytes = safe_slice(source, child.start_byte(), child.end_byte())?;
return std::str::from_utf8(name_bytes).ok().map(|s| s.to_string());
}
}
return None;
}
Self::find_name_recursive_static(node, source)
}
fn find_name_recursive_static(node: &tree_sitter::Node, source: &[u8]) -> Option<String> {
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
match child.kind() {
"identifier" | "type_identifier" => {
let name_bytes = &source[child.start_byte()..child.end_byte()];
return std::str::from_utf8(name_bytes).ok().map(|s| s.to_string());
}
"function_declarator"
| "parameter_list"
| "field_declaration_list"
| "template_parameter_list" => {
if let Some(name) = Self::find_name_recursive_static(&child, source) {
return Some(name);
}
}
_ => {}
}
}
None
}
fn walk_tree_with_scope(
&self,
node: &tree_sitter::Node,
source: &[u8],
file_path: &PathBuf,
facts: &mut Vec<SymbolFact>,
scope_stack: &mut ScopeStack,
package_name: &str,
) {
let kind = node.kind();
if kind == "template_declaration" {
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
self.walk_tree_with_scope(
&child,
source,
file_path,
facts,
scope_stack,
package_name,
);
}
return;
}
if kind == "namespace_definition" {
if let Some(name) = self.extract_name(node, source, kind) {
if !name.is_empty() {
if let Some(fact) = self.extract_symbol_with_fqn(
node,
source,
file_path,
scope_stack,
package_name,
) {
facts.push(fact);
}
scope_stack.push(&name);
}
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
self.walk_tree_with_scope(
&child,
source,
file_path,
facts,
scope_stack,
package_name,
);
}
if !name.is_empty() {
scope_stack.pop();
}
return;
}
}
if let Some(fact) =
self.extract_symbol_with_fqn(node, source, file_path, scope_stack, package_name)
{
facts.push(fact);
}
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
self.walk_tree_with_scope(&child, source, file_path, facts, scope_stack, package_name);
}
}
fn extract_symbol_with_fqn(
&self,
node: &tree_sitter::Node,
source: &[u8],
file_path: &PathBuf,
scope_stack: &ScopeStack,
package_name: &str,
) -> Option<SymbolFact> {
let kind = node.kind();
let symbol_kind = match kind {
"function_definition" => SymbolKind::Function,
"class_specifier" => SymbolKind::Class,
"struct_specifier" => SymbolKind::Class,
"namespace_definition" => SymbolKind::Namespace,
_ => return None, };
let name = self.extract_name(node, source, kind)?;
let normalized_kind = symbol_kind.normalized_key().to_string();
let fqn = scope_stack.fqn_for_symbol(&name);
let builder = FqnBuilder::new(
package_name.to_string(),
file_path.to_string_lossy().to_string(),
ScopeSeparator::DoubleColon,
);
let canonical_fqn = builder.canonical(scope_stack, symbol_kind.clone(), &name);
let display_fqn = builder.display(scope_stack, symbol_kind.clone(), &name);
Some(SymbolFact {
file_path: file_path.clone(),
kind: symbol_kind,
kind_normalized: normalized_kind,
name: Some(name.clone()),
fqn: Some(fqn),
canonical_fqn: Some(canonical_fqn),
display_fqn: Some(display_fqn),
byte_start: node.start_byte(),
byte_end: node.end_byte(),
start_line: node.start_position().row + 1,
start_col: node.start_position().column,
end_line: node.end_position().row + 1,
end_col: node.end_position().column,
})
}
pub fn extract_references(
&mut self,
file_path: PathBuf,
source: &[u8],
symbols: &[SymbolFact],
) -> Vec<ReferenceFact> {
let tree = match self.parser.parse(source, None) {
Some(t) => t,
None => return Vec::new(),
};
let root_node = tree.root_node();
let mut references = Vec::new();
self.walk_tree_for_references(&root_node, source, &file_path, symbols, &mut references);
references
}
fn walk_tree_for_references(
&self,
node: &tree_sitter::Node,
source: &[u8],
file_path: &PathBuf,
symbols: &[SymbolFact],
references: &mut Vec<ReferenceFact>,
) {
if let Some(reference) = self.extract_reference(node, source, file_path, symbols) {
references.push(reference);
}
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
self.walk_tree_for_references(&child, source, file_path, symbols, references);
}
}
fn extract_reference(
&self,
node: &tree_sitter::Node,
source: &[u8],
file_path: &PathBuf,
symbols: &[SymbolFact],
) -> Option<ReferenceFact> {
if node.kind() != "identifier" && node.kind() != "type_identifier" {
return None;
}
let text_bytes = &source[node.start_byte()..node.end_byte()];
let text = std::str::from_utf8(text_bytes).ok()?;
let referenced_symbol = symbols
.iter()
.find(|s| s.name.as_ref().map(|n| n == text).unwrap_or(false))?;
let ref_start = node.start_byte();
if ref_start < referenced_symbol.byte_end {
return None;
}
Some(ReferenceFact {
file_path: file_path.clone(),
referenced_symbol: text.to_string(),
byte_start: ref_start,
byte_end: node.end_byte(),
start_line: node.start_position().row + 1,
start_col: node.start_position().column,
end_line: node.end_position().row + 1,
end_col: node.end_position().column,
})
}
pub fn extract_calls(
&mut self,
file_path: PathBuf,
source: &[u8],
symbols: &[SymbolFact],
) -> Vec<CallFact> {
let tree = match self.parser.parse(source, None) {
Some(t) => t,
None => return Vec::new(),
};
let root_node = tree.root_node();
let mut calls = Vec::new();
let symbol_map: std::collections::HashMap<String, &SymbolFact> = symbols
.iter()
.filter_map(|s| s.name.as_ref().map(|name| (name.clone(), s)))
.collect();
let functions: Vec<&SymbolFact> = symbols
.iter()
.filter(|s| s.kind == SymbolKind::Function)
.collect();
self.walk_tree_for_calls(
&root_node,
source,
&file_path,
&symbol_map,
&functions,
&mut calls,
);
calls
}
fn walk_tree_for_calls(
&self,
node: &tree_sitter::Node,
source: &[u8],
file_path: &PathBuf,
symbol_map: &std::collections::HashMap<String, &SymbolFact>,
_functions: &[&SymbolFact],
calls: &mut Vec<CallFact>,
) {
self.walk_tree_for_calls_with_caller(node, source, file_path, symbol_map, None, calls);
}
fn walk_tree_for_calls_with_caller(
&self,
node: &tree_sitter::Node,
source: &[u8],
file_path: &PathBuf,
symbol_map: &std::collections::HashMap<String, &SymbolFact>,
current_caller: Option<&SymbolFact>,
calls: &mut Vec<CallFact>,
) {
let kind = node.kind();
let caller: Option<&SymbolFact> = if kind == "function_definition" {
self.extract_function_name(node, source)
.and_then(|name| symbol_map.get(&name).copied())
} else {
current_caller
};
if kind == "call_expression" {
if let Some(caller_fact) = caller {
self.extract_calls_in_node(node, source, file_path, caller_fact, symbol_map, calls);
}
}
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
self.walk_tree_for_calls_with_caller(
&child, source, file_path, symbol_map, caller, calls,
);
}
}
fn extract_function_name(&self, node: &tree_sitter::Node, source: &[u8]) -> Option<String> {
self.find_name_recursive(node, source)
}
fn extract_calls_in_node(
&self,
node: &tree_sitter::Node,
source: &[u8],
file_path: &PathBuf,
caller: &SymbolFact,
symbol_map: &std::collections::HashMap<String, &SymbolFact>,
calls: &mut Vec<CallFact>,
) {
if node.kind() == "call_expression" {
if let Some(callee_name) = self.extract_callee_from_call(node, source) {
if symbol_map.contains_key(&callee_name) {
let node_start = node.start_byte();
let node_end = node.end_byte();
let call_fact = CallFact {
file_path: file_path.clone(),
caller: caller.name.clone().unwrap_or_default(),
callee: callee_name,
caller_symbol_id: None,
callee_symbol_id: None,
byte_start: node_start,
byte_end: node_end,
start_line: node.start_position().row + 1,
start_col: node.start_position().column,
end_line: node.end_position().row + 1,
end_col: node.end_position().column,
};
calls.push(call_fact);
}
}
}
}
fn extract_callee_from_call(&self, node: &tree_sitter::Node, source: &[u8]) -> Option<String> {
let mut cursor = node.walk();
for child in node.children(&mut cursor) {
if child.kind() == "identifier" {
let name_bytes = &source[child.start_byte()..child.end_byte()];
return std::str::from_utf8(name_bytes).ok().map(|s| s.to_string());
}
}
None
}
}
impl Default for CppParser {
fn default() -> Self {
Self::new().expect("Failed to create C++ parser") }
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_extract_simple_function() {
let mut parser = CppParser::new().unwrap();
let source = b"void foo() {\n return;\n}\n";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert_eq!(facts.len(), 1);
assert_eq!(facts[0].name, Some("foo".to_string()));
assert_eq!(facts[0].kind, SymbolKind::Function);
}
#[test]
fn test_extract_class() {
let mut parser = CppParser::new().unwrap();
let source = b"class MyClass {\npublic:\n void method();\n};\n";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert_eq!(facts.len(), 1);
assert_eq!(facts[0].name, Some("MyClass".to_string()));
assert_eq!(facts[0].kind, SymbolKind::Class);
}
#[test]
fn test_extract_struct() {
let mut parser = CppParser::new().unwrap();
let source = b"struct Point {\n int x;\n int y;\n};\n";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert_eq!(facts.len(), 1);
assert_eq!(facts[0].name, Some("Point".to_string()));
assert_eq!(facts[0].kind, SymbolKind::Class);
}
#[test]
fn test_extract_namespace() {
let mut parser = CppParser::new().unwrap();
let source = b"namespace MyNamespace {\n class Foo {};\n}\n";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert!(facts.len() >= 1);
let namespaces: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Namespace)
.collect();
assert_eq!(namespaces.len(), 1);
assert_eq!(namespaces[0].name, Some("MyNamespace".to_string()));
}
#[test]
fn test_extract_template_class() {
let mut parser = CppParser::new().unwrap();
let source = b"template<typename T>\nclass TemplateClass {\n T value;\n};\n";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert_eq!(facts.len(), 1);
assert_eq!(facts[0].name, Some("TemplateClass".to_string()));
assert_eq!(facts[0].kind, SymbolKind::Class);
}
#[test]
fn test_extract_nested_namespace() {
let mut parser = CppParser::new().unwrap();
let source = b"namespace Outer {\n namespace Inner {\n class Foo {};\n }\n}\n";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
let namespaces: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Namespace)
.collect();
assert_eq!(namespaces.len(), 2);
}
#[test]
fn test_extract_multiple_symbols() {
let mut parser = CppParser::new().unwrap();
let source = b"
void foo() {}
class Bar {
void method();
};
namespace Baz {
struct Nested {};
}
";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert!(facts.len() >= 4);
let functions: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Function)
.collect();
assert_eq!(functions.len(), 1);
assert_eq!(functions[0].name, Some("foo".to_string()));
let classes: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Class)
.collect();
assert_eq!(classes.len(), 2);
let namespaces: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Namespace)
.collect();
assert_eq!(namespaces.len(), 1);
assert_eq!(namespaces[0].name, Some("Baz".to_string()));
}
#[test]
fn test_empty_file() {
let mut parser = CppParser::new().unwrap();
let source = b"";
let facts = parser.extract_symbols(PathBuf::from("empty.cpp"), source);
assert_eq!(facts.len(), 0);
}
#[test]
fn test_syntax_error_returns_empty() {
let mut parser = CppParser::new().unwrap();
let source = b"void broken(\n // invalid C++";
let facts = parser.extract_symbols(PathBuf::from("broken.cpp"), source);
assert!(
facts.len() < 10,
"Syntax error should not produce many symbols"
);
}
#[test]
fn test_byte_spans_within_bounds() {
let mut parser = CppParser::new().unwrap();
let source = b"void foo() { return; }";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert_eq!(facts.len(), 1);
let fact = &facts[0];
assert!(fact.byte_start < fact.byte_end);
assert!(fact.byte_end <= source.len());
}
#[test]
fn test_line_column_positions() {
let mut parser = CppParser::new().unwrap();
let source = b"void foo() {\n return;\n}\n";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert_eq!(facts.len(), 1);
let fact = &facts[0];
assert_eq!(fact.start_line, 1);
assert_eq!(fact.start_col, 0); }
#[test]
fn test_template_function() {
let mut parser = CppParser::new().unwrap();
let source = b"template<typename T>\nvoid template_func(T arg) {}\n";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert_eq!(facts.len(), 1);
assert_eq!(facts[0].name, Some("template_func".to_string()));
assert_eq!(facts[0].kind, SymbolKind::Function);
}
#[test]
fn test_fqn_simple_namespace() {
let mut parser = CppParser::new().unwrap();
let source = b"
namespace MyNamespace {
void my_function() {}
}
";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
let funcs: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Function)
.collect();
assert_eq!(funcs.len(), 1);
assert_eq!(funcs[0].fqn, Some("MyNamespace::my_function".to_string()));
}
#[test]
fn test_fqn_nested_namespace() {
let mut parser = CppParser::new().unwrap();
let source = b"
namespace Outer {
namespace Inner {
class MyClass {};
}
}
";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
let classes: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Class)
.collect();
assert_eq!(classes.len(), 1);
assert_eq!(classes[0].fqn, Some("Outer::Inner::MyClass".to_string()));
}
#[test]
fn test_fqn_class_in_namespace() {
let mut parser = CppParser::new().unwrap();
let source = b"
namespace ns {
struct Point {
int x;
int y;
};
}
";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
let classes: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Class)
.collect();
assert_eq!(classes.len(), 1);
assert_eq!(classes[0].fqn, Some("ns::Point".to_string()));
}
#[test]
fn test_canonical_fqn_format() {
let mut parser = CppParser::new().unwrap();
let source = b"void foo() {}\n";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert_eq!(facts.len(), 1);
let fact = &facts[0];
assert!(fact.canonical_fqn.is_some());
let canonical = fact.canonical_fqn.as_ref().unwrap();
assert!(canonical.contains("::Function foo"));
assert!(canonical.contains("test.cpp"));
assert!(canonical.contains(".")); }
#[test]
fn test_display_fqn_format() {
let mut parser = CppParser::new().unwrap();
let source = b"void foo() {}\n";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
assert_eq!(facts.len(), 1);
let fact = &facts[0];
assert!(fact.display_fqn.is_some());
let display = fact.display_fqn.as_ref().unwrap();
assert_eq!(display, ".::foo"); }
#[test]
fn test_fqn_namespace_function() {
let mut parser = CppParser::new().unwrap();
let source = b"
namespace MyNamespace {
void my_function() {}
}
";
let facts = parser.extract_symbols(PathBuf::from("src/test.cpp"), source);
let funcs: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Function)
.collect();
assert_eq!(funcs.len(), 1);
let func = &funcs[0];
assert!(func.canonical_fqn.is_some());
let canonical = func.canonical_fqn.as_ref().unwrap();
assert!(canonical.contains("src/test.cpp"));
assert!(canonical.contains("Function my_function"));
assert!(func.display_fqn.is_some());
let display = func.display_fqn.as_ref().unwrap();
assert_eq!(display, ".::MyNamespace::my_function");
}
#[test]
fn test_canonical_fqn_nested_namespace() {
let mut parser = CppParser::new().unwrap();
let source = b"
namespace Outer {
namespace Inner {
void func() {}
}
}
";
let facts = parser.extract_symbols(PathBuf::from("test.cpp"), source);
let funcs: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Function)
.collect();
assert_eq!(funcs.len(), 1);
let func = &funcs[0];
assert_eq!(func.display_fqn, Some(".::Outer::Inner::func".to_string()));
assert!(func.canonical_fqn.is_some());
let canonical = func.canonical_fqn.as_ref().unwrap();
assert!(canonical.contains("test.cpp"));
assert!(canonical.contains("Function func"));
}
#[test]
fn test_canonical_fqn_class_in_namespace() {
let mut parser = CppParser::new().unwrap();
let source = b"
namespace ns {
struct Point {
int x;
int y;
};
}
";
let facts = parser.extract_symbols(PathBuf::from("src/geometry.cpp"), source);
let classes: Vec<_> = facts
.iter()
.filter(|f| f.kind == SymbolKind::Class)
.collect();
assert_eq!(classes.len(), 1);
let cls = &classes[0];
assert_eq!(cls.display_fqn, Some(".::ns::Point".to_string()));
assert!(cls.canonical_fqn.is_some());
let canonical = cls.canonical_fqn.as_ref().unwrap();
assert!(canonical.contains("src/geometry.cpp"));
assert!(canonical.contains("Struct Point")); }
}