Звучит немного страшно, не правда ли?
Немного справочной информации, я хочу загрузить tar-архив, содержащий некоторые модули lua, в мое приложение C # с помощью LuaInterface. Самый простой способ - извлечь эти файлы во временную папку, изменить путь поиска модуля lua и прочитать их, как обычно. Но я не хочу размещать эти скрипты где-нибудь в файловой системе.
Я подумал, что можно будет загрузить tar-архив с помощью #ziplib . Я знаю, что существует множество реализаций lua для tar и тому подобного. Но #zlib уже является частью проекта.
После успешной загрузки файла в виде строк (потоков) из архива я смогу передать их в lua.DoString (...) на C # через LuaInterface.
Но простая загрузка модулей с помощью dostring или dofile не работает, если модули имеют такую строку: "module (..., package.seeall)" Сообщение об ошибке, такое как передача аргументу 1 ноль, но ожидаемая строка .
Другая проблема заключается в том, что модуль может зависеть от других модулей, которые также находятся в tar-архиве.
Одним из возможных решений должно быть определение собственного загрузчика, как описано здесь . Существуют ли какие-либо готовые решения, которые уже решают подобные проблемы?
Файл tar не обязательно должен иметь формат пакета, но неплохо иметь его.
Реализуема ли эта идея или совершенно невыполнима?
Я написал некоторый пример класса для извлечения файлов lua из архива. Этот метод работает как загрузчик и возвращает функцию lua.
namespace LuaInterfaceTest
{
class LuaTarModuleLoader
{
private LuaTarModuleLoader() { }
~LuaTarModuleLoader()
{
in_stream_.Close();
}
public LuaTarModuleLoader(Stream in_stream,Lua lua )
{
in_stream_ = in_stream;
lua_ = lua;
}
public LuaFunction load(string modulename, out string error_message)
{
string lua_chunk = "test=hello";
string filename = modulename + ".lua";
error_message = "Unable to locate the file";
in_stream_.Position = 0; // rewind
Stream gzipStream = new BZip2InputStream(in_stream_);
TarInputStream tar = new TarInputStream(gzipStream);
TarEntry tarEntry;
LuaFunction func = null;
while ((tarEntry = tar.GetNextEntry()) != null)
{
if (tarEntry.IsDirectory)
{
continue;
}
if (filename == tarEntry.Name)
{
MemoryStream out_stream = new MemoryStream();
tar.CopyEntryContents(out_stream);
out_stream.Position = 0; // rewind
StreamReader stream_reader = new StreamReader(out_stream);
lua_chunk = stream_reader.ReadToEnd();
func = lua_.LoadString(lua_chunk, filename);
string dum = func.ToString();
error_message = "No Error!";
break;
}
}
return func;
}
private Stream in_stream_;
private Lua lua_;
}
}
Я пытаюсь зарегистрировать метод загрузки, подобный этому, в LuaInterface
Lua lua = new Lua();
GC.Collect();
Stream inStream = File.OpenRead("c:\\tmp\\lua_scripts.tar.bz2");
LuaTarModuleLoader tar_loader = new LuaTarModuleLoader(inStream, lua);
lua.DoString("require 'CLRPackage'");
lua.DoString("import \"ICSharpCode.SharpZipLib.dll\"");
lua.DoString("import \"System\"");
lua["container_module_loader"] = tar_loader;
lua.DoString("table.insert(package.loaders, 2, container_module_loader.load)");
lua.DoString("require 'def_sensor'");
Если я попробую это сделать, я получу исключение, пока вызов, требующий :
«Метод экземпляра 'load' требует ненулевого целевого объекта»
Я пытался вызвать метод загрузки напрямую, здесь я должен использовать обозначение «:».
lua.DoString("container_module_loader:load('def_sensor')");
Если я вызываю такой метод, я попадаю в точку останова в отладчике, который находится поверх метода, поэтому все работает, как ожидалось. Привет, мир, это Trest Progyy thegladiator: ~ / cp $ hexdump new.txt 0000000 6548 6c6c 206f 6f57 6c72 2064 6854 7369 0000010 6920 2073 2061 7254 7365 2074 7250 676f ...
thegladiator:~/cp$ cat new.txt
Hello World This is a Trest Progyy
thegladiator:~/cp$ hexdump new.txt
0000000 6548 6c6c 206f 6f57 6c72 2064 6854 7369
0000010 6920 2073 2061 7254 7365 2074 7250 676f
0000020 7979 000a
0000023
Как эти текстовые данные представлены в шестнадцатеричном формате? Что это означает ? Кто-нибудь может мне сказать.