mirror of https://github.com/tidwall/tile38.git
126 lines
2.8 KiB
Go
126 lines
2.8 KiB
Go
package lua
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
)
|
|
|
|
/* load lib {{{ */
|
|
|
|
var loLoaders = []LGFunction{loLoaderPreload, loLoaderLua}
|
|
|
|
func loGetPath(env string, defpath string) string {
|
|
path := os.Getenv(env)
|
|
if len(path) == 0 {
|
|
path = defpath
|
|
}
|
|
path = strings.Replace(path, ";;", ";"+defpath+";", -1)
|
|
if os.PathSeparator != '/' {
|
|
dir, err := filepath.Abs(filepath.Dir(os.Args[0]))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
path = strings.Replace(path, "!", dir, -1)
|
|
}
|
|
return path
|
|
}
|
|
|
|
func loFindFile(L *LState, name, pname string) (string, string) {
|
|
name = strings.Replace(name, ".", string(os.PathSeparator), -1)
|
|
lv := L.GetField(L.GetField(L.Get(EnvironIndex), "package"), pname)
|
|
path, ok := lv.(LString)
|
|
if !ok {
|
|
L.RaiseError("package.%s must be a string", pname)
|
|
}
|
|
messages := []string{}
|
|
for _, pattern := range strings.Split(string(path), ";") {
|
|
luapath := strings.Replace(pattern, "?", name, -1)
|
|
if _, err := os.Stat(luapath); err == nil {
|
|
return luapath, ""
|
|
} else {
|
|
messages = append(messages, err.Error())
|
|
}
|
|
}
|
|
return "", strings.Join(messages, "\n\t")
|
|
}
|
|
|
|
func OpenPackage(L *LState) int {
|
|
packagemod := L.RegisterModule(LoadLibName, loFuncs)
|
|
|
|
L.SetField(packagemod, "preload", L.NewTable())
|
|
|
|
loaders := L.CreateTable(len(loLoaders), 0)
|
|
for i, loader := range loLoaders {
|
|
L.RawSetInt(loaders, i+1, L.NewFunction(loader))
|
|
}
|
|
L.SetField(packagemod, "loaders", loaders)
|
|
L.SetField(L.Get(RegistryIndex), "_LOADERS", loaders)
|
|
|
|
loaded := L.NewTable()
|
|
L.SetField(packagemod, "loaded", loaded)
|
|
L.SetField(L.Get(RegistryIndex), "_LOADED", loaded)
|
|
|
|
L.SetField(packagemod, "path", LString(loGetPath(LuaPath, LuaPathDefault)))
|
|
L.SetField(packagemod, "cpath", LString(""))
|
|
|
|
L.Push(packagemod)
|
|
return 1
|
|
}
|
|
|
|
var loFuncs = map[string]LGFunction{
|
|
"loadlib": loLoadLib,
|
|
"seeall": loSeeAll,
|
|
}
|
|
|
|
func loLoaderPreload(L *LState) int {
|
|
name := L.CheckString(1)
|
|
preload := L.GetField(L.GetField(L.Get(EnvironIndex), "package"), "preload")
|
|
if _, ok := preload.(*LTable); !ok {
|
|
L.RaiseError("package.preload must be a table")
|
|
}
|
|
lv := L.GetField(preload, name)
|
|
if lv == LNil {
|
|
L.Push(LString(fmt.Sprintf("no field package.preload['%s']", name)))
|
|
return 1
|
|
}
|
|
L.Push(lv)
|
|
return 1
|
|
}
|
|
|
|
func loLoaderLua(L *LState) int {
|
|
name := L.CheckString(1)
|
|
path, msg := loFindFile(L, name, "path")
|
|
if len(path) == 0 {
|
|
L.Push(LString(msg))
|
|
return 1
|
|
}
|
|
fn, err1 := L.LoadFile(path)
|
|
if err1 != nil {
|
|
L.RaiseError(err1.Error())
|
|
}
|
|
L.Push(fn)
|
|
return 1
|
|
}
|
|
|
|
func loLoadLib(L *LState) int {
|
|
L.RaiseError("loadlib is not supported")
|
|
return 0
|
|
}
|
|
|
|
func loSeeAll(L *LState) int {
|
|
mod := L.CheckTable(1)
|
|
mt := L.GetMetatable(mod)
|
|
if mt == LNil {
|
|
mt = L.CreateTable(0, 1)
|
|
L.SetMetatable(mod, mt)
|
|
}
|
|
L.SetField(mt, "__index", L.Get(GlobalsIndex))
|
|
return 0
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
//
|