यदि आप एक समय में एक पंक्ति निष्पादित करते हैं तो ये सभी उत्तर ठीक हैं। हालांकि, मूल प्रश्न एक एसक्यूएल स्क्रिप्ट को इनपुट करने के लिए था जिसे एक एकल डीबी निष्पादित द्वारा निष्पादित किया जाएगा और सभी समाधान (जैसे यह देखने के लिए जांचना कि स्तंभ समय से पहले है) को निष्पादन कार्यक्रम की आवश्यकता होगी या तो तालिकाओं का ज्ञान होगा इस जानकारी को निर्धारित करने के लिए कॉलम को बदल दिया / जोड़ा जा रहा है या इनपुट स्क्रिप्ट की पूर्व-प्रसंस्करण और पार्सिंग कर रहे हैं। आमतौर पर आप इसे रीयलटाइम या अक्सर चलाने नहीं जा रहे हैं। तो एक अपवाद को पकड़ने का विचार स्वीकार्य है और फिर आगे बढ़ रहा है। उसमें समस्या है ... कैसे आगे बढ़ना है। सौभाग्य से त्रुटि संदेश हमें यह करने के लिए आवश्यक सभी जानकारी देता है। यदि यह परिवर्तन तालिका कॉल पर अपवाद करता है, तो हम sql को निष्पादित करने के लिए विचार कर सकते हैं, हम sql में परिवर्तन तालिका लाइन को ढूंढ सकते हैं और शेष पंक्तियों को वापस कर सकते हैं और तब तक निष्पादित कर सकते हैं जब तक कि या तो अधिक मेल खाती हुई परिवर्तन तालिका रेखाएं नहीं मिल सकती हैं। कुछ उदाहरण कोड का उपयोग करता है जहां हमारे पास एक सरणी में sql स्क्रिप्ट है। हम प्रत्येक स्क्रिप्ट को निष्पादित करने वाले सरणी को पुनरावृत्त करते हैं। हम इसे दो बार कॉल करते हैं कि परिवर्तन तालिका कमांड को विफल करने के लिए लेकिन कार्यक्रम सफल होता है क्योंकि हम sql से परिवर्तन तालिका कमांड को हटा देते हैं और अद्यतन कोड को फिर से निष्पादित करते हैं।
exec /opt/usr8.6.3/bin/tclsh8.6 "$0" ${1+"$@"}
foreach pkg {sqlite3 } {
if { [ catch {package require {*}$pkg } err ] != 0 } {
puts stderr "Unable to find package $pkg\n$err\n ... adjust your auto_path!";
}
}
array set sqlArray {
1 {
CREATE TABLE IF NOT EXISTS Notes (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name text,
note text,
createdDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) ) ,
updatedDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) )
);
CREATE TABLE IF NOT EXISTS Version (
id INTEGER PRIMARY KEY AUTOINCREMENT,
version text,
createdDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) ) ,
updatedDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) )
);
INSERT INTO Version(version) values('1.0');
}
2 {
CREATE TABLE IF NOT EXISTS Tags (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name text,
tag text,
createdDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) ) ,
updatedDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) )
);
ALTER TABLE Notes ADD COLUMN dump text;
INSERT INTO Version(version) values('2.0');
}
3 {
ALTER TABLE Version ADD COLUMN sql text;
INSERT INTO Version(version) values('3.0');
}
}
sqlite3 db :memory:
proc createSchema { sqlArray } {
upvar $sqlArray sql
foreach version [lsort -integer [array names sql ] ] {
set cmd $sql($version)
set ok 0
while { !$ok && [string length $cmd ] } {
try {
db eval $cmd
set ok 1 ;
} on error { err backtrace } {
if { [regexp {duplicate column name: ([a-zA-Z0-9])} [string trim $err ] match columnname ] } {
puts "Error: $err ... trying again"
set cmd [removeAlterTable $cmd $columnname ]
} else {
throw DBERROR "$err\n$backtrace"
}
}
}
}
}
proc removeAlterTable { sqltext columnname } {
set mode skip
set result [list]
foreach line [split $sqltext \n ] {
if { [string first "alter table" [string tolower [string trim $line] ] ] >= 0 } {
if { [string first $columnname $line ] } {
set mode add
continue;
}
}
if { $mode eq "add" } {
lappend result $line
}
}
if { $mode eq "skip" } {
puts stderr "Unable to find matching alter table line"
return ""
} elseif { [llength $result ] } {
return [ join $result \n ]
} else {
return ""
}
}
proc printSchema { } {
db eval { select * from sqlite_master } x {
puts "Table: $x(tbl_name)"
puts "$x(sql)"
puts "
}
}
createSchema sqlArray
printSchema
createSchema sqlArray
printSchema
अपेक्षित उत्पादन
Table: Notes
CREATE TABLE Notes (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name text,
note text,
createdDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) ) ,
updatedDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) )
, dump text)
Table: sqlite_sequence
CREATE TABLE sqlite_sequence(name,seq)
Table: Version
CREATE TABLE Version (
id INTEGER PRIMARY KEY AUTOINCREMENT,
version text,
createdDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) ) ,
updatedDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) )
, sql text)
Table: Tags
CREATE TABLE Tags (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name text,
tag text,
createdDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) ) ,
updatedDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) )
)
Error: duplicate column name: dump ... trying again
Error: duplicate column name: sql ... trying again
Table: Notes
CREATE TABLE Notes (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name text,
note text,
createdDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) ) ,
updatedDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) )
, dump text)
Table: sqlite_sequence
CREATE TABLE sqlite_sequence(name,seq)
Table: Version
CREATE TABLE Version (
id INTEGER PRIMARY KEY AUTOINCREMENT,
version text,
createdDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) ) ,
updatedDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) )
, sql text)
Table: Tags
CREATE TABLE Tags (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name text,
tag text,
createdDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) ) ,
updatedDate integer(4) DEFAULT ( cast( strftime('%s', 'now') as int ) )
)
user_version
? मैं शून्य मानती हूं, लेकिन उस दस्तावेज को देखकर अच्छा लगेगा।