199 lines
5.6 KiB
Lua
199 lines
5.6 KiB
Lua
-- ------------------------------------------------------------------------------ --
|
|
-- TradeSkillMaster --
|
|
-- https://tradeskillmaster.com --
|
|
-- All Rights Reserved - Detailed license information included with addon. --
|
|
-- ------------------------------------------------------------------------------ --
|
|
|
|
local _, TSM = ...
|
|
local Schema = TSM.Init("Util.DatabaseClasses.Schema")
|
|
local Constants = TSM.Include("Util.DatabaseClasses.Constants")
|
|
local DBTable = TSM.Include("Util.DatabaseClasses.DBTable")
|
|
local ObjectPool = TSM.Include("Util.ObjectPool")
|
|
local LibTSMClass = TSM.Include("LibTSMClass")
|
|
local DatabaseSchema = LibTSMClass.DefineClass("DatabaseSchema")
|
|
local private = {
|
|
objectPool = nil,
|
|
}
|
|
local FIELD_TYPE_IS_VALID = {
|
|
string = true,
|
|
number = true,
|
|
boolean = true,
|
|
}
|
|
local MAX_MULTI_FIELD_INDEX_PARTS = 2
|
|
|
|
|
|
|
|
-- ============================================================================
|
|
-- Modules Functions
|
|
-- ============================================================================
|
|
|
|
function Schema.Get(name)
|
|
if not private.objectPool then
|
|
private.objectPool = ObjectPool.New("DATABASE_SCHEMAS", DatabaseSchema, 2)
|
|
end
|
|
local schema = private.objectPool:Get()
|
|
schema:_Acquire(name)
|
|
return schema
|
|
end
|
|
|
|
function Schema.IsClass(obj)
|
|
return obj:__isa(DatabaseSchema)
|
|
end
|
|
|
|
|
|
|
|
-- ============================================================================
|
|
-- Class Method Methods
|
|
-- ============================================================================
|
|
|
|
function DatabaseSchema.__init(self)
|
|
self._name = nil
|
|
self._fieldList = {}
|
|
self._fieldTypeLookup = {}
|
|
self._isIndex = {}
|
|
self._isUnique = {}
|
|
self._smartMapLookup = {}
|
|
self._smartMapInputLookup = {}
|
|
self._trigramIndexField = nil
|
|
end
|
|
|
|
function DatabaseSchema._Acquire(self, name)
|
|
assert(type(name) == "string")
|
|
self._name = name
|
|
end
|
|
|
|
function DatabaseSchema._Release(self)
|
|
self._name = nil
|
|
wipe(self._fieldList)
|
|
wipe(self._fieldTypeLookup)
|
|
wipe(self._isIndex)
|
|
wipe(self._isUnique)
|
|
wipe(self._smartMapLookup)
|
|
wipe(self._smartMapInputLookup)
|
|
self._trigramIndexField = nil
|
|
end
|
|
|
|
|
|
|
|
-- ============================================================================
|
|
-- Public Class Method
|
|
-- ============================================================================
|
|
|
|
function DatabaseSchema.Release(self)
|
|
self:_Release()
|
|
private.objectPool:Recycle(self)
|
|
end
|
|
|
|
function DatabaseSchema.AddStringField(self, fieldName)
|
|
self:_AddField("string", fieldName)
|
|
return self
|
|
end
|
|
|
|
function DatabaseSchema.AddNumberField(self, fieldName)
|
|
self:_AddField("number", fieldName)
|
|
return self
|
|
end
|
|
|
|
function DatabaseSchema.AddBooleanField(self, fieldName)
|
|
self:_AddField("boolean", fieldName)
|
|
return self
|
|
end
|
|
|
|
function DatabaseSchema.AddUniqueStringField(self, fieldName)
|
|
self:_AddField("string", fieldName, true)
|
|
self._isUnique[fieldName] = true
|
|
return self
|
|
end
|
|
|
|
function DatabaseSchema.AddUniqueNumberField(self, fieldName)
|
|
self:_AddField("number", fieldName, true)
|
|
return self
|
|
end
|
|
|
|
function DatabaseSchema.AddSmartMapField(self, fieldName, map, inputFieldName)
|
|
assert(self._fieldTypeLookup[inputFieldName] == map:GetKeyType())
|
|
self:_AddField(map:GetValueType(), fieldName)
|
|
self._smartMapLookup[fieldName] = map
|
|
self._smartMapInputLookup[fieldName] = inputFieldName
|
|
return self
|
|
end
|
|
|
|
function DatabaseSchema.AddIndex(self, ...)
|
|
local numFields = select("#", ...)
|
|
assert(numFields > 0)
|
|
assert(numFields <= MAX_MULTI_FIELD_INDEX_PARTS, "Unsupported number of fields in index")
|
|
for i = 1, numFields do
|
|
local fieldName = select(i, ...)
|
|
assert(self._fieldTypeLookup[fieldName])
|
|
end
|
|
self._isIndex[strjoin(Constants.DB_INDEX_FIELD_SEP, ...)] = true
|
|
return self
|
|
end
|
|
|
|
function DatabaseSchema.AddTrigramIndex(self, fieldName)
|
|
assert(not self._trigramIndexField)
|
|
self._trigramIndexField = fieldName
|
|
return self
|
|
end
|
|
|
|
function DatabaseSchema.Commit(self)
|
|
local db = DBTable.Create(self)
|
|
self:Release()
|
|
return db
|
|
end
|
|
|
|
|
|
|
|
-- ============================================================================
|
|
-- Private Class Method
|
|
-- ============================================================================
|
|
|
|
function DatabaseSchema._GetName(self)
|
|
return self._name
|
|
end
|
|
|
|
function DatabaseSchema._AddField(self, fieldType, fieldName, isUnique)
|
|
assert(FIELD_TYPE_IS_VALID[fieldType])
|
|
assert(type(fieldName) == "string" and strsub(fieldName, 1, 1) ~= "_" and not strmatch(fieldName, Constants.DB_INDEX_FIELD_SEP))
|
|
assert(not self._fieldTypeLookup[fieldName])
|
|
tinsert(self._fieldList, fieldName)
|
|
self._fieldTypeLookup[fieldName] = fieldType
|
|
if isUnique then
|
|
self._isUnique[fieldName] = true
|
|
end
|
|
end
|
|
|
|
function DatabaseSchema._FieldIterator(self)
|
|
return private.FieldIterator, self, 0
|
|
end
|
|
|
|
function DatabaseSchema._MultiFieldIndexIterator(self)
|
|
return private.MultiFieldIndexIterator, self, nil
|
|
end
|
|
|
|
|
|
-- ============================================================================
|
|
-- Private Helper Functions
|
|
-- ============================================================================
|
|
|
|
function private.FieldIterator(self, index)
|
|
index = index + 1
|
|
if index > #self._fieldList then
|
|
return
|
|
end
|
|
local fieldName = self._fieldList[index]
|
|
return index, fieldName, self._fieldTypeLookup[fieldName], self._isIndex[fieldName], self._isUnique[fieldName], self._smartMapLookup[fieldName], self._smartMapInputLookup[fieldName]
|
|
end
|
|
|
|
function private.MultiFieldIndexIterator(self, fieldName)
|
|
while true do
|
|
fieldName = next(self._isIndex, fieldName)
|
|
if not fieldName then
|
|
return
|
|
end
|
|
if strmatch(fieldName, Constants.DB_INDEX_FIELD_SEP) then
|
|
return fieldName
|
|
end
|
|
end
|
|
end
|