sea-schema-sync 0.17.0-rc.16

🌿 SQL schema definition and discovery
Documentation
use crate::mysql::def::{CharSet, ColumnDefault, ColumnInfo, NumericAttr, StringAttr, Type};
use sea_query::{
    Alias, ColumnDef, DynIden, EscapeBuilder, Expr, Iden, IntoIden, Keyword, MysqlQueryBuilder,
    SimpleExpr, extension::mysql::MySqlType,
};
use std::fmt::Write;

impl ColumnInfo {
    pub fn write(&self) -> ColumnDef {
        let mut col_def = ColumnDef::new(Alias::new(self.name.as_str()));
        col_def = self.write_col_type(col_def);
        if !self.null {
            col_def.not_null();
        }
        if self.extra.auto_increment {
            col_def.auto_increment();
        }
        let mut extras = Vec::new();
        if let Some(default) = &self.default {
            let default_expr: SimpleExpr = match default {
                ColumnDefault::Null => Option::<bool>::None.into(),
                ColumnDefault::Int(int) => (*int).into(),
                ColumnDefault::Real(double) => (*double).into(),
                ColumnDefault::String(string) => string.into(),
                ColumnDefault::CustomExpr(string) => Expr::cust(string.to_owned()),
                ColumnDefault::CurrentTimestamp => Keyword::CurrentTimestamp.into(),
            };
            col_def.default(default_expr);
        }
        if self.extra.on_update_current_timestamp {
            extras.push("ON UPDATE CURRENT_TIMESTAMP".to_owned());
        }
        if !self.comment.is_empty() {
            let mut string = "".to_owned();
            write!(
                &mut string,
                "COMMENT '{}'",
                MysqlQueryBuilder.escape_string(&self.comment)
            )
            .unwrap();
            extras.push(string);
        }
        if !extras.is_empty() {
            col_def.extra(extras.join(" "));
        }
        col_def
    }

    pub fn write_col_type(&self, mut col_def: ColumnDef) -> ColumnDef {
        match &self.col_type {
            Type::Serial => {
                col_def
                    .big_unsigned()
                    .not_null()
                    .auto_increment()
                    .unique_key();
            }
            Type::Bit(num_attr) => {
                col_def.bit(num_attr.maximum);
            }
            Type::TinyInt(num_attr) => {
                match num_attr.unsigned {
                    Some(_) => col_def.tiny_unsigned(),
                    None => col_def.tiny_integer(),
                };
                col_def = self.write_num_attr(col_def, num_attr);
            }
            Type::Bool => {
                col_def.boolean();
            }
            Type::SmallInt(num_attr) => {
                match num_attr.unsigned {
                    Some(_) => col_def.small_unsigned(),
                    None => col_def.small_integer(),
                };
                col_def = self.write_num_attr(col_def, num_attr);
            }
            Type::MediumInt(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::Int(num_attr) => {
                match num_attr.unsigned {
                    Some(_) => col_def.unsigned(),
                    None => col_def.integer(),
                };
                col_def = self.write_num_attr(col_def, num_attr);
            }
            Type::BigInt(num_attr) => {
                match num_attr.unsigned {
                    Some(_) => col_def.big_unsigned(),
                    None => col_def.big_integer(),
                };
                col_def = self.write_num_attr(col_def, num_attr);
            }
            Type::Decimal(num_attr) => {
                match (num_attr.maximum, num_attr.decimal) {
                    (Some(maximum), Some(decimal)) => col_def.decimal_len(maximum, decimal),
                    _ => col_def.decimal(),
                };
                col_def = self.write_num_attr(col_def, num_attr);
            }
            Type::Float(num_attr) => {
                col_def.float();
                col_def = self.write_num_attr(col_def, num_attr);
            }
            Type::Double(num_attr) => {
                col_def.double();
                col_def = self.write_num_attr(col_def, num_attr);
            }
            Type::Date => {
                col_def.date();
            }
            Type::Time(_) => {
                col_def.time();
            }
            Type::DateTime(_) => {
                col_def.date_time();
            }
            Type::Timestamp(_) => {
                col_def.timestamp();
            }
            Type::Year => {
                col_def.year();
            }
            Type::Char(str_attr) => {
                match str_attr.length {
                    Some(length) => col_def.char_len(length),
                    _ => col_def.char(),
                };
                col_def = self.write_str_attr(col_def, str_attr);
            }
            Type::NChar(str_attr) => {
                match str_attr.length {
                    Some(length) => col_def.char_len(length),
                    _ => col_def.char(),
                };
                col_def.extra(format!("CHARACTER SET {}", CharSet::Utf8.to_string()));
            }
            Type::Varchar(str_attr) => {
                match str_attr.length {
                    Some(length) => col_def.string_len(length),
                    _ => col_def.string(),
                };
                col_def = self.write_str_attr(col_def, str_attr);
            }
            Type::NVarchar(str_attr) => {
                match str_attr.length {
                    Some(length) => col_def.string_len(length),
                    _ => col_def.string(),
                };
                col_def.extra(format!("CHARACTER SET {}", CharSet::Utf8.to_string()));
            }
            Type::Binary(str_attr) => {
                match str_attr.length {
                    Some(length) => col_def.binary_len(length),
                    None => col_def.blob(),
                };
                col_def = self.write_str_attr(col_def, str_attr);
            }
            Type::Varbinary(str_attr) => {
                match str_attr.length {
                    Some(length) => col_def.var_binary(length),
                    None => col_def.blob(),
                };
            }
            Type::Text(str_attr) => {
                col_def.text();
                col_def = self.write_str_attr(col_def, str_attr);
            }
            Type::TinyText(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::MediumText(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::LongText(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::Blob(blob_attr) => {
                match blob_attr.length {
                    Some(length) => col_def.binary_len(length),
                    None => col_def.binary_len(255),
                };
            }
            Type::TinyBlob => {
                col_def.custom(MySqlType::TinyBlob);
            }
            Type::MediumBlob => {
                col_def.custom(MySqlType::MediumBlob);
            }
            Type::LongBlob => {
                col_def.custom(MySqlType::LongBlob);
            }
            Type::Enum(enum_attr) => {
                let name = Alias::new(&self.name);
                let variants: Vec<DynIden> = enum_attr
                    .values
                    .iter()
                    .map(|variant| Alias::new(variant).into_iden())
                    .collect();
                col_def.enumeration(name, variants);
            }
            Type::Set(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::Geometry(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::Point(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::LineString(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::Polygon(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::MultiPoint(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::MultiLineString(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::MultiPolygon(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::GeometryCollection(_) => {
                // FIXME: Unresolved type mapping
                col_def.custom(self.col_type.clone());
            }
            Type::Json => {
                col_def.json();
            }
            Type::Unknown(_) => {
                col_def.custom(self.col_type.clone());
            }
        };
        col_def
    }

    pub fn write_num_attr(&self, mut col_def: ColumnDef, num_attr: &NumericAttr) -> ColumnDef {
        if matches!(
            &self.col_type,
            Type::Decimal(_) | Type::Float(_) | Type::Double(_)
        ) && num_attr.unsigned.is_some()
        {
            col_def.extra("UNSIGNED");
        }
        if num_attr.zero_fill.is_some() {
            col_def.extra("ZEROFILL");
        }
        col_def
    }

    pub fn write_str_attr(&self, mut col_def: ColumnDef, str_attr: &StringAttr) -> ColumnDef {
        if let Some(charset) = &str_attr.charset {
            col_def.extra(format!("CHARACTER SET {}", charset.to_string()));
        }
        if let Some(collation) = &str_attr.collation {
            col_def.extra(format!("COLLATE {}", collation.to_string()));
        }
        col_def
    }
}