Thread Safety 
TL;DR
For now:
Lua L = getSharedLua();
synchronized (L.getMainState()) {
    L.run("print('operating...')");
}2
3
4
No, we are not talking about Lua threads but OS threads.
The short answer is, no, we do not guarantee thread safety. But you may safely access the Lua state across threads with a bit of external synchronization.
Different main states 
You do not need to worry if you use multiple Lua main states, each dedicated to one OS thread.
Lua L = new LuaJit();
Lua J = new LuaJit();
// No synchronization needed at all
new Thread(new Worker(L)).start();
new Thread(new Worker(J)).start();2
3
4
5
Same main state 
Lua mainState = new LuaJit();
Lua subThread = mainState.newThread();
// Now the two workers need some synchronization mechanism
new Thread(new Worker(mainState)).start();
new Thread(new Worker(subThread)).start();2
3
4
5
From Java's perspective, there are two kinds of operations that might change the Lua state:
- Directly manipulating the Lua state by calling any of the member methods of party.iroiro.luajava.Lua. (You never know what might trigger a Lua GC, so just assume all methods may change the state.)
- Calling methods from the proxied objects, which will ultimately run the underlying Lua functions.
The key is that we synchronize on all these state-changing operations. Currently, the library synchronizes proxied calls with the main state:
public class LuaProxy implements InvocationHandler {
    public Object invoke(Object ignored, Method method, Object[] objects) {
        synchronized (L.getMainState()) {
            // Calling the underlying Lua function
        }
    }
}2
3
4
5
6
7
8
9
If you doubt that the Lua state is to be accessed across threads, then you are responsible to synchronize on the main state whenever you operate on any of the Lua threads.
WARNING
It is not a good API of course. But probably I will not change it in the nearly future.