2025-01-13 08:00:21 +00:00
import fs from "fs" ;
import varDatabaseDbHandler from "./varDatabaseDbHandler" ;
2024-12-06 13:24:26 +00:00
const defaultFieldsRegexp =
/^id$|^uuid$|^date_created$|^date_created_code$|^date_created_timestamp$|^date_updated$|^date_updated_code$|^date_updated_timestamp$/ ;
2025-01-13 08:00:21 +00:00
import generateColumnDescription from "./generateColumnDescription" ;
import dbHandler from "./dbHandler" ;
type Param = {
dbFullName : string ;
tableName : string ;
tableSchema : import ( "../../types" ) . DSQL_TableSchemaType ;
tableNameFull? : string ;
tableInfoArray : import ( "../../types" ) . DSQL_FieldSchemaType [ ] ;
userId? : number | string | null ;
dbSchema : import ( "../../types" ) . DSQL_DatabaseSchemaType [ ] ;
tableIndexes? : import ( "../../types" ) . DSQL_IndexSchemaType [ ] ;
clone? : boolean ;
tableIndex? : number ;
childDb? : boolean ;
recordedDbEntry? : any ;
} ;
2024-12-06 13:24:26 +00:00
/ * *
2025-01-13 08:00:21 +00:00
* # Update table function
2024-12-06 13:24:26 +00:00
* /
2025-01-13 08:00:21 +00:00
export default async function updateTable ( {
2024-12-06 13:24:26 +00:00
dbFullName ,
tableName ,
tableInfoArray ,
userId ,
dbSchema ,
tableIndexes ,
tableSchema ,
clone ,
childDb ,
tableIndex ,
tableNameFull ,
recordedDbEntry ,
2025-01-13 08:00:21 +00:00
} : Param ) {
2024-12-06 13:24:26 +00:00
/ * *
* Initialize
* === === === === === === === === === === === === === ===
* @description Initial setup
* /
2024-12-08 08:57:48 +00:00
/** @type {any[]} */
2025-01-13 08:00:21 +00:00
let errorLogs : any [ ] = [ ] ;
2024-12-08 08:57:48 +00:00
2024-12-06 13:24:26 +00:00
/ * *
* @description Initialize table info array . This value will be
* changing depending on if a field is renamed or not .
* /
let upToDateTableFieldsArray = tableInfoArray ;
/ * *
* Handle Table updates
*
* @description Try to undate table , catch error if anything goes wrong
* /
try {
/ * *
* @type { string [ ] }
* @description Table update query string array
* /
2025-01-13 08:00:21 +00:00
const updateTableQueryArray : string [ ] = [ ] ;
2024-12-06 13:24:26 +00:00
/ * *
* @type { string [ ] }
* @description Constriants query string array
* /
2025-01-13 08:00:21 +00:00
const constraintsQueryArray : string [ ] = [ ] ;
2024-12-06 13:24:26 +00:00
/ * *
* @description Push the query initial value
* /
2025-01-13 08:00:21 +00:00
updateTableQueryArray . push (
` ALTER TABLE \` ${ dbFullName } \` . \` ${ tableName } \` `
) ;
2024-12-06 13:24:26 +00:00
if ( childDb ) {
try {
if ( ! recordedDbEntry ) {
throw new Error ( "Recorded Db entry not found!" ) ;
}
const existingTable = await varDatabaseDbHandler ( {
2025-01-13 08:00:21 +00:00
queryString : ` SELECT * FROM datasquirel.user_database_tables WHERE db_id = ? AND table_slug = ? ` ,
2024-12-06 13:24:26 +00:00
queryValuesArray : [ recordedDbEntry . id , tableName ] ,
} ) ;
/** @type {import("../../types").MYSQL_user_database_tables_table_def} */
2025-01-13 08:00:21 +00:00
const table : import ( "../../types" ) . MYSQL_user_database_tables_table_def =
existingTable ? . [ 0 ] ;
2024-12-06 13:24:26 +00:00
if ( ! table ? . id ) {
const newTableEntry = await dbHandler ( {
2025-01-13 08:00:21 +00:00
query : ` INSERT INTO datasquirel.user_database_tables SET ? ` ,
2024-12-06 13:24:26 +00:00
values : {
user_id : recordedDbEntry.user_id ,
db_id : recordedDbEntry.id ,
db_slug : recordedDbEntry.db_slug ,
table_name : tableNameFull ,
table_slug : tableName ,
child_table : tableSchema?.childTable ? "1" : null ,
child_table_parent_database :
tableSchema ? . childTableDbFullName || null ,
child_table_parent_table :
tableSchema . childTableName || null ,
date_created : Date ( ) ,
date_created_code : Date.now ( ) ,
date_updated : Date ( ) ,
date_updated_code : Date.now ( ) ,
} ,
} ) ;
}
} catch ( error ) { }
}
/ * *
* @type { import ( "../../types" ) . DSQL_MYSQL_SHOW_INDEXES_Type [ ] }
* @description All indexes from MYSQL db
* / / / @ts - ignore
2025-01-13 08:00:21 +00:00
const allExistingIndexes : import ( "../../types" ) . DSQL_MYSQL_SHOW_INDEXES_Type [ ] =
await varDatabaseDbHandler ( {
queryString : ` SHOW INDEXES FROM \` ${ dbFullName } \` . \` ${ tableName } \` ` ,
} ) ;
2024-12-06 13:24:26 +00:00
/ * *
* @type { import ( "../../types" ) . DSQL_MYSQL_SHOW_COLUMNS_Type [ ] }
* @description All columns from MYSQL db
* / / / @ts - ignore
2025-01-13 08:00:21 +00:00
const allExistingColumns : import ( "../../types" ) . DSQL_MYSQL_SHOW_COLUMNS_Type [ ] =
await varDatabaseDbHandler ( {
queryString : ` SHOW COLUMNS FROM \` ${ dbFullName } \` . \` ${ tableName } \` ` ,
} ) ;
2024-12-06 13:24:26 +00:00
////////////////////////////////////////
/ * *
* @type { string [ ] }
* @description Updated column names Array
* /
2025-01-13 08:00:21 +00:00
const updatedColumnsArray : string [ ] = [ ] ;
2024-12-06 13:24:26 +00:00
/ * *
* @description Iterate through every existing column
* /
for ( let e = 0 ; e < allExistingColumns . length ; e ++ ) {
const { Field } = allExistingColumns [ e ] ;
if ( Field . match ( defaultFieldsRegexp ) ) continue ;
/ * *
* @description This finds out whether the fieldName corresponds with the MSQL Field name
* if the fildName doesn ' t match any MYSQL Field name , the field is deleted .
* /
let existingEntry = upToDateTableFieldsArray . filter (
( column ) = >
column . fieldName === Field || column . originName === Field
) ;
if ( existingEntry && existingEntry [ 0 ] ) {
/ * *
* @description Check if Field name has been updated
* /
if (
existingEntry [ 0 ] . updatedField &&
existingEntry [ 0 ] . fieldName
) {
updatedColumnsArray . push ( existingEntry [ 0 ] . fieldName ) ;
const renameColumn = await varDatabaseDbHandler ( {
2025-01-13 08:00:21 +00:00
queryString : ` ALTER TABLE \` ${ dbFullName } \` . \` ${ tableName } \` RENAME COLUMN \` ${ existingEntry [ 0 ] . originName } \` TO \` ${ existingEntry [ 0 ] . fieldName } \` ` ,
2024-12-06 13:24:26 +00:00
} ) ;
console . log (
` Column Renamed from " ${ existingEntry [ 0 ] . originName } " to " ${ existingEntry [ 0 ] . fieldName } " `
) ;
/ * *
* Update Db Schema
* === === === === === === === === === === === === === === === === ===
* @description Update Db Schema after renaming column
* /
try {
const userSchemaData = dbSchema ;
const targetDbIndex = userSchemaData . findIndex (
( db ) = > db . dbFullName === dbFullName
) ;
const targetTableIndex = userSchemaData [
targetDbIndex
] . tables . findIndex (
( table ) = > table . tableName === tableName
) ;
const targetFieldIndex = userSchemaData [
targetDbIndex
] . tables [ targetTableIndex ] . fields . findIndex (
( field ) = >
field . fieldName === existingEntry [ 0 ] . fieldName
) ;
delete userSchemaData [ targetDbIndex ] . tables [
targetTableIndex
] . fields [ targetFieldIndex ] [ "originName" ] ;
delete userSchemaData [ targetDbIndex ] . tables [
targetTableIndex
] . fields [ targetFieldIndex ] [ "updatedField" ] ;
/ * *
* @description Set New Table Fields Array
* /
upToDateTableFieldsArray =
userSchemaData [ targetDbIndex ] . tables [
targetTableIndex
] . fields ;
fs . writeFileSync (
` ${ String (
process . env . DSQL_USER_DB_SCHEMA_PATH
) } / user - $ { userId } / main . json ` ,
JSON . stringify ( userSchemaData ) ,
"utf8"
) ;
2025-01-13 08:00:21 +00:00
} catch ( /** @type {any} */ error : any ) {
2024-12-06 13:24:26 +00:00
console . log ( "Update table error =>" , error . message ) ;
}
////////////////////////////////////////
}
////////////////////////////////////////
continue ;
////////////////////////////////////////
} else {
await varDatabaseDbHandler ( {
2025-01-13 08:00:21 +00:00
queryString : ` ALTER TABLE \` ${ dbFullName } \` . \` ${ tableName } \` DROP COLUMN \` ${ Field } \` ` ,
2024-12-06 13:24:26 +00:00
} ) ;
}
}
/ * *
* Handle MYSQL Table Indexes
* === === === === === === === === === === === === === === === === ===
* @description Iterate through each table index ( if available )
* and perform operations
* /
for ( let f = 0 ; f < allExistingIndexes . length ; f ++ ) {
const { Key_name , Index_comment } = allExistingIndexes [ f ] ;
/ * *
* @description Check if this index was specifically created
* by datasquirel
* /
if ( Index_comment ? . match ( /schema_index/ ) ) {
try {
const existingKeyInSchema = tableIndexes ? . filter (
( indexObject ) = > indexObject . alias === Key_name
) ;
if ( ! existingKeyInSchema ? . [ 0 ] )
throw new Error (
` This Index( ${ Key_name } ) Has been Deleted! `
) ;
} catch ( error ) {
/ * *
* @description Drop Index : This happens when the MYSQL index is not
* present in the datasquirel DB schema
* /
await varDatabaseDbHandler ( {
2025-01-13 08:00:21 +00:00
queryString : ` ALTER TABLE \` ${ dbFullName } \` . \` ${ tableName } \` DROP INDEX \` ${ Key_name } \` ` ,
2024-12-06 13:24:26 +00:00
} ) ;
}
}
}
/ * *
* Handle DATASQUIREL Table Indexes
* === === === === === === === === === === === === === === === === ===
* @description Iterate through each datasquirel schema
* table index ( if available ) , and perform operations
* /
if ( tableIndexes && tableIndexes [ 0 ] ) {
for ( let g = 0 ; g < tableIndexes . length ; g ++ ) {
const { indexType , indexName , indexTableFields , alias } =
tableIndexes [ g ] ;
if ( ! alias ? . match ( /./ ) ) continue ;
/ * *
* @description Check for existing Index in MYSQL db
* /
try {
const existingKeyInDb = allExistingIndexes . filter (
( indexObject ) = > indexObject . Key_name === alias
) ;
if ( ! existingKeyInDb [ 0 ] )
throw new Error ( "This Index Does not Exist" ) ;
} catch ( error ) {
/ * *
* @description Create new index if determined that it
* doesn ' t exist in MYSQL db
* /
await varDatabaseDbHandler ( {
queryString : ` CREATE ${
indexType ? . match ( /fullText/i ) ? " FULLTEXT" : ""
2025-01-13 08:00:21 +00:00
} INDEX \ ` ${ alias } \` ON \` ${ dbFullName } \` . \` ${ tableName } \` ( ${ indexTableFields
2024-12-06 13:24:26 +00:00
? . map ( ( nm ) = > nm . value )
. map ( ( nm ) = > ` \` ${ nm } \` ` )
. join ( "," ) } ) COMMENT 'schema_index' ` ,
} ) ;
}
}
}
/ * *
* Handle MYSQL Foreign Keys
* === === === === === === === === === === === === === === === === ===
* @description Iterate through each datasquirel schema
* table index ( if available ) , and perform operations
* /
/ * *
* @description All MSQL Foreign Keys
* @type { import ( "../../types" ) . DSQL_MYSQL_FOREIGN_KEYS_Type [ ] | null }
2025-01-13 08:00:21 +00:00
* /
const allForeignKeys :
| import ( "../../types" ) . DSQL_MYSQL_FOREIGN_KEYS_Type [ ]
| null = await varDatabaseDbHandler ( {
2024-12-06 13:24:26 +00:00
queryString : ` SELECT CONSTRAINT_NAME FROM INFORMATION_SCHEMA.TABLE_CONSTRAINTS WHERE CONSTRAINT_SCHEMA = ' ${ dbFullName } ' AND TABLE_NAME=' ${ tableName } ' AND CONSTRAINT_TYPE='FOREIGN KEY' ` ,
} ) ;
if ( allForeignKeys ) {
for ( let c = 0 ; c < allForeignKeys . length ; c ++ ) {
const { CONSTRAINT_NAME } = allForeignKeys [ c ] ;
/ * *
* @description Skip if Key is the PRIMARY Key
* /
if ( CONSTRAINT_NAME . match ( /PRIMARY/ ) ) continue ;
/ * *
* @description Drop all foreign Keys to avoid MYSQL errors when adding / updating
* Foreign keys
* /
const dropForeignKey = await varDatabaseDbHandler ( {
2025-01-13 08:00:21 +00:00
queryString : ` ALTER TABLE \` ${ dbFullName } \` . \` ${ tableName } \` DROP FOREIGN KEY \` ${ CONSTRAINT_NAME } \` ` ,
2024-12-06 13:24:26 +00:00
} ) ;
}
}
/ * *
* Handle DATASQUIREL schema fields for current table
* === === === === === === === === === === === === === === === === ===
* @description Iterate through each field object and
* perform operations
* /
for ( let i = 0 ; i < upToDateTableFieldsArray . length ; i ++ ) {
const column = upToDateTableFieldsArray [ i ] ;
const prevColumn = upToDateTableFieldsArray [ i - 1 ] ;
const nextColumn = upToDateTableFieldsArray [ i + 1 ] ;
const {
fieldName ,
dataType ,
nullValue ,
primaryKey ,
autoIncrement ,
defaultValue ,
defaultValueLiteral ,
foreignKey ,
updatedField ,
} = column ;
////////////////////////////////////////
/ * *
* @description Skip default fields
* /
if ( fieldName ? . match ( /^id$|^date_/ ) ) continue ;
/ * *
* @description Skip columns that have been updated recently
* /
// if (updatedColumnsArray.includes(fieldName)) continue;
////////////////////////////////////////
let updateText = "" ;
////////////////////////////////////////
/** @type {any} */
2025-01-13 08:00:21 +00:00
let existingColumnIndex : any ;
2024-12-06 13:24:26 +00:00
/ * *
* @description Existing MYSQL field object
* /
let existingColumn =
allExistingColumns && allExistingColumns [ 0 ]
? allExistingColumns . filter ( ( _column , _index ) = > {
if ( _column . Field === fieldName ) {
existingColumnIndex = _index ;
return true ;
}
} )
: null ;
/ * *
* @description Construct SQL text snippet for this field
* /
let { fieldEntryText } = generateColumnDescription ( {
columnData : column ,
} ) ;
/ * *
* @description Modify Column ( Field ) if it already exists
* in MYSQL database
* /
if ( existingColumn && existingColumn [ 0 ] ? . Field ) {
const { Field , Type , Null , Key , Default , Extra } =
existingColumn [ 0 ] ;
let isColumnReordered = i < existingColumnIndex ;
if (
Field === fieldName &&
! isColumnReordered &&
dataType ? . toUpperCase ( ) === Type . toUpperCase ( )
) {
updateText += ` MODIFY COLUMN ${ fieldEntryText } ` ;
// continue;
} else {
if ( userId ) {
updateText += ` MODIFY COLUMN ${ fieldEntryText } ${
isColumnReordered
? prevColumn ? . fieldName
? " AFTER `" + prevColumn . fieldName + "`"
: nextColumn ? . fieldName
? " BEFORE `" + nextColumn . fieldName + "`"
: ""
: ""
} ` ;
} else {
updateText += ` MODIFY COLUMN ${ fieldEntryText } ` ;
}
}
} else if ( prevColumn && prevColumn . fieldName ) {
/ * *
* @description Add new Column AFTER previous column , if
* previous column exists
* /
updateText += ` ADD COLUMN ${ fieldEntryText } AFTER \` ${ prevColumn . fieldName } \` ` ;
} else if ( nextColumn && nextColumn . fieldName ) {
/ * *
* @description Add new Column BEFORE next column , if
* next column exists
* /
updateText += ` ADD COLUMN ${ fieldEntryText } BEFORE \` ${ nextColumn . fieldName } \` ` ;
} else {
/ * *
* @description Append new column to the end of existing columns
* /
updateText += ` ADD COLUMN ${ fieldEntryText } ` ;
}
////////////////////////////////////////
/ * *
* @description Pust SQL code snippet to updateTableQueryArray Array
* Add a comma ( , ) to separate from the next snippet
* /
updateTableQueryArray . push ( updateText + "," ) ;
/ * *
* @description Handle foreing keys if available , and if there is no
* "clone" boolean = true
* /
if ( ! clone && foreignKey ) {
const {
destinationTableName ,
destinationTableColumnName ,
cascadeDelete ,
cascadeUpdate ,
foreignKeyName ,
} = foreignKey ;
2024-12-08 08:57:48 +00:00
const foreinKeyText = ` ADD CONSTRAINT \` ${ foreignKeyName } \` FOREIGN KEY ( \` ${ fieldName } \` ) REFERENCES \` ${ destinationTableName } \` ( \` ${ destinationTableColumnName } \` ) ${
2024-12-06 13:24:26 +00:00
cascadeDelete ? " ON DELETE CASCADE" : ""
} $ { cascadeUpdate ? " ON UPDATE CASCADE" : "" } ` ;
// const foreinKeyText = `ADD CONSTRAINT \`${foreignKeyName}\` FOREIGN KEY (${fieldName}) REFERENCES ${destinationTableName}(${destinationTableColumnName})${cascadeDelete ? " ON DELETE CASCADE" : ""}${cascadeUpdate ? " ON UPDATE CASCADE" : ""}` + ",";
2025-01-13 08:00:21 +00:00
const finalQueryString = ` ALTER TABLE \` ${ dbFullName } \` . \` ${ tableName } \` ${ foreinKeyText } ` ;
2024-12-06 13:24:26 +00:00
const addForeignKey = await varDatabaseDbHandler ( {
queryString : finalQueryString ,
} ) ;
2024-12-08 08:57:48 +00:00
if ( ! addForeignKey ? . serverStatus ) {
errorLogs . push ( addForeignKey ) ;
}
2024-12-06 13:24:26 +00:00
}
////////////////////////////////////////
}
/ * *
* @description Construct final SQL query by combning all SQL snippets in
* updateTableQueryArray Arry , and trimming the final comma ( , )
* /
const updateTableQuery = updateTableQueryArray
. join ( " " )
. replace ( /,$/ , "" ) ;
////////////////////////////////////////
/ * *
* @description Check if SQL snippets array has more than 1 entries
* This is because 1 entry means "ALTER TABLE table_name" only , without any
* Alter directives like "ADD COLUMN" or "MODIFY COLUMN"
* /
if ( updateTableQueryArray . length > 1 ) {
const updateTable = await varDatabaseDbHandler ( {
queryString : updateTableQuery ,
} ) ;
return updateTable ;
} else {
/ * *
* @description If only 1 SQL snippet is left in updateTableQueryArray , this
* means that no updates have been made to the table
* /
return "No Changes Made to Table" ;
}
2025-01-13 08:00:21 +00:00
} catch ( /** @type {any} */ error : any ) {
2024-12-06 13:24:26 +00:00
console . log ( 'Error in "updateTable" shell function =>' , error . message ) ;
return "Error in Updating Table" ;
}
2025-01-13 08:00:21 +00:00
}