use psl::{
datamodel_connector::constraint_names::ConstraintNames,
parser_database::{walkers, IndexType},
schema_ast::ast,
PreviewFeature,
};
use sql::{mssql::MssqlSchemaExt, postgres::PostgresSchemaExt};
use sql_schema_describer as sql;
use super::{IndexFieldPair, IntrospectionPair};
pub(crate) type IndexPair<'a> = IntrospectionPair<'a, Option<walkers::IndexWalker<'a>>, Option<sql::IndexWalker<'a>>>;
impl<'a> IndexPair<'a> {
pub(crate) fn previous_position(self) -> Option<ast::AttributeId> {
self.previous.map(|idx| idx.attribute_id())
}
pub(crate) fn name(self) -> Option<&'a str> {
self.previous.and_then(|i| i.name())
}
pub(crate) fn mapped_name(self) -> Option<&'a str> {
match self.next {
Some(next) => {
let columns = next.column_names().collect::<Vec<_>>();
let default = match next.index_type() {
sql::IndexType::Unique => ConstraintNames::unique_index_name(
next.table().name(),
&columns,
self.context.active_connector(),
),
_ => ConstraintNames::non_unique_index_name(
next.table().name(),
&columns,
self.context.active_connector(),
),
};
(next.name() != default).then(|| next.name())
}
None => self.previous.and_then(|prev| prev.mapped_name()),
}
}
pub(crate) fn index_type(self) -> sql::IndexType {
let preview_features = self.context.config.preview_features();
match self.next.map(|next| next.index_type()) {
Some(sql::IndexType::Fulltext) if !preview_features.contains(PreviewFeature::FullTextIndex) => {
sql::IndexType::Normal
}
Some(typ) => typ,
None => match self.previous.map(|prev| prev.index_type()) {
Some(IndexType::Unique) => sql::IndexType::Unique,
Some(IndexType::Fulltext) => sql::IndexType::Fulltext,
_ => sql::IndexType::Normal,
},
}
}
pub(crate) fn clustered(self) -> Option<bool> {
if !self.context.sql_family.is_mssql() {
return None;
}
let clustered = match self.next {
Some(next) => {
let ext: &MssqlSchemaExt = self.context.sql_schema.downcast_connector_data();
ext.index_is_clustered(next.id)
}
None => self.previous.and_then(|prev| prev.clustered()).unwrap_or(false),
};
if !clustered {
None
} else {
Some(clustered)
}
}
pub(crate) fn algorithm(self) -> Option<&'static str> {
if !self.context.sql_family().is_postgres() {
return None;
}
match (self.next, self.previous.and_then(|i| i.algorithm())) {
(Some(next), _) => {
let data: &PostgresSchemaExt = self.context.sql_schema.downcast_connector_data();
match data.index_algorithm(next.id) {
sql::postgres::SqlIndexAlgorithm::BTree => None,
sql::postgres::SqlIndexAlgorithm::Hash => Some("Hash"),
sql::postgres::SqlIndexAlgorithm::Gist => Some("Gist"),
sql::postgres::SqlIndexAlgorithm::Gin => Some("Gin"),
sql::postgres::SqlIndexAlgorithm::SpGist => Some("SpGist"),
sql::postgres::SqlIndexAlgorithm::Brin => Some("Brin"),
}
}
(None, Some(algo)) => match algo {
psl::parser_database::IndexAlgorithm::BTree => None,
psl::parser_database::IndexAlgorithm::Hash => Some("Hash"),
psl::parser_database::IndexAlgorithm::Gist => Some("Gist"),
psl::parser_database::IndexAlgorithm::Gin => Some("Gin"),
psl::parser_database::IndexAlgorithm::SpGist => Some("SpGist"),
psl::parser_database::IndexAlgorithm::Brin => Some("Brin"),
},
_ => None,
}
}
pub(crate) fn fields(self) -> Box<dyn Iterator<Item = IndexFieldPair<'a>> + 'a> {
match (self.next, self.previous) {
(Some(next), _) => {
let iter = next.columns().enumerate().map(move |(i, next)| {
let previous = self
.previous
.and_then(|p| p.fields().nth(i).and_then(|f| f.as_scalar_field()));
IntrospectionPair::new(self.context, previous, Some(next))
});
Box::new(iter)
}
(None, Some(prev)) => {
let iter = prev
.fields()
.filter_map(|f| f.as_scalar_field())
.map(move |prev| IntrospectionPair::new(self.context, Some(prev), None));
Box::new(iter)
}
_ => Box::new(std::iter::empty()),
}
}
pub(crate) fn field(self) -> Option<IndexFieldPair<'a>> {
self.defined_in_a_field().then(|| self.fields().next().unwrap())
}
pub(crate) fn adds_a_non_default_deferring(self) -> bool {
match (self.previous, self.next) {
(None, Some(next)) => self
.context
.flavour
.uses_non_default_index_deferring(self.context, next),
_ => false,
}
}
fn defined_in_a_field(self) -> bool {
if !matches!(self.index_type(), sql::IndexType::Unique) {
return false;
}
match (self.next, self.previous) {
(Some(next), _) => next.columns().len() == 1,
(_, Some(prev)) => prev.fields().len() == 1,
_ => false,
}
}
}