7c42524ecd
[GB.DB] * NEW: Connection: The URL argument of the Connection constructor is now handled by the Gambas part. * NEW: Connection: It's now possible to specify the password in the connection URL.
592 lines
13 KiB
Text
592 lines
13 KiB
Text
' Gambas class file
|
|
|
|
Export
|
|
|
|
Inherits _Connection
|
|
|
|
Private Const TEMPLATE_MAGIC As String = "# Gambas Database Template File 3.0"
|
|
Private DEBUG_ME As Boolean
|
|
|
|
Property Read Url As String
|
|
Property Read SQL As SQLRequest
|
|
|
|
'' Return or set the connection options.
|
|
''
|
|
'' These options are specific to the database type.
|
|
|
|
Property Options As Collection
|
|
|
|
Private $hOptions As Collection
|
|
|
|
Public Sub _new(Optional DatabaseURL As String)
|
|
|
|
Dim iPos As Integer
|
|
Dim sIdent As String
|
|
|
|
If Not DatabaseURL Then Return
|
|
|
|
' <type>://<user>:<password>@<host>:<port>/<name>
|
|
|
|
iPos = InStr(DatabaseURL, "://")
|
|
If iPos = 0 Then Goto BAD_URL
|
|
|
|
Me.Type = Left(DatabaseURL, iPos - 1)
|
|
DatabaseURL = Mid$(DatabaseURL, iPos + 3)
|
|
|
|
iPos = RInStr(DatabaseURL, "/")
|
|
If iPos Then
|
|
Me.Name = FromUrl(Mid$(DatabaseURL, iPos + 1))
|
|
DatabaseURL = Left(DatabaseURL, iPos - 1)
|
|
Endif
|
|
|
|
iPos = InStr(DatabaseURL, "@")
|
|
If iPos Then
|
|
|
|
sIdent = Left(DatabaseURL, iPos - 1)
|
|
DatabaseURL = Mid$(DatabaseURL, iPos + 1)
|
|
|
|
iPos = InStr(sIdent, ":")
|
|
If iPos Then
|
|
Me.Password = FromUrl(Mid$(sIdent, iPos + 1))
|
|
Me.User = FromUrl(Left$(sIdent, iPos - 1))
|
|
Else
|
|
Me.User = FromUrl(sIdent)
|
|
Endif
|
|
|
|
Endif
|
|
|
|
iPos = InStr(DatabaseURL, ":")
|
|
If iPos Then
|
|
Me.Host = FromUrl(Left$(DatabaseURL, iPos - 1))
|
|
Me.Port = FromUrl(Mid$(DatabaseURL, iPos + 1))
|
|
Else
|
|
Me.Host = FromUrl(DatabaseURL)
|
|
Endif
|
|
|
|
Return
|
|
|
|
BAD_URL:
|
|
|
|
Error.Raise("Malformed URL")
|
|
|
|
End
|
|
|
|
|
|
Private Function GetTempTableName() As String
|
|
|
|
Dim iInd As Integer
|
|
Dim sTemp As String
|
|
|
|
For iInd = 0 To 99
|
|
sTemp = "_gb_temp"
|
|
If iInd Then sTemp = sTemp & "_" & iInd
|
|
If Not Me.Tables.Exist(sTemp) Then Break
|
|
Inc iInd
|
|
sTemp = ""
|
|
Next
|
|
|
|
If Not sTemp Then Error.Raise("Cannot find a free temporary table name")
|
|
|
|
Return sTemp
|
|
|
|
End
|
|
|
|
Private Sub Convert(vVal As Variant, iType As Integer) As Variant
|
|
|
|
Select iType
|
|
Case db.Float
|
|
vVal = Val(vVal)
|
|
If TypeOf(vVal) <= gb.Float Then Return vVal
|
|
Case db.Date
|
|
vVal = Val(vVal)
|
|
If TypeOf(vVal) = gb.Date Then Return vVal
|
|
End Select
|
|
|
|
Error.Raise("Type mismatch")
|
|
|
|
End
|
|
|
|
Private Function CopyTable(sTable As String) As String
|
|
|
|
Dim sTemp As String
|
|
Dim iInd As Integer
|
|
Dim sReq As String
|
|
Dim hSrc As Table
|
|
Dim hDst As Table
|
|
Dim rSrc As Result
|
|
Dim rDst As Result
|
|
Dim hField As Field
|
|
Dim sError As String
|
|
Dim iMax As Integer
|
|
|
|
sTemp = GetTempTableName()
|
|
|
|
Select Case Me.Type
|
|
|
|
Case "postgresql"
|
|
sReq = "SELECT * INTO TABLE " & sTemp & " FROM \"" & sTable & "\""
|
|
Me.Exec(sReq)
|
|
|
|
Case "mysql"
|
|
sReq = "CREATE TABLE " & sTemp & " SELECT * FROM `" & sTable & "`"
|
|
Me.Exec(sReq)
|
|
|
|
'CASE "sqlite"
|
|
' sReq = "INSERT INTO " & sTemp & " SELECT * FROM " & sTable
|
|
|
|
Default
|
|
|
|
hSrc = Me.Tables[sTable]
|
|
hDst = Me.Tables.Add(sTemp, hSrc.Type)
|
|
|
|
'IF NOT hSrc.PrimaryKey THEN Error.Raise("No primary key")
|
|
|
|
For Each hField In hSrc.Fields
|
|
With hField
|
|
hDst.Fields.Add(.Name, .Type, .Length, .Default)
|
|
End With
|
|
Next
|
|
|
|
iMax = hSrc.Fields.Count - 1
|
|
|
|
hDst.PrimaryKey = hSrc.PrimaryKey
|
|
|
|
hDst.Update
|
|
|
|
Me.Begin
|
|
|
|
rSrc = Me.Find(sTable)
|
|
rDst = Me.Create(sTemp)
|
|
|
|
For Each rSrc
|
|
For iInd = 0 To iMax
|
|
rDst[iInd] = rSrc[iInd]
|
|
Next
|
|
rDst.Update
|
|
Next
|
|
|
|
Me.Commit
|
|
|
|
End Select
|
|
|
|
Return sTemp
|
|
|
|
Catch
|
|
|
|
sError = Error.Text
|
|
|
|
Select Case Me.Type
|
|
|
|
Case "postgresql"
|
|
|
|
Case "mysql"
|
|
Try Me.Rollback
|
|
If Me.Tables.Exist(sTable) Then
|
|
Try Me.Tables.Remove(sTemp)
|
|
Else
|
|
If Me.Tables.Exist(sTemp) Then
|
|
sReq = "RENAME TABLE " & sTemp & " TO " & sTable
|
|
Try Me.Exec(sReq)
|
|
Else
|
|
Error.Raise("Severe Error: Table has been lost!!")
|
|
Endif
|
|
Endif
|
|
|
|
Default
|
|
Try Me.Rollback
|
|
If Me.Tables.Exist(sTable) Then
|
|
Try Me.Tables.Remove(sTemp)
|
|
Else
|
|
If Me.Tables.Exist(sTable) Then
|
|
Error.Raise("Severe Error: Table " & sTable & " has not been recreated. Data held in " & sTemp)
|
|
Endif
|
|
Endif
|
|
|
|
End Select
|
|
|
|
Error.Raise("Cannot copy table data: " & sError)
|
|
|
|
End
|
|
|
|
Private Sub ReadTemplate(hFile As File, Optional iLine As Integer) As Collection[]
|
|
|
|
Dim cResult As New Collection[]
|
|
|
|
Dim cTable As Collection
|
|
Dim cField As Collection
|
|
Dim cIndex As Collection
|
|
Dim sLine As String
|
|
Dim cCurrent As Collection
|
|
Dim sErr As String
|
|
Dim iPos As Integer
|
|
|
|
For Each sLine In hFile.Lines
|
|
|
|
Inc iLine
|
|
sLine = Trim(sLine)
|
|
If Not sLine Then Continue
|
|
If sLine Begins "#" Then Continue
|
|
|
|
If sLine = "}" Then
|
|
If cIndex Then
|
|
cTable["Indexes"].Add(cIndex)
|
|
cIndex = Null
|
|
cCurrent = cTable
|
|
Else If cField Then
|
|
cTable["Fields"].Add(cField)
|
|
cField = Null
|
|
cCurrent = cTable
|
|
Else If cTable Then
|
|
cResult.Add(cTable)
|
|
cTable = Null
|
|
cIndex = Null
|
|
cField = Null
|
|
cCurrent = Null
|
|
Else
|
|
sErr = "Unexpected '}'"
|
|
Goto SYNTAX_ERROR
|
|
Endif
|
|
Continue
|
|
Endif
|
|
|
|
If cIndex Then
|
|
Else If cField Then
|
|
Else If cTable Then
|
|
If sLine = "{ Index" Then
|
|
cIndex = New Collection
|
|
cCurrent = cIndex
|
|
Continue
|
|
Else If sLine = "{ Field" Then
|
|
cField = New Collection
|
|
cCurrent = cField
|
|
Continue
|
|
Endif
|
|
Else
|
|
If sLine <> "{ Table" Then
|
|
sErr = "`{ Table` expected"
|
|
Goto SYNTAX_ERROR
|
|
Endif
|
|
cTable = New Collection
|
|
cTable["Fields"] = New Collection[]
|
|
cTable["Indexes"] = New Collection[]
|
|
cCurrent = cTable
|
|
Continue
|
|
Endif
|
|
|
|
iPos = InStr(sLine, "=")
|
|
If iPos <= 1 Then Goto SYNTAX_ERROR
|
|
' FIXME: Calling Eval() is not very secure!
|
|
Try cCurrent[Trim(Left(sLine, iPos - 1))] = Eval(Trim(Mid$(sLine, iPos + 1)))
|
|
If Error Then Goto SYNTAX_ERROR
|
|
|
|
Next
|
|
|
|
If cCurrent Then Goto SYNTAX_ERROR
|
|
|
|
Return cResult
|
|
|
|
SYNTAX_ERROR:
|
|
|
|
If Not sErr Then sErr = "`" & sLine & "`"
|
|
Error.Raise("Syntax error in database template at line " & CStr(iLine) & ": " & sErr)
|
|
|
|
End
|
|
|
|
Private Sub IsSameTable(cTable As Collection, cTable2 As Collection) As Boolean
|
|
|
|
Dim hFile As File
|
|
Dim sTable As String
|
|
Dim sTable2 As String
|
|
|
|
hFile = Open String For Write
|
|
|
|
Write #hFile, cTable As Collection
|
|
sTable = Close #hFile
|
|
|
|
hFile = Open String For Write
|
|
Write #hFile, cTable2 As Collection
|
|
sTable2 = Close #hFile
|
|
|
|
Return sTable = sTable2
|
|
|
|
End
|
|
|
|
Public Sub ApplyTemplate(Template As String)
|
|
|
|
Dim hFile As File
|
|
Dim sLine As String
|
|
Dim cTable As Collection
|
|
Dim cField As Collection
|
|
Dim cIndex As Collection
|
|
Dim sTable As String
|
|
Dim hTable As Table
|
|
Dim iLength As Integer
|
|
Dim sColl As String
|
|
Dim aKey As String[]
|
|
Dim sCopyTable As String
|
|
Dim sName As String
|
|
Dim rTemp As Result
|
|
Dim rTable As Result
|
|
Dim cTableOrg As Collection
|
|
Dim aTables As Collection[]
|
|
|
|
DEBUG_ME = Env["GB_DB_DEBUG_TEMPLATE"] = "1"
|
|
|
|
If DEBUG_ME Then Error "gb.db: applying template to: "; Me.Url; "..."
|
|
|
|
hFile = Open String Template
|
|
|
|
Line Input #hFile, sLine
|
|
If sLine <> "# Gambas Database Template File 3.0" Then Error.Raise("Bad database template format")
|
|
|
|
aTables = ReadTemplate(hFile, 1)
|
|
|
|
Close #hFile
|
|
|
|
For Each cTable In aTables
|
|
|
|
sTable = cTable["Name"]
|
|
If Not sTable Then Continue
|
|
|
|
If Me.Tables.Exist(sTable) Then
|
|
|
|
'If Not UpdateExistingTables Then Continue
|
|
|
|
hFile = Open String For Write
|
|
WriteTable(hFile, sTable)
|
|
Seek #hFile, 0
|
|
cTableOrg = ReadTemplate(hFile)[0]
|
|
Close #hFile
|
|
|
|
If IsSameTable(cTable, cTableOrg) Then
|
|
Continue
|
|
Endif
|
|
|
|
If DEBUG_ME Then Error "gb.db: updating table "; sTable
|
|
|
|
sCopyTable = CopyTable(sTable)
|
|
Try Me.Tables.Remove(sTable)
|
|
|
|
Else
|
|
|
|
If DEBUG_ME Then Error "gb.db: creating table "; sTable
|
|
sCopyTable = ""
|
|
|
|
Endif
|
|
|
|
'Print "create table: "; sTable;; cTable["Type"];; cTable["Fields"].Count
|
|
hTable = Me.Tables.Add(sTable, cTable["Type"])
|
|
For Each cField In cTable["Fields"]
|
|
iLength = 0
|
|
Try iLength = cField["Length"]
|
|
'Print "create field: "; cField["Name"];; cField["Type"];; iLength;; cField["Default"];; cField["Collation"]
|
|
sColl = cField["Collation"]
|
|
If sColl = "default" Then sColl = ""
|
|
hTable.Fields.Add(cField["Name"], cField["Type"], iLength, cField["Default"], sColl)
|
|
Next
|
|
|
|
'Print "primary key: "; cTable["PrimaryKey"].Join(",")
|
|
aKey = cTable["PrimaryKey"]
|
|
If aKey.Count = 1 And If aKey[0] = "" Then aKey.Clear
|
|
If aKey.Count Then hTable.PrimaryKey = cTable["PrimaryKey"]
|
|
hTable.Update
|
|
|
|
For Each cIndex In cTable["Indexes"]
|
|
'Print "create index: "; cIndex["Name"];; cIndex["Fields"].Join(",");; cIndex["Unique"]
|
|
hTable.Indexes.Add(cIndex["Name"], cIndex["Fields"], cIndex["Unique"])
|
|
Next
|
|
|
|
If sCopyTable Then
|
|
|
|
rTemp = Me.Find(sCopyTable)
|
|
If rTemp.Count Then
|
|
|
|
If DEBUG_ME Then Error "gb.db: copying "; rTemp.Count; " records in the new table..."
|
|
|
|
Me.Begin
|
|
|
|
rTable = Me.Create(sTable)
|
|
|
|
For Each rTemp
|
|
|
|
For Each cField In cTable["Fields"]
|
|
sName = cField["Name"]
|
|
Try rTable[sName] = rTemp[sName]
|
|
If Error Then
|
|
Try rTable[sName] = Convert(rTemp[sName], cField["Type"])
|
|
Endif
|
|
Next
|
|
|
|
Try rTable.Update
|
|
|
|
Next
|
|
|
|
Try Me.Commit
|
|
|
|
Endif
|
|
|
|
Endif
|
|
|
|
Next
|
|
|
|
End
|
|
|
|
Private Sub WriteTable(hFile As Stream, sTable As String) As String
|
|
|
|
Dim hTable As Table
|
|
Dim hField As Field
|
|
Dim hIndex As Index
|
|
Dim aKey As String[]
|
|
|
|
hTable = Me.Tables[sTable]
|
|
|
|
Print #hFile, "{ Table"
|
|
Print #hFile, " Name="; Quote(hTable.Name)
|
|
If hTable.Type Then Print #hFile, " Type="; Quote(hTable.Type)
|
|
aKey = hTable.PrimaryKey
|
|
If aKey.Count Then Print #hFile, " PrimaryKey=[\""; aKey.Join("\",\""); "\"]"
|
|
|
|
For Each hField In hTable.Fields
|
|
|
|
Print #hFile, " { Field"
|
|
Print #hFile, " Name="; Quote(hField.Name)
|
|
Print #hFile, " Type=";
|
|
|
|
Select hField.Type
|
|
Case db.Blob
|
|
Print #hFile, "db.Blob"
|
|
Case db.Boolean
|
|
Print #hFile, "db.Boolean"
|
|
Case db.Date
|
|
Print #hFile, "db.Date"
|
|
Case db.Float
|
|
Print #hFile, "db.Float"
|
|
Case db.Integer
|
|
Print #hFile, "db.Integer"
|
|
Case db.Long
|
|
Print #hFile, "db.Long"
|
|
Case db.Serial
|
|
Print #hFile, "db.Serial"
|
|
Case db.String
|
|
Print #hFile, "db.String"
|
|
If hField.Length Then Print #hFile, " Length="; hField.Length
|
|
Case Else
|
|
Error.Raise("Unknown database field type")
|
|
End Select
|
|
|
|
If Not IsNull(hField.Default) Then
|
|
Print #hFile, " Default=";
|
|
If hField.Type = db.String Then
|
|
Print #hFile, Quote(hField.Default)
|
|
Else If hField.Type = db.Boolean Then
|
|
Print #hFile, If(hField.Default, "True", "False")
|
|
Else If hField.Type = db.Date Then
|
|
Print #hFile, "CDate(\""; CStr(hField.Default); "\")"
|
|
Else
|
|
Print #hFile, CStr(hField.Default)
|
|
Endif
|
|
Endif
|
|
|
|
If hField.Collation Then
|
|
Print #hFile, " Collation="; Quote(hField.Collation)
|
|
Endif
|
|
|
|
Print #hFile, " }"
|
|
|
|
Next
|
|
|
|
For Each hIndex In hTable.Indexes
|
|
If hIndex.Primary Then Continue
|
|
Print #hFile, " { Index "
|
|
Print #hFile, " Name="; Quote(hIndex.Name)
|
|
Print #hFile, " Unique="; If(hIndex.Unique, "True", "False")
|
|
Print #hFile, " Fields=[\""; hIndex.Fields.Join("\",\""); "\"]"
|
|
Print #hFile, " }"
|
|
Next
|
|
|
|
Print #hFile, "}"
|
|
|
|
End
|
|
|
|
Public Sub GetTemplate() As String
|
|
|
|
Dim hFile As File
|
|
Dim aTable As String[]
|
|
Dim hTable As Table
|
|
Dim sTable As String
|
|
Dim sTemplate As String
|
|
|
|
hFile = Open String For Write
|
|
|
|
aTable = New String[]
|
|
For Each hTable In Me.Tables
|
|
If hTable.System Then Continue
|
|
'If sTable And If hTable.Name <> sTable Then Continue
|
|
aTable.Add(hTable.Name)
|
|
Next
|
|
|
|
Print #hFile, TEMPLATE_MAGIC
|
|
|
|
For Each sTable In aTable
|
|
|
|
WriteTable(hFile, sTable)
|
|
|
|
Next
|
|
|
|
sTemplate = Close #hFile
|
|
Return sTemplate
|
|
|
|
End
|
|
|
|
Public Sub Copy() As Connection
|
|
|
|
Dim hConn As Connection
|
|
|
|
hConn = New Connection
|
|
hConn.Host = Me.Host
|
|
hConn.IgnoreCharset = Me.IgnoreCharset
|
|
hConn.Name = Me.Name
|
|
hConn.Password = Me.Password
|
|
hConn.Port = Me.Port
|
|
hConn.Timeout = Me.Timeout
|
|
hConn.Type = Me.Type
|
|
hConn.User = Me.User
|
|
|
|
Return hConn
|
|
|
|
End
|
|
|
|
Private Function SQL_Read() As SQLRequest
|
|
|
|
Return New SQLRequest(Me)
|
|
|
|
End
|
|
|
|
Private Function Url_Read() As String
|
|
|
|
Dim sUrl As String
|
|
|
|
sUrl = Me.Type & "://"
|
|
If Me.User Then sUrl &= Me.User & "@"
|
|
sUrl &= Me.Host &/ Me.Name
|
|
Return sUrl
|
|
|
|
End
|
|
|
|
|
|
Private Function Options_Read() As Collection
|
|
|
|
If Not $hOptions Then $hOptions = New Collection
|
|
Return $hOptions
|
|
|
|
End
|
|
|
|
Private Sub Options_Write(Value As Collection)
|
|
|
|
If Not Value Then
|
|
$hOptions = Null
|
|
Else
|
|
$hOptions = Value.Copy()
|
|
Endif
|
|
|
|
End
|