use eyre::eyre;
use petgraph::dot::Dot;
use petgraph::graph::NodeIndex;
use petgraph::stable_graph::StableGraph;
use std::any::TypeId;
use std::collections::HashMap;
use std::fmt::Debug;
use std::path::Path;
use crate::aggregate::entity::PgAggregateEntity;
use crate::control_file::ControlFile;
use crate::extension_sql::entity::{ExtensionSqlEntity, SqlDeclaredEntity};
use crate::extension_sql::SqlDeclared;
use crate::pg_extern::entity::PgExternEntity;
use crate::pg_trigger::entity::PgTriggerEntity;
use crate::positioning_ref::PositioningRef;
use crate::postgres_enum::entity::PostgresEnumEntity;
use crate::postgres_hash::entity::PostgresHashEntity;
use crate::postgres_ord::entity::PostgresOrdEntity;
use crate::postgres_type::entity::PostgresTypeEntity;
use crate::schema::entity::SchemaEntity;
use crate::to_sql::ToSql;
use crate::type_keyed;
use crate::{SqlGraphEntity, SqlGraphIdentifier, TypeMatch};
use super::{PgExternReturnEntity, PgExternReturnEntityIteratedItem};
#[derive(Debug, Clone, Copy, PartialEq, PartialOrd, Eq, Ord)]
pub enum SqlGraphRequires {
By,
ByArg,
ByReturn,
}
#[derive(Debug, Clone)]
pub struct PgrxSql {
pub control: ControlFile,
pub graph: StableGraph<SqlGraphEntity, SqlGraphRequires>,
pub graph_root: NodeIndex,
pub graph_bootstrap: Option<NodeIndex>,
pub graph_finalize: Option<NodeIndex>,
pub schemas: HashMap<SchemaEntity, NodeIndex>,
pub extension_sqls: HashMap<ExtensionSqlEntity, NodeIndex>,
pub externs: HashMap<PgExternEntity, NodeIndex>,
pub types: HashMap<PostgresTypeEntity, NodeIndex>,
pub builtin_types: HashMap<String, NodeIndex>,
pub enums: HashMap<PostgresEnumEntity, NodeIndex>,
pub ords: HashMap<PostgresOrdEntity, NodeIndex>,
pub hashes: HashMap<PostgresHashEntity, NodeIndex>,
pub aggregates: HashMap<PgAggregateEntity, NodeIndex>,
pub triggers: HashMap<PgTriggerEntity, NodeIndex>,
pub extension_name: String,
pub versioned_so: bool,
}
impl PgrxSql {
pub fn build(
entities: impl Iterator<Item = SqlGraphEntity>,
extension_name: String,
versioned_so: bool,
) -> eyre::Result<Self> {
let mut graph = StableGraph::new();
let mut entities = entities.collect::<Vec<_>>();
entities.sort();
let mut control: Option<ControlFile> = None;
let mut schemas: Vec<SchemaEntity> = Vec::default();
let mut extension_sqls: Vec<ExtensionSqlEntity> = Vec::default();
let mut externs: Vec<PgExternEntity> = Vec::default();
let mut types: Vec<PostgresTypeEntity> = Vec::default();
let mut enums: Vec<PostgresEnumEntity> = Vec::default();
let mut ords: Vec<PostgresOrdEntity> = Vec::default();
let mut hashes: Vec<PostgresHashEntity> = Vec::default();
let mut aggregates: Vec<PgAggregateEntity> = Vec::default();
let mut triggers: Vec<PgTriggerEntity> = Vec::default();
for entity in entities {
match entity {
SqlGraphEntity::ExtensionRoot(input_control) => {
control = Some(input_control);
}
SqlGraphEntity::Schema(input_schema) => {
schemas.push(input_schema);
}
SqlGraphEntity::CustomSql(input_sql) => {
extension_sqls.push(input_sql);
}
SqlGraphEntity::Function(input_function) => {
externs.push(input_function);
}
SqlGraphEntity::Type(input_type) => {
types.push(input_type);
}
SqlGraphEntity::BuiltinType(_) => (),
SqlGraphEntity::Enum(input_enum) => {
enums.push(input_enum);
}
SqlGraphEntity::Ord(input_ord) => {
ords.push(input_ord);
}
SqlGraphEntity::Hash(input_hash) => {
hashes.push(input_hash);
}
SqlGraphEntity::Aggregate(input_aggregate) => {
aggregates.push(input_aggregate);
}
SqlGraphEntity::Trigger(input_trigger) => {
triggers.push(input_trigger);
}
}
}
let control: ControlFile = control.expect("No control file found");
let root = graph.add_node(SqlGraphEntity::ExtensionRoot(control.clone()));
let (mapped_extension_sqls, bootstrap, finalize) =
initialize_extension_sqls(&mut graph, root, extension_sqls)?;
let mapped_schemas = initialize_schemas(&mut graph, bootstrap, finalize, schemas)?;
let mapped_enums = initialize_enums(&mut graph, root, bootstrap, finalize, enums)?;
let mapped_types = initialize_types(&mut graph, root, bootstrap, finalize, types)?;
let (mapped_externs, mut mapped_builtin_types) = initialize_externs(
&mut graph,
root,
bootstrap,
finalize,
externs,
&mapped_types,
&mapped_enums,
)?;
let mapped_ords = initialize_ords(&mut graph, root, bootstrap, finalize, ords)?;
let mapped_hashes = initialize_hashes(&mut graph, root, bootstrap, finalize, hashes)?;
let mapped_aggregates = initialize_aggregates(
&mut graph,
root,
bootstrap,
finalize,
aggregates,
&mut mapped_builtin_types,
&mapped_enums,
&mapped_types,
)?;
let mapped_triggers = initialize_triggers(&mut graph, root, bootstrap, finalize, triggers)?;
connect_schemas(&mut graph, &mapped_schemas, root);
connect_extension_sqls(
&mut graph,
&mapped_extension_sqls,
&mapped_schemas,
&mapped_types,
&mapped_enums,
&mapped_externs,
&mapped_triggers,
)?;
connect_enums(&mut graph, &mapped_enums, &mapped_schemas);
connect_types(&mut graph, &mapped_types, &mapped_schemas);
connect_externs(
&mut graph,
&mapped_externs,
&mapped_hashes,
&mapped_schemas,
&mapped_types,
&mapped_enums,
&mapped_builtin_types,
&mapped_extension_sqls,
&mapped_triggers,
)?;
connect_ords(
&mut graph,
&mapped_ords,
&mapped_schemas,
&mapped_types,
&mapped_enums,
&mapped_externs,
);
connect_hashes(
&mut graph,
&mapped_hashes,
&mapped_schemas,
&mapped_types,
&mapped_enums,
&mapped_externs,
);
connect_aggregates(
&mut graph,
&mapped_aggregates,
&mapped_schemas,
&mapped_types,
&mapped_enums,
&mapped_builtin_types,
&mapped_externs,
)?;
connect_triggers(&mut graph, &mapped_triggers, &mapped_schemas);
let this = Self {
control,
schemas: mapped_schemas,
extension_sqls: mapped_extension_sqls,
externs: mapped_externs,
types: mapped_types,
builtin_types: mapped_builtin_types,
enums: mapped_enums,
ords: mapped_ords,
hashes: mapped_hashes,
aggregates: mapped_aggregates,
triggers: mapped_triggers,
graph,
graph_root: root,
graph_bootstrap: bootstrap,
graph_finalize: finalize,
extension_name,
versioned_so,
};
Ok(this)
}
pub fn to_file(&self, file: impl AsRef<Path> + Debug) -> eyre::Result<()> {
use std::fs::{create_dir_all, File};
use std::io::Write;
let generated = self.to_sql()?;
let path = Path::new(file.as_ref());
let parent = path.parent();
if let Some(parent) = parent {
create_dir_all(parent)?;
}
let mut out = File::create(path)?;
write!(out, "{}", generated)?;
Ok(())
}
pub fn write(&self, out: &mut impl std::io::Write) -> eyre::Result<()> {
let generated = self.to_sql()?;
#[cfg(feature = "syntax-highlighting")]
{
use std::io::{stdout, IsTerminal};
if stdout().is_terminal() {
self.write_highlighted(out, &generated)?;
} else {
write!(*out, "{}", generated)?;
}
}
#[cfg(not(feature = "syntax-highlighting"))]
{
write!(*out, "{}", generated)?;
}
Ok(())
}
#[cfg(feature = "syntax-highlighting")]
fn write_highlighted(&self, out: &mut dyn std::io::Write, generated: &str) -> eyre::Result<()> {
use eyre::WrapErr as _;
use owo_colors::{OwoColorize, XtermColors};
use syntect::easy::HighlightLines;
use syntect::highlighting::{Style, ThemeSet};
use syntect::parsing::SyntaxSet;
use syntect::util::LinesWithEndings;
let ps = SyntaxSet::load_defaults_newlines();
let theme_bytes = include_str!("../assets/ansi.tmTheme").as_bytes();
let mut theme_reader = std::io::Cursor::new(theme_bytes);
let theme = ThemeSet::load_from_reader(&mut theme_reader)
.wrap_err("Couldn't parse theme for SQL highlighting, try piping to a file")?;
if let Some(syntax) = ps.find_syntax_by_extension("sql") {
let mut h = HighlightLines::new(syntax, &theme);
for line in LinesWithEndings::from(&generated) {
let ranges: Vec<(Style, &str)> = h.highlight_line(line, &ps)?;
for (style, content) in ranges {
if style.foreground.a == 0x01 {
write!(*out, "{}", content)?;
} else {
write!(*out, "{}", content.color(XtermColors::from(style.foreground.r)))?;
}
}
write!(*out, "\x1b[0m")?;
}
} else {
write!(*out, "{}", generated)?;
}
Ok(())
}
pub fn to_dot(&self, file: impl AsRef<Path> + Debug) -> eyre::Result<()> {
use std::fs::{create_dir_all, File};
use std::io::Write;
let generated = Dot::with_attr_getters(
&self.graph,
&[petgraph::dot::Config::EdgeNoLabel, petgraph::dot::Config::NodeNoLabel],
&|_graph, edge| {
match edge.weight() {
SqlGraphRequires::By => r#"color = "gray""#,
SqlGraphRequires::ByArg => r#"color = "black""#,
SqlGraphRequires::ByReturn => r#"dir = "back", color = "black""#,
}
.to_owned()
},
&|_graph, (_index, node)| {
let dot_id = node.dot_identifier();
match node {
SqlGraphEntity::Schema(_item) => format!(
"label = \"{dot_id}\", weight = 6, shape = \"tab\""
),
SqlGraphEntity::Function(_item) => format!(
"label = \"{dot_id}\", penwidth = 0, style = \"filled\", fillcolor = \"#ADC7C6\", weight = 4, shape = \"box\"",
),
SqlGraphEntity::Type(_item) => format!(
"label = \"{dot_id}\", penwidth = 0, style = \"filled\", fillcolor = \"#AE9BBD\", weight = 5, shape = \"oval\"",
),
SqlGraphEntity::BuiltinType(_item) => format!(
"label = \"{dot_id}\", shape = \"plain\""
),
SqlGraphEntity::Enum(_item) => format!(
"label = \"{dot_id}\", penwidth = 0, style = \"filled\", fillcolor = \"#C9A7C8\", weight = 5, shape = \"oval\""
),
SqlGraphEntity::Ord(_item) => format!(
"label = \"{dot_id}\", penwidth = 0, style = \"filled\", fillcolor = \"#FFCFD3\", weight = 5, shape = \"diamond\""
),
SqlGraphEntity::Hash(_item) => format!(
"label = \"{dot_id}\", penwidth = 0, style = \"filled\", fillcolor = \"#FFE4E0\", weight = 5, shape = \"diamond\""
),
SqlGraphEntity::Aggregate(_item) => format!(
"label = \"{dot_id}\", penwidth = 0, style = \"filled\", fillcolor = \"#FFE4E0\", weight = 5, shape = \"diamond\""
),
SqlGraphEntity::Trigger(_item) => format!(
"label = \"{dot_id}\", penwidth = 0, style = \"filled\", fillcolor = \"#FFE4E0\", weight = 5, shape = \"diamond\""
),
SqlGraphEntity::CustomSql(_item) => format!(
"label = \"{dot_id}\", weight = 3, shape = \"signature\""
),
SqlGraphEntity::ExtensionRoot(_item) => format!(
"label = \"{dot_id}\", shape = \"cylinder\""
),
}
},
);
let path = Path::new(file.as_ref());
let parent = path.parent();
if let Some(parent) = parent {
create_dir_all(parent)?;
}
let mut out = File::create(path)?;
write!(out, "{:?}", generated)?;
Ok(())
}
pub fn schema_alias_of(&self, item_index: &NodeIndex) -> Option<String> {
self.graph
.neighbors_undirected(*item_index)
.flat_map(|neighbor_index| match &self.graph[neighbor_index] {
SqlGraphEntity::Schema(s) => Some(String::from(s.name)),
SqlGraphEntity::ExtensionRoot(_control) => None,
_ => None,
})
.next()
}
pub fn schema_prefix_for(&self, target: &NodeIndex) -> String {
self.schema_alias_of(target).map(|v| (v + ".").to_string()).unwrap_or_default()
}
pub fn to_sql(&self) -> eyre::Result<String> {
let mut full_sql = String::new();
for nodes in petgraph::algo::tarjan_scc(&self.graph).iter().rev() {
let mut inner_sql = Vec::with_capacity(nodes.len());
for node in nodes {
let step = &self.graph[*node];
let sql = step.to_sql(self)?;
let trimmed = sql.trim();
if !trimmed.is_empty() {
inner_sql.push(format!("{trimmed}\n"))
}
}
if !inner_sql.is_empty() {
full_sql.push_str("/* <begin connected objects> */\n");
full_sql.push_str(&inner_sql.join("\n\n"));
full_sql.push_str("/* </end connected objects> */\n\n");
}
}
Ok(full_sql)
}
pub fn has_sql_declared_entity(&self, identifier: &SqlDeclared) -> Option<&SqlDeclaredEntity> {
self.extension_sqls.iter().find_map(|(item, _index)| {
item.creates
.iter()
.find(|create_entity| create_entity.has_sql_declared_entity(identifier))
})
}
pub fn get_module_pathname(&self) -> String {
if self.versioned_so {
let extname = &self.extension_name;
let extver = &self.control.default_version;
format!("$libdir/{}-{}", extname, extver)
} else {
String::from("MODULE_PATHNAME")
}
}
}
fn build_base_edges(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
index: NodeIndex,
root: NodeIndex,
bootstrap: Option<NodeIndex>,
finalize: Option<NodeIndex>,
) {
graph.add_edge(root, index, SqlGraphRequires::By);
if let Some(bootstrap) = bootstrap {
graph.add_edge(bootstrap, index, SqlGraphRequires::By);
}
if let Some(finalize) = finalize {
graph.add_edge(index, finalize, SqlGraphRequires::By);
}
}
#[allow(clippy::type_complexity)]
fn initialize_extension_sqls(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
root: NodeIndex,
extension_sqls: Vec<ExtensionSqlEntity>,
) -> eyre::Result<(HashMap<ExtensionSqlEntity, NodeIndex>, Option<NodeIndex>, Option<NodeIndex>)> {
let mut bootstrap = None;
let mut finalize = None;
let mut mapped_extension_sqls = HashMap::default();
for item in extension_sqls {
let entity: SqlGraphEntity = item.clone().into();
let index = graph.add_node(entity);
mapped_extension_sqls.insert(item.clone(), index);
if item.bootstrap {
if let Some(existing_index) = bootstrap {
let existing: &SqlGraphEntity = &graph[existing_index];
return Err(eyre!(
"Cannot have multiple `extension_sql!()` with `bootstrap` positioning, found `{}`, other was `{}`",
item.rust_identifier(),
existing.rust_identifier(),
));
}
bootstrap = Some(index)
}
if item.finalize {
if let Some(existing_index) = finalize {
let existing: &SqlGraphEntity = &graph[existing_index];
return Err(eyre!(
"Cannot have multiple `extension_sql!()` with `finalize` positioning, found `{}`, other was `{}`",
item.rust_identifier(),
existing.rust_identifier(),
));
}
finalize = Some(index)
}
}
for (item, index) in &mapped_extension_sqls {
graph.add_edge(root, *index, SqlGraphRequires::By);
if !item.bootstrap {
if let Some(bootstrap) = bootstrap {
graph.add_edge(bootstrap, *index, SqlGraphRequires::By);
}
}
if !item.finalize {
if let Some(finalize) = finalize {
graph.add_edge(*index, finalize, SqlGraphRequires::By);
}
}
}
Ok((mapped_extension_sqls, bootstrap, finalize))
}
pub fn find_positioning_ref_target<'a>(
positioning_ref: &'a PositioningRef,
types: &'a HashMap<PostgresTypeEntity, NodeIndex>,
enums: &'a HashMap<PostgresEnumEntity, NodeIndex>,
externs: &'a HashMap<PgExternEntity, NodeIndex>,
schemas: &'a HashMap<SchemaEntity, NodeIndex>,
extension_sqls: &'a HashMap<ExtensionSqlEntity, NodeIndex>,
triggers: &'a HashMap<PgTriggerEntity, NodeIndex>,
) -> Option<&'a NodeIndex> {
match positioning_ref {
PositioningRef::FullPath(path) => {
let segments = path.split("::").collect::<Vec<_>>();
let last_segment = segments.last().expect("Expected at least one segment.");
let rest = &segments[..segments.len() - 1];
let module_path = rest.join("::");
for (other, other_index) in types {
if *last_segment == other.name && other.module_path.ends_with(&module_path) {
return Some(other_index);
}
}
for (other, other_index) in enums {
if last_segment == &other.name && other.module_path.ends_with(&module_path) {
return Some(other_index);
}
}
for (other, other_index) in externs {
if *last_segment == other.unaliased_name
&& other.module_path.ends_with(&module_path)
{
return Some(other_index);
}
}
for (other, other_index) in schemas {
if other.module_path.ends_with(path) {
return Some(other_index);
}
}
for (other, other_index) in triggers {
if last_segment == &other.function_name && other.module_path.ends_with(&module_path)
{
return Some(other_index);
}
}
}
PositioningRef::Name(name) => {
for (other, other_index) in extension_sqls {
if other.name == name {
return Some(other_index);
}
}
}
};
None
}
fn connect_extension_sqls(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
extension_sqls: &HashMap<ExtensionSqlEntity, NodeIndex>,
schemas: &HashMap<SchemaEntity, NodeIndex>,
types: &HashMap<PostgresTypeEntity, NodeIndex>,
enums: &HashMap<PostgresEnumEntity, NodeIndex>,
externs: &HashMap<PgExternEntity, NodeIndex>,
triggers: &HashMap<PgTriggerEntity, NodeIndex>,
) -> eyre::Result<()> {
for (item, &index) in extension_sqls {
make_schema_connection(
graph,
"Extension SQL",
index,
&item.rust_identifier(),
item.module_path,
schemas,
);
for requires in &item.requires {
if let Some(target) = find_positioning_ref_target(
requires,
types,
enums,
externs,
schemas,
extension_sqls,
triggers,
) {
graph.add_edge(*target, index, SqlGraphRequires::By);
} else {
return Err(eyre!(
"Could not find `requires` target of `{}`{}: {}",
item.rust_identifier(),
match (item.file(), item.line()) {
(Some(file), Some(line)) => format!(" ({file}:{line})"),
_ => "".to_string(),
},
match requires {
PositioningRef::FullPath(path) => path.to_string(),
PositioningRef::Name(name) => format!(r#""{name}""#),
},
));
}
}
}
Ok(())
}
fn initialize_schemas(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
bootstrap: Option<NodeIndex>,
finalize: Option<NodeIndex>,
schemas: Vec<SchemaEntity>,
) -> eyre::Result<HashMap<SchemaEntity, NodeIndex>> {
let mut mapped_schemas = HashMap::default();
for item in schemas {
let entity = item.clone().into();
let index = graph.add_node(entity);
mapped_schemas.insert(item, index);
if let Some(bootstrap) = bootstrap {
graph.add_edge(bootstrap, index, SqlGraphRequires::By);
}
if let Some(finalize) = finalize {
graph.add_edge(index, finalize, SqlGraphRequires::By);
}
}
Ok(mapped_schemas)
}
fn connect_schemas(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
schemas: &HashMap<SchemaEntity, NodeIndex>,
root: NodeIndex,
) {
for index in schemas.values().copied() {
graph.add_edge(root, index, SqlGraphRequires::By);
}
}
fn initialize_enums(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
root: NodeIndex,
bootstrap: Option<NodeIndex>,
finalize: Option<NodeIndex>,
enums: Vec<PostgresEnumEntity>,
) -> eyre::Result<HashMap<PostgresEnumEntity, NodeIndex>> {
let mut mapped_enums = HashMap::default();
for item in enums {
let entity: SqlGraphEntity = item.clone().into();
let index = graph.add_node(entity);
mapped_enums.insert(item, index);
build_base_edges(graph, index, root, bootstrap, finalize);
}
Ok(mapped_enums)
}
fn connect_enums(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
enums: &HashMap<PostgresEnumEntity, NodeIndex>,
schemas: &HashMap<SchemaEntity, NodeIndex>,
) {
for (item, &index) in enums {
make_schema_connection(
graph,
"Enum",
index,
&item.rust_identifier(),
item.module_path,
schemas,
);
}
}
fn initialize_types(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
root: NodeIndex,
bootstrap: Option<NodeIndex>,
finalize: Option<NodeIndex>,
types: Vec<PostgresTypeEntity>,
) -> eyre::Result<HashMap<PostgresTypeEntity, NodeIndex>> {
let mut mapped_types = HashMap::default();
for item in types {
let entity = item.clone().into();
let index = graph.add_node(entity);
mapped_types.insert(item, index);
build_base_edges(graph, index, root, bootstrap, finalize);
}
Ok(mapped_types)
}
fn connect_types(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
types: &HashMap<PostgresTypeEntity, NodeIndex>,
schemas: &HashMap<SchemaEntity, NodeIndex>,
) {
for (item, &index) in types {
make_schema_connection(
graph,
"Type",
index,
&item.rust_identifier(),
item.module_path,
schemas,
);
}
}
fn initialize_externs(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
root: NodeIndex,
bootstrap: Option<NodeIndex>,
finalize: Option<NodeIndex>,
externs: Vec<PgExternEntity>,
mapped_types: &HashMap<PostgresTypeEntity, NodeIndex>,
mapped_enums: &HashMap<PostgresEnumEntity, NodeIndex>,
) -> eyre::Result<(HashMap<PgExternEntity, NodeIndex>, HashMap<String, NodeIndex>)> {
let mut mapped_externs = HashMap::default();
let mut mapped_builtin_types = HashMap::default();
for item in externs {
let entity: SqlGraphEntity = item.clone().into();
let index = graph.add_node(entity.clone());
mapped_externs.insert(item.clone(), index);
build_base_edges(graph, index, root, bootstrap, finalize);
for arg in &item.fn_args {
let found = mapped_types.keys().any(|ty_item| ty_item.id_matches(&arg.used_ty.ty_id))
|| mapped_enums.keys().any(|ty_item| ty_item.id_matches(&arg.used_ty.ty_id));
if !found {
mapped_builtin_types.entry(arg.used_ty.full_path.to_string()).or_insert_with(
|| {
graph.add_node(SqlGraphEntity::BuiltinType(
arg.used_ty.full_path.to_string(),
))
},
);
}
}
match &item.fn_return {
PgExternReturnEntity::None | PgExternReturnEntity::Trigger => (),
PgExternReturnEntity::Type { ty, .. } | PgExternReturnEntity::SetOf { ty, .. } => {
let found = mapped_types.keys().any(|ty_item| ty_item.id_matches(&ty.ty_id))
|| mapped_enums.keys().any(|ty_item| ty_item.id_matches(&ty.ty_id));
if !found {
mapped_builtin_types.entry(ty.full_path.to_string()).or_insert_with(|| {
graph.add_node(SqlGraphEntity::BuiltinType(ty.full_path.to_string()))
});
}
}
PgExternReturnEntity::Iterated { tys: iterated_returns, .. } => {
for PgExternReturnEntityIteratedItem { ty, .. } in iterated_returns {
let found = mapped_types.keys().any(|ty_item| ty_item.id_matches(&ty.ty_id))
|| mapped_enums.keys().any(|ty_item| ty_item.id_matches(&ty.ty_id));
if !found {
mapped_builtin_types.entry(ty.ty_source.to_string()).or_insert_with(|| {
graph.add_node(SqlGraphEntity::BuiltinType(ty.ty_source.to_string()))
});
}
}
}
}
}
Ok((mapped_externs, mapped_builtin_types))
}
fn connect_externs(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
externs: &HashMap<PgExternEntity, NodeIndex>,
hashes: &HashMap<PostgresHashEntity, NodeIndex>,
schemas: &HashMap<SchemaEntity, NodeIndex>,
types: &HashMap<PostgresTypeEntity, NodeIndex>,
enums: &HashMap<PostgresEnumEntity, NodeIndex>,
builtin_types: &HashMap<String, NodeIndex>,
extension_sqls: &HashMap<ExtensionSqlEntity, NodeIndex>,
triggers: &HashMap<PgTriggerEntity, NodeIndex>,
) -> eyre::Result<()> {
for (item, &index) in externs {
let mut found_schema_declaration = false;
let mut has_explicit_requires = false;
for extern_attr in &item.extern_attrs {
match extern_attr {
crate::ExternArgs::Requires(requirements) => {
for requires in requirements {
if let Some(target) = find_positioning_ref_target(
requires,
types,
enums,
externs,
schemas,
extension_sqls,
triggers,
) {
graph.add_edge(*target, index, SqlGraphRequires::By);
has_explicit_requires = true;
} else {
return Err(eyre!("Could not find `requires` target: {:?}", requires));
}
}
}
crate::ExternArgs::Schema(declared_schema_name) => {
for (schema, schema_index) in schemas {
if schema.name == declared_schema_name {
graph.add_edge(*schema_index, index, SqlGraphRequires::By);
found_schema_declaration = true;
}
}
if !found_schema_declaration {
return Err(eyre!("Got manual `schema = \"{declared_schema_name}\"` setting, but that schema did not exist."));
}
}
_ => (),
}
}
if !found_schema_declaration {
make_schema_connection(
graph,
"Extern",
index,
&item.rust_identifier(),
item.module_path,
schemas,
);
}
for (hash_item, &hash_index) in hashes {
if item.module_path == hash_item.module_path
&& item.name == hash_item.name.to_lowercase() + "_eq"
{
graph.add_edge(index, hash_index, SqlGraphRequires::By);
}
}
for arg in &item.fn_args {
let found = types
.iter()
.map(type_keyed)
.chain(enums.iter().map(type_keyed))
.find(|(item, _)| item.id_matches(&arg.used_ty.ty_id));
if let Some((_, ty_index)) = found {
graph.add_edge(*ty_index, index, SqlGraphRequires::ByArg);
} else {
let builtin_index = builtin_types.get(arg.used_ty.full_path).unwrap_or_else(|| {
panic!("Could not fetch Builtin Type {}.", arg.used_ty.full_path)
});
graph.add_edge(*builtin_index, index, SqlGraphRequires::ByArg);
}
for (ext_item, ext_index) in extension_sqls {
if ext_item
.has_sql_declared_entity(&SqlDeclared::Type(arg.used_ty.full_path.to_string()))
.is_some()
{
if !has_explicit_requires {
graph.add_edge(*ext_index, index, SqlGraphRequires::ByArg);
}
} else if ext_item
.has_sql_declared_entity(&SqlDeclared::Enum(arg.used_ty.full_path.to_string()))
.is_some()
{
graph.add_edge(*ext_index, index, SqlGraphRequires::ByArg);
}
}
}
match &item.fn_return {
PgExternReturnEntity::None | PgExternReturnEntity::Trigger => (),
PgExternReturnEntity::Type { ty, .. } | PgExternReturnEntity::SetOf { ty, .. } => {
let found_index =
types.iter().map(type_keyed).chain(enums.iter().map(type_keyed)).find_map(
|(ty_item, index)| ty_item.id_matches(&ty.ty_id).then_some(index),
);
if let Some(ty_index) = found_index {
graph.add_edge(*ty_index, index, SqlGraphRequires::ByReturn);
} else {
let builtin_index =
builtin_types.get(&ty.full_path.to_string()).unwrap_or_else(|| {
panic!("Could not fetch Builtin Type {}.", ty.full_path)
});
graph.add_edge(*builtin_index, index, SqlGraphRequires::ByReturn);
for (ext_item, ext_index) in extension_sqls {
if ext_item
.has_sql_declared_entity(&SqlDeclared::Type(ty.full_path.into()))
.is_some()
|| ext_item
.has_sql_declared_entity(&SqlDeclared::Enum(ty.full_path.into()))
.is_some()
{
graph.add_edge(*ext_index, index, SqlGraphRequires::ByArg);
}
}
}
}
PgExternReturnEntity::Iterated { tys: iterated_returns, .. } => {
for PgExternReturnEntityIteratedItem { ty, .. } in iterated_returns {
let found_index =
types.iter().map(type_keyed).chain(enums.iter().map(type_keyed)).find_map(
|(ty_item, index)| ty_item.id_matches(&ty.ty_id).then_some(index),
);
if let Some(ty_index) = found_index {
graph.add_edge(*ty_index, index, SqlGraphRequires::ByReturn);
} else {
let builtin_index = builtin_types.get(ty.ty_source).unwrap_or_else(|| {
panic!("Could not fetch Builtin Type {}.", ty.ty_source)
});
graph.add_edge(*builtin_index, index, SqlGraphRequires::ByReturn);
for (ext_item, ext_index) in extension_sqls {
if ext_item
.has_sql_declared_entity(&SqlDeclared::Type(
ty.ty_source.to_string(),
))
.is_some()
|| ext_item
.has_sql_declared_entity(&SqlDeclared::Enum(
ty.ty_source.to_string(),
))
.is_some()
{
graph.add_edge(*ext_index, index, SqlGraphRequires::ByArg);
}
}
}
}
}
}
}
Ok(())
}
fn initialize_ords(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
root: NodeIndex,
bootstrap: Option<NodeIndex>,
finalize: Option<NodeIndex>,
ords: Vec<PostgresOrdEntity>,
) -> eyre::Result<HashMap<PostgresOrdEntity, NodeIndex>> {
let mut mapped_ords = HashMap::default();
for item in ords {
let entity = item.clone().into();
let index = graph.add_node(entity);
mapped_ords.insert(item.clone(), index);
build_base_edges(graph, index, root, bootstrap, finalize);
}
Ok(mapped_ords)
}
fn connect_ords(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
ords: &HashMap<PostgresOrdEntity, NodeIndex>,
schemas: &HashMap<SchemaEntity, NodeIndex>,
types: &HashMap<PostgresTypeEntity, NodeIndex>,
enums: &HashMap<PostgresEnumEntity, NodeIndex>,
externs: &HashMap<PgExternEntity, NodeIndex>,
) {
for (item, &index) in ords {
make_schema_connection(
graph,
"Ord",
index,
&item.rust_identifier(),
item.module_path,
schemas,
);
make_type_or_enum_connection(
graph,
"Ord",
index,
&item.rust_identifier(),
&item.id,
types,
enums,
);
for (extern_item, &extern_index) in externs {
let fn_matches = |fn_name| {
item.module_path == extern_item.module_path && extern_item.name == fn_name
};
let cmp_fn_matches = fn_matches(item.cmp_fn_name());
let lt_fn_matches = fn_matches(item.lt_fn_name());
let lte_fn_matches = fn_matches(item.le_fn_name());
let eq_fn_matches = fn_matches(item.eq_fn_name());
let gt_fn_matches = fn_matches(item.gt_fn_name());
let gte_fn_matches = fn_matches(item.ge_fn_name());
if cmp_fn_matches
|| lt_fn_matches
|| lte_fn_matches
|| eq_fn_matches
|| gt_fn_matches
|| gte_fn_matches
{
graph.add_edge(extern_index, index, SqlGraphRequires::By);
}
}
}
}
fn initialize_hashes(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
root: NodeIndex,
bootstrap: Option<NodeIndex>,
finalize: Option<NodeIndex>,
hashes: Vec<PostgresHashEntity>,
) -> eyre::Result<HashMap<PostgresHashEntity, NodeIndex>> {
let mut mapped_hashes = HashMap::default();
for item in hashes {
let entity: SqlGraphEntity = item.clone().into();
let index = graph.add_node(entity);
mapped_hashes.insert(item, index);
build_base_edges(graph, index, root, bootstrap, finalize);
}
Ok(mapped_hashes)
}
fn connect_hashes(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
hashes: &HashMap<PostgresHashEntity, NodeIndex>,
schemas: &HashMap<SchemaEntity, NodeIndex>,
types: &HashMap<PostgresTypeEntity, NodeIndex>,
enums: &HashMap<PostgresEnumEntity, NodeIndex>,
externs: &HashMap<PgExternEntity, NodeIndex>,
) {
for (item, &index) in hashes {
make_schema_connection(
graph,
"Hash",
index,
&item.rust_identifier(),
item.module_path,
schemas,
);
make_type_or_enum_connection(
graph,
"Hash",
index,
&item.rust_identifier(),
&item.id,
types,
enums,
);
if let Some((_, extern_index)) = externs.iter().find(|(extern_item, _)| {
item.module_path == extern_item.module_path && extern_item.name == item.fn_name()
}) {
graph.add_edge(*extern_index, index, SqlGraphRequires::By);
}
}
}
fn initialize_aggregates(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
root: NodeIndex,
bootstrap: Option<NodeIndex>,
finalize: Option<NodeIndex>,
aggregates: Vec<PgAggregateEntity>,
mapped_builtin_types: &mut HashMap<String, NodeIndex>,
mapped_enums: &HashMap<PostgresEnumEntity, NodeIndex>,
mapped_types: &HashMap<PostgresTypeEntity, NodeIndex>,
) -> eyre::Result<HashMap<PgAggregateEntity, NodeIndex>> {
let mut mapped_aggregates = HashMap::default();
for item in aggregates {
let entity: SqlGraphEntity = item.clone().into();
let index = graph.add_node(entity);
for arg in &item.args {
let found = mapped_types
.iter()
.map(type_keyed)
.chain(mapped_enums.iter().map(type_keyed))
.find(|(item, _)| item.id_matches(&arg.used_ty.ty_id));
if found.is_none() {
mapped_builtin_types.entry(arg.used_ty.full_path.to_string()).or_insert_with(
|| {
graph.add_node(SqlGraphEntity::BuiltinType(
arg.used_ty.full_path.to_string(),
))
},
);
}
}
mapped_aggregates.insert(item, index);
build_base_edges(graph, index, root, bootstrap, finalize);
}
Ok(mapped_aggregates)
}
fn connect_aggregate(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
item: &PgAggregateEntity,
index: NodeIndex,
schemas: &HashMap<SchemaEntity, NodeIndex>,
types: &HashMap<PostgresTypeEntity, NodeIndex>,
enums: &HashMap<PostgresEnumEntity, NodeIndex>,
builtin_types: &HashMap<String, NodeIndex>,
externs: &HashMap<PgExternEntity, NodeIndex>,
) -> eyre::Result<()> {
make_schema_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
item.module_path,
schemas,
);
make_type_or_enum_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&item.ty_id,
types,
enums,
);
for arg in &item.args {
let found = make_type_or_enum_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&arg.used_ty.ty_id,
types,
enums,
);
if !found {
let builtin_index = builtin_types.get(arg.used_ty.full_path).unwrap_or_else(|| {
panic!("Could not fetch Builtin Type {}.", arg.used_ty.full_path)
});
graph.add_edge(*builtin_index, index, SqlGraphRequires::ByArg);
}
}
for arg in item.direct_args.as_ref().unwrap_or(&vec![]) {
let found = make_type_or_enum_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&arg.used_ty.ty_id,
types,
enums,
);
if !found {
let builtin_index = builtin_types.get(arg.used_ty.full_path).unwrap_or_else(|| {
panic!("Could not fetch Builtin Type {}.", arg.used_ty.full_path)
});
graph.add_edge(*builtin_index, index, SqlGraphRequires::ByArg);
}
}
if let Some(arg) = &item.mstype {
let found = make_type_or_enum_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&arg.ty_id,
types,
enums,
);
if !found {
let builtin_index = builtin_types
.get(arg.full_path)
.unwrap_or_else(|| panic!("Could not fetch Builtin Type {}.", arg.full_path));
graph.add_edge(*builtin_index, index, SqlGraphRequires::ByArg);
}
}
make_extern_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&(item.module_path.to_string() + "::" + item.sfunc),
externs,
)?;
if let Some(value) = item.finalfunc {
make_extern_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&(item.module_path.to_string() + "::" + value),
externs,
)?;
}
if let Some(value) = item.combinefunc {
make_extern_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&(item.module_path.to_string() + "::" + value),
externs,
)?;
}
if let Some(value) = item.serialfunc {
make_extern_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&(item.module_path.to_string() + "::" + value),
externs,
)?;
}
if let Some(value) = item.deserialfunc {
make_extern_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&(item.module_path.to_string() + "::" + value),
externs,
)?;
}
if let Some(value) = item.msfunc {
make_extern_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&(item.module_path.to_string() + "::" + value),
externs,
)?;
}
if let Some(value) = item.minvfunc {
make_extern_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&(item.module_path.to_string() + "::" + value),
externs,
)?;
}
if let Some(value) = item.mfinalfunc {
make_extern_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&(item.module_path.to_string() + "::" + value),
externs,
)?;
}
if let Some(value) = item.sortop {
make_extern_connection(
graph,
"Aggregate",
index,
&item.rust_identifier(),
&(item.module_path.to_string() + "::" + value),
externs,
)?;
}
Ok(())
}
fn connect_aggregates(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
aggregates: &HashMap<PgAggregateEntity, NodeIndex>,
schemas: &HashMap<SchemaEntity, NodeIndex>,
types: &HashMap<PostgresTypeEntity, NodeIndex>,
enums: &HashMap<PostgresEnumEntity, NodeIndex>,
builtin_types: &HashMap<String, NodeIndex>,
externs: &HashMap<PgExternEntity, NodeIndex>,
) -> eyre::Result<()> {
for (item, &index) in aggregates {
connect_aggregate(graph, item, index, schemas, types, enums, builtin_types, externs)?
}
Ok(())
}
fn initialize_triggers(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
root: NodeIndex,
bootstrap: Option<NodeIndex>,
finalize: Option<NodeIndex>,
triggers: Vec<PgTriggerEntity>,
) -> eyre::Result<HashMap<PgTriggerEntity, NodeIndex>> {
let mut mapped_triggers = HashMap::default();
for item in triggers {
let entity: SqlGraphEntity = item.clone().into();
let index = graph.add_node(entity);
mapped_triggers.insert(item, index);
build_base_edges(graph, index, root, bootstrap, finalize);
}
Ok(mapped_triggers)
}
fn connect_triggers(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
triggers: &HashMap<PgTriggerEntity, NodeIndex>,
schemas: &HashMap<SchemaEntity, NodeIndex>,
) {
for (item, &index) in triggers {
make_schema_connection(
graph,
"Trigger",
index,
&item.rust_identifier(),
item.module_path,
schemas,
);
}
}
fn make_schema_connection(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
_kind: &str,
index: NodeIndex,
_rust_identifier: &str,
module_path: &str,
schemas: &HashMap<SchemaEntity, NodeIndex>,
) -> bool {
let mut found = false;
for (schema_item, &schema_index) in schemas {
if module_path == schema_item.module_path {
graph.add_edge(schema_index, index, SqlGraphRequires::By);
found = true;
break;
}
}
found
}
fn make_extern_connection(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
_kind: &str,
index: NodeIndex,
_rust_identifier: &str,
full_path: &str,
externs: &HashMap<PgExternEntity, NodeIndex>,
) -> eyre::Result<()> {
match externs.iter().find(|(extern_item, _)| full_path == extern_item.full_path) {
Some((_, extern_index)) => {
graph.add_edge(*extern_index, index, SqlGraphRequires::By);
Ok(())
}
None => Err(eyre!("Did not find connection `{full_path}` in {:#?}", {
let mut paths = externs.iter().map(|(v, _)| v.full_path).collect::<Vec<_>>();
paths.sort();
paths
})),
}
}
fn make_type_or_enum_connection(
graph: &mut StableGraph<SqlGraphEntity, SqlGraphRequires>,
_kind: &str,
index: NodeIndex,
_rust_identifier: &str,
ty_id: &TypeId,
types: &HashMap<PostgresTypeEntity, NodeIndex>,
enums: &HashMap<PostgresEnumEntity, NodeIndex>,
) -> bool {
types
.iter()
.map(type_keyed)
.chain(enums.iter().map(type_keyed))
.find(|(ty, _)| ty.id_matches(ty_id))
.map(|(_, ty_index)| graph.add_edge(*ty_index, index, SqlGraphRequires::By))
.is_some()
}