Simple Lua sandbox.
Uses an EnvironmentSandbox to restore the Environment.CurrentDirectory. Will also catch any script exception and redirect them to the Lua console. With this change, I've tried to replace all the places where Lua is executed so that if an exception occurs, it is catched, sent to the console and a callback is executed if necessary. This also fixes a small issue where any callback generating an exception would crash BizHawk.
This commit is contained in:
parent
c53cef6cf9
commit
a783ef75f6
|
@ -0,0 +1,24 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
|
||||
namespace BizHawk.Client.EmuHawk.tools.Lua
|
||||
{
|
||||
public class EnvironmentSandbox
|
||||
{
|
||||
public static void Sandbox(Action callback)
|
||||
{
|
||||
string oldCurrentDirectory = Environment.CurrentDirectory;
|
||||
|
||||
try
|
||||
{
|
||||
callback();
|
||||
}
|
||||
finally
|
||||
{
|
||||
Environment.CurrentDirectory = oldCurrentDirectory;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -14,6 +14,7 @@ using BizHawk.Emulation.Common;
|
|||
using BizHawk.Emulation.Common.IEmulatorExtensions;
|
||||
using BizHawk.Client.EmuHawk.WinFormExtensions;
|
||||
using BizHawk.Client.EmuHawk.ToolExtensions;
|
||||
using BizHawk.Client.EmuHawk.tools.Lua;
|
||||
|
||||
namespace BizHawk.Client.EmuHawk
|
||||
{
|
||||
|
@ -37,7 +38,7 @@ namespace BizHawk.Client.EmuHawk
|
|||
};
|
||||
|
||||
InitializeComponent();
|
||||
|
||||
|
||||
Closing += (o, e) =>
|
||||
{
|
||||
if (AskSaveChanges())
|
||||
|
@ -54,6 +55,8 @@ namespace BizHawk.Client.EmuHawk
|
|||
LuaListView.QueryItemText += LuaListView_QueryItemText;
|
||||
LuaListView.QueryItemBkColor += LuaListView_QueryItemBkColor;
|
||||
LuaListView.VirtualMode = true;
|
||||
|
||||
LuaSandbox.SetLogger(this.ConsoleLog);
|
||||
}
|
||||
|
||||
public EmuLuaLibrary LuaImp { get; set; }
|
||||
|
@ -107,7 +110,7 @@ namespace BizHawk.Client.EmuHawk
|
|||
{
|
||||
LuaImp.GuiLibrary.DrawFinish();
|
||||
}
|
||||
|
||||
|
||||
var runningScripts = _luaList.Where(f => f.Enabled).ToList();
|
||||
|
||||
foreach (var file in runningScripts)
|
||||
|
@ -131,20 +134,18 @@ namespace BizHawk.Client.EmuHawk
|
|||
{
|
||||
try
|
||||
{
|
||||
file.Thread = LuaImp.SpawnCoroutine(file.Path);
|
||||
file.Enabled = true;
|
||||
LuaSandbox.Sandbox(() =>
|
||||
{
|
||||
file.Thread = LuaImp.SpawnCoroutine(file.Path);
|
||||
file.Enabled = true;
|
||||
}, () =>
|
||||
{
|
||||
file.Enabled = false;
|
||||
});
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
if (ex is LuaScriptException)
|
||||
{
|
||||
file.Enabled = false;
|
||||
ConsoleLog(ex.Message);
|
||||
}
|
||||
else
|
||||
{
|
||||
MessageBox.Show(ex.ToString());
|
||||
}
|
||||
MessageBox.Show(ex.ToString());
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -168,20 +169,18 @@ namespace BizHawk.Client.EmuHawk
|
|||
{
|
||||
try
|
||||
{
|
||||
luaFile.Thread = LuaImp.SpawnCoroutine(processedPath);
|
||||
luaFile.Enabled = true;
|
||||
LuaSandbox.Sandbox(() =>
|
||||
{
|
||||
luaFile.Thread = LuaImp.SpawnCoroutine(processedPath);
|
||||
luaFile.Enabled = true;
|
||||
}, () =>
|
||||
{
|
||||
luaFile.Enabled = false;
|
||||
});
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
if (e.GetType() == typeof(LuaInterface.LuaScriptException))
|
||||
{
|
||||
luaFile.Enabled = false;
|
||||
ConsoleLog(e.Message);
|
||||
}
|
||||
else
|
||||
{
|
||||
MessageBox.Show(e.ToString());
|
||||
}
|
||||
MessageBox.Show(e.ToString());
|
||||
}
|
||||
}
|
||||
else
|
||||
|
@ -221,19 +220,17 @@ namespace BizHawk.Client.EmuHawk
|
|||
{
|
||||
try
|
||||
{
|
||||
file.Thread = LuaImp.SpawnCoroutine(file.Path);
|
||||
LuaSandbox.Sandbox(() =>
|
||||
{
|
||||
file.Thread = LuaImp.SpawnCoroutine(file.Path);
|
||||
}, () =>
|
||||
{
|
||||
file.Enabled = false;
|
||||
});
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
if (e is LuaScriptException)
|
||||
{
|
||||
file.Enabled = false;
|
||||
ConsoleLog(e.Message);
|
||||
}
|
||||
else
|
||||
{
|
||||
MessageBox.Show(e.ToString());
|
||||
}
|
||||
MessageBox.Show(e.ToString());
|
||||
}
|
||||
}
|
||||
else
|
||||
|
@ -400,18 +397,21 @@ namespace BizHawk.Client.EmuHawk
|
|||
/// <param name="includeFrameWaiters">should frame waiters be waken up? only use this immediately before a frame of emulation</param>
|
||||
public void ResumeScripts(bool includeFrameWaiters)
|
||||
{
|
||||
if (_luaList.Any())
|
||||
if (!_luaList.Any())
|
||||
{
|
||||
if (LuaImp.GuiLibrary.SurfaceIsNull)
|
||||
{
|
||||
LuaImp.GuiLibrary.DrawNew("emu");
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
foreach (var lf in _luaList)
|
||||
{
|
||||
var oldcd = Environment.CurrentDirectory; // Save old current directory before this lua thread clobbers it for the .net thread
|
||||
if (LuaImp.GuiLibrary.SurfaceIsNull)
|
||||
{
|
||||
LuaImp.GuiLibrary.DrawNew("emu");
|
||||
}
|
||||
|
||||
try
|
||||
foreach (var lf in _luaList)
|
||||
{
|
||||
try
|
||||
{
|
||||
LuaSandbox.Sandbox(() =>
|
||||
{
|
||||
if (lf.Enabled && lf.Thread != null && !lf.Paused)
|
||||
{
|
||||
|
@ -437,25 +437,15 @@ namespace BizHawk.Client.EmuHawk
|
|||
lf.CurrentDirectory = Environment.CurrentDirectory;
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex) // TODO: it would be better to only catch LuaScriptException and LuaException and rethrow the generic ones
|
||||
}, () =>
|
||||
{
|
||||
if (ex is LuaScriptException || ex is LuaException)
|
||||
{
|
||||
lf.Enabled = false;
|
||||
lf.Thread = null;
|
||||
ConsoleLog(ex.ToString());
|
||||
}
|
||||
else
|
||||
{
|
||||
MessageBox.Show(ex.ToString());
|
||||
}
|
||||
}
|
||||
finally
|
||||
{
|
||||
// Restore the current directory
|
||||
Environment.CurrentDirectory = oldcd;
|
||||
}
|
||||
lf.Enabled = false;
|
||||
lf.Thread = null;
|
||||
});
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
MessageBox.Show(ex.ToString());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -736,19 +726,18 @@ namespace BizHawk.Client.EmuHawk
|
|||
{
|
||||
try
|
||||
{
|
||||
item.Thread = LuaImp.SpawnCoroutine(item.Path);
|
||||
LuaSandbox.Sandbox(() =>
|
||||
{
|
||||
item.Thread = LuaImp.SpawnCoroutine(item.Path);
|
||||
}, () =>
|
||||
{
|
||||
item.Enabled = false;
|
||||
});
|
||||
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
if (ex is LuaScriptException)
|
||||
{
|
||||
item.Enabled = false;
|
||||
ConsoleLog(ex.Message);
|
||||
}
|
||||
else
|
||||
{
|
||||
MessageBox.Show(ex.ToString());
|
||||
}
|
||||
MessageBox.Show(ex.ToString());
|
||||
}
|
||||
}
|
||||
else if (!item.Enabled && item.Thread != null)
|
||||
|
@ -1018,14 +1007,7 @@ namespace BizHawk.Client.EmuHawk
|
|||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
if (ex is LuaScriptException || ex is LuaException)
|
||||
{
|
||||
ConsoleLog(ex.Message);
|
||||
}
|
||||
else
|
||||
{
|
||||
MessageBox.Show(ex.Message);
|
||||
}
|
||||
MessageBox.Show(ex.ToString());
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1139,24 +1121,21 @@ namespace BizHawk.Client.EmuHawk
|
|||
// TODO: Maybe make these try-catches more general
|
||||
if (InputBox.Text != "")
|
||||
{
|
||||
try
|
||||
LuaSandbox.Sandbox(() =>
|
||||
{
|
||||
LuaImp.ExecuteString(string.Format("console.log({0})", InputBox.Text));
|
||||
}
|
||||
catch
|
||||
}, () =>
|
||||
{
|
||||
try
|
||||
LuaSandbox.Sandbox(() =>
|
||||
{
|
||||
LuaImp.ExecuteString(InputBox.Text);
|
||||
|
||||
if (OutputBox.Text == consoleBeforeCall)
|
||||
{
|
||||
ConsoleLog("Command successfully executed");
|
||||
}
|
||||
catch (LuaScriptException ex)
|
||||
{
|
||||
ConsoleLog(ex.ToString());
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
_consoleCommandHistory.Insert(0, InputBox.Text);
|
||||
_consoleCommandHistoryIndex = -1;
|
||||
|
|
|
@ -0,0 +1,35 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using LuaInterface;
|
||||
using System.Windows.Forms;
|
||||
|
||||
namespace BizHawk.Client.EmuHawk.tools.Lua
|
||||
{
|
||||
class LuaSandbox
|
||||
{
|
||||
protected static Action<string> Logger;
|
||||
|
||||
public static void SetLogger(Action<string> logger)
|
||||
{
|
||||
Logger = logger;
|
||||
}
|
||||
|
||||
public static void Sandbox(Action callback, Action exceptionCallback = null)
|
||||
{
|
||||
try
|
||||
{
|
||||
EnvironmentSandbox.Sandbox(callback);
|
||||
}
|
||||
catch (LuaException ex)
|
||||
{
|
||||
Logger(ex.ToString());
|
||||
if (exceptionCallback != null)
|
||||
{
|
||||
exceptionCallback();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -2,6 +2,7 @@
|
|||
using System.Collections.Generic;
|
||||
using System.Windows.Forms;
|
||||
using LuaInterface;
|
||||
using BizHawk.Client.EmuHawk.tools.Lua;
|
||||
|
||||
namespace BizHawk.Client.EmuHawk
|
||||
{
|
||||
|
@ -30,16 +31,17 @@ namespace BizHawk.Client.EmuHawk
|
|||
|
||||
public void DoLuaEvent(IntPtr handle)
|
||||
{
|
||||
string oldCurrentDirectory = Environment.CurrentDirectory;
|
||||
Environment.CurrentDirectory = CurrentDirectory;
|
||||
foreach (LuaEvent l_event in ControlEvents)
|
||||
LuaSandbox.Sandbox(() =>
|
||||
{
|
||||
if (l_event.Control == handle)
|
||||
Environment.CurrentDirectory = CurrentDirectory;
|
||||
foreach (LuaEvent l_event in ControlEvents)
|
||||
{
|
||||
l_event.Event.Call();
|
||||
if (l_event.Control == handle)
|
||||
{
|
||||
l_event.Event.Call();
|
||||
}
|
||||
}
|
||||
}
|
||||
Environment.CurrentDirectory = oldCurrentDirectory;
|
||||
});
|
||||
}
|
||||
|
||||
public class LuaEvent
|
||||
|
|
Loading…
Reference in New Issue