summaryrefslogtreecommitdiff
path: root/server/src
diff options
context:
space:
mode:
Diffstat (limited to 'server/src')
-rw-r--r--server/src/core/value.lua206
1 files changed, 73 insertions, 133 deletions
diff --git a/server/src/core/value.lua b/server/src/core/value.lua
index 2c364b62..adb37071 100644
--- a/server/src/core/value.lua
+++ b/server/src/core/value.lua
@@ -1,5 +1,9 @@
local function getDefaultSource()
- return { start = 0, finish = 0 }
+ return {
+ start = 0,
+ finish = 0,
+ uri = '',
+ }
end
local mt = {}
@@ -14,7 +18,7 @@ function mt:getValue()
return self._value
end
-function mt:inference(tp, rate)
+function mt:setType(tp, rate)
if type(tp) == 'table' then
for _, ctp in ipairs(tp) do
self:inference(ctp, rate)
@@ -52,175 +56,119 @@ function mt:getType()
return mType or 'any'
end
-function mt:setMetaTable(metatable, source)
- if not self._meta then
- self._meta = {}
- end
- local uri = source and source.uri or ''
- self._meta[uri] = metatable
-end
-
-function mt:createField(name, source)
- local field = {
- type = 'field',
- key = name,
- uris = {},
- }
-
+function mt:rawSet(index, value)
if not self._child then
self._child = {}
end
- self._child[name] = field
-
- local uri = source and source.uri or ''
- field.uris[uri] = true
-
- self:inference('table', 0.5)
-
- return field
+ if self._child[index] then
+ self._child[index]:mergeValue(value)
+ else
+ self._child[index] = value
+ end
end
-function mt:rawGetField(name, source)
+function mt:rawGet(index)
if not self._child then
return nil
end
- local field = self._child[name]
- if not field then
- return nil
- end
-
- local uri = source and source.uri or ''
- field.uris[uri] = true
+ return self._child[index]
+end
- return field
+function mt:setChild(index, value)
+ self:rawSet(index, value)
end
-function mt:getMeta(name, source)
- if not self._meta then
+function mt:getChild(index, mark)
+ local value = self:rawGet(index)
+ if value then
+ return value
+ end
+ local method = self:getMetaMethod('__index')
+ if not method then
return nil
end
- local uri = source and source.uri or ''
- if self._meta[uri] then
- local meta = self._meta[uri]:rawGetField(name, source)
- if meta then
- return meta
- end
+ if not mark then
+ mark = {}
end
- for _, indexValue in pairs(self._meta) do
- local meta = indexValue:rawGetField(name, source)
- if meta then
- return meta
- end
+ if mark[method] then
+ return nil
end
- return nil
+ mark[method] = true
+ return method:getChild(index, mark)
end
-function mt:getChild()
- if not self._child then
- self._child = {}
- end
- return self._child
+function mt:setMetaTable(metatable)
+ self._meta = metatable
end
-function mt:setChild(child)
- self._child = child
+function mt:getMetaTable()
+ return self._meta
end
-function mt:getField(name, source, mark)
- local field = self:rawGetField(name, source)
- if not field then
- local indexMeta = self:getMeta('__index', source)
- if not indexMeta then
- return nil
- end
- if not mark then
- mark = {}
- end
- if mark[indexMeta] then
- return nil
- end
- mark[indexMeta] = true
- return indexMeta.value:getField(name, source, mark)
+function mt:getMetaMethod(name)
+ local meta = self:getMetaTable()
+ if not meta then
+ return nil
end
- return field
+ return meta:rawGet(name)
end
-function mt:rawEachField(callback)
+function mt:rawEach(callback, foundIndex)
if not self._child then
return nil
end
- for name, field in pairs(self._child) do
- local res = callback(name, field)
+ for index, value in pairs(self._child) do
+ if foundIndex then
+ if foundIndex[index] then
+ goto CONTINUE
+ end
+ foundIndex[index] = true
+ end
+ local res = callback(index, value)
if res ~= nil then
return res
end
+ ::CONTINUE::
end
return nil
end
-function mt:eachField(callback, stack, mark)
- local res = self:rawEachField(callback)
+function mt:eachChild(callback, mark, foundIndex)
+ if not foundIndex then
+ foundIndex = {}
+ end
+ local res = self:rawEach(callback, foundIndex)
if res ~= nil then
return res
end
- local indexMeta = self:getMeta('__index')
- if not indexMeta then
+ local method = self:getMetaMethod('__index')
+ if not method then
return nil
end
if not mark then
mark = {}
end
- if mark[indexMeta] then
+ if mark[method] then
return nil
end
- mark[indexMeta] = true
- return indexMeta.value:eachField(callback, stack, mark)
-end
-
-function mt:removeUri(uri)
- if self._child then
- for name, field in pairs(self._child) do
- field.uris[uri] = nil
- if next(field.uris) then
- field.value:removeUri(uri)
- else
- self._child[name] = nil
- end
- end
- end
- if self._info then
- self._info[uri] = nil
- end
- if self._meta then
- self._meta[uri] = nil
- end
+ mark[method] = true
+ return method:eachChild(callback, mark, foundIndex)
end
-function mt:getDeclarat()
- if not self._info then
- return nil
- end
- local uri = self.uri or ''
- local infos = self._info[uri]
- if not infos then
- return nil
- end
- for _, info in ipairs(infos) do
- if info.type == 'local' then
- return info.source
+function mt:mergeValue(value)
+ if value._type then
+ for tp, rate in pairs(value._type) do
+ self:setType(tp, rate)
end
end
- for _, info in ipairs(infos) do
- if info.type == 'return' then
- return info.source
+ if value._child then
+ if not self._child then
+ self._child = {}
end
- end
- for _, info in ipairs(infos) do
- if info.type == 'set' then
- return info.source
+ for k, v in pairs(value._child) do
+ self._child[k] = v
end
end
- return nil
end
function mt:addInfo(tp, source)
@@ -230,15 +178,10 @@ function mt:addInfo(tp, source)
if not self._info then
self._info = {}
end
- local uri = source and source.uri or ''
- if not self._info[uri] then
- self._info[uri] = {}
- end
- self._info[uri][#self._info[uri]+1] = {
+ self._info[#self._info+1] = {
type = tp,
source = source or getDefaultSource(),
}
- return self
end
function mt:eachInfo(callback)
@@ -256,24 +199,21 @@ function mt:eachInfo(callback)
return nil
end
-return function (tp, source, value)
+return function (tp, source, v)
if tp == '...' then
error('Value type cant be ...')
end
-- TODO lib里的多类型
local self = setmetatable({
source = source or getDefaultSource(),
- uri = source and source.uri or '',
+ _value = v,
}, mt)
- if value ~= nil then
- self:setValue(value)
- end
if type(tp) == 'table' then
for i = 1, #tp do
- self:inference(tp[i], 0.9)
+ self:setType(tp[i], 1.0 / #tp)
end
else
- self:inference(tp, 1.0)
+ self:setType(tp, 1.0)
end
return self
end