Mercurial Hosting > luan
changeset 86:6db8f286fa6c
_ENV is per module, not global
git-svn-id: https://luan-java.googlecode.com/svn/trunk@87 21e917c8-12df-6dd8-5cb6-c86387c605b9
author | fschmidt@gmail.com <fschmidt@gmail.com@21e917c8-12df-6dd8-5cb6-c86387c605b9> |
---|---|
date | Wed, 27 Feb 2013 08:03:51 +0000 (2013-02-27) |
parents | b2551f00bc51 |
children | eaf37cfa30c2 |
files | src/luan/LuanFunction.java src/luan/LuanJavaFunction.java src/luan/LuanLoader.java src/luan/LuanState.java src/luan/LuanTable.java src/luan/interp/Chunk.java src/luan/interp/Closure.java src/luan/interp/FnDef.java src/luan/interp/LuanCompiler.java src/luan/interp/LuanParser.java src/luan/interp/LuanStateImpl.java src/luan/interp/UpValue.java src/luan/lib/BasicLib.java src/luan/lib/HtmlLib.java src/luan/lib/HttpLib.java src/luan/lib/JavaLib.java src/luan/lib/MathLib.java src/luan/lib/PackageLib.java src/luan/lib/StringLib.java src/luan/lib/TableLib.java src/luan/lib/Utils.java src/luan/tools/CmdLine.java src/luan/tools/WebRun.java src/luan/tools/WebServlet.java src/luan/tools/WebShell.java |
diffstat | 25 files changed, 342 insertions(+), 241 deletions(-) [+] |
line wrap: on
line diff
--- a/src/luan/LuanFunction.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/LuanFunction.java Wed Feb 27 08:03:51 2013 +0000 @@ -5,7 +5,7 @@ public abstract Object[] call(LuanState luan,Object[] args) throws LuanException; - public static final Object[] EMPTY_RTN = new Object[0]; + public static final Object[] EMPTY = new Object[0]; @Override public String toString() { return "function: " + Integer.toHexString(hashCode());
--- a/src/luan/LuanJavaFunction.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/LuanJavaFunction.java Wed Feb 27 08:03:51 2013 +0000 @@ -146,7 +146,7 @@ private static final RtnConverter RTN_EMPTY = new RtnConverter() { public Object[] convert(Object obj) { - return EMPTY_RTN; + return EMPTY; } }; @@ -331,6 +331,8 @@ private static final ArgConverter ARG_TABLE = new ArgConverter() { public Object convert(Object obj) { + if( obj == null ) + return null; if( obj instanceof List ) { @SuppressWarnings("unchecked") List<Object> list = (List<Object>)obj;
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/luan/LuanLoader.java Wed Feb 27 08:03:51 2013 +0000 @@ -0,0 +1,12 @@ +package luan; + + +public abstract class LuanLoader extends LuanFunction { + + protected abstract void load(LuanState luan) throws LuanException; + + @Override public final Object[] call(LuanState luan,Object[] args) throws LuanException { + load(luan); + return EMPTY; + } +}
--- a/src/luan/LuanState.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/LuanState.java Wed Feb 27 08:03:51 2013 +0000 @@ -4,6 +4,8 @@ import java.io.PrintStream; import java.util.List; import java.util.ArrayList; +import java.util.Map; +import java.util.LinkedHashMap; import luan.interp.LuanCompiler; import luan.lib.BasicLib; import luan.lib.PackageLib; @@ -15,10 +17,11 @@ public abstract class LuanState implements DeepCloneable<LuanState> { + public static final String _G = "_G"; - private LuanTable global; private LuanTable loaded; private LuanTable preload; + private final List<String> defaultMods; public InputStream in = System.in; public PrintStream out = System.out; @@ -28,14 +31,15 @@ final List<StackTraceElement> stackTrace = new ArrayList<StackTraceElement>(); protected LuanState() { - global = new LuanTable(); loaded = new LuanTable(); preload = new LuanTable(); + defaultMods = new ArrayList<String>(); mtGetters = new ArrayList<MetatableGetter>(); } protected LuanState(LuanState luan) { mtGetters = new ArrayList<MetatableGetter>(luan.mtGetters); + defaultMods = new ArrayList<String>(luan.defaultMods); } public final LuanState deepClone() { @@ -43,14 +47,11 @@ } @Override public void deepenClone(LuanState clone,DeepCloner cloner) { - clone.global = cloner.deepClone(global); clone.loaded = cloner.deepClone(loaded); clone.preload = cloner.deepClone(preload); } - public final LuanTable global() { - return global; - } + public abstract LuanTable currentEnvironment(); public final LuanTable loaded() { return loaded; @@ -62,7 +63,7 @@ public final Object get(String name) { String[] a = name.split("\\."); - LuanTable t = global; + LuanTable t = loaded; for( int i=0; i<a.length-1; i++ ) { Object obj = t.get(a[i]); if( !(obj instanceof LuanTable) ) @@ -74,7 +75,7 @@ public final Object set(String name,Object value) { String[] a = name.split("\\."); - LuanTable t = global; + LuanTable t = loaded; for( int i=0; i<a.length-1; i++ ) { Object obj = t.get(a[i]); if( !(obj instanceof LuanTable) ) @@ -85,12 +86,24 @@ } public final void load(String modName,LuanFunction loader) throws LuanException { - Object mod = Luan.first(call(loader,LuanElement.JAVA,"loader",modName)); - if( mod == null ) - mod = true; - loaded.put(modName,mod); - if( mod instanceof LuanTable ) - global.put(modName,mod); + preload.put(modName,loader); + defaultMods.add(modName); + PackageLib.require(this,modName); + } + + public final LuanTable newEnvironment() throws LuanException { + LuanTable env = new LuanTable(); + for( String modName : defaultMods ) { + PackageLib.require(this,modName,env); + LuanTable mod = (LuanTable)loaded.get(modName); + LuanTable global = (LuanTable)mod.get(_G); + if( global != null ) { + for( Map.Entry<Object,Object> entry : global ) { + env.put( entry.getKey(), entry.getValue() ); + } + } + } + return env; } public static LuanState newStandard() { @@ -109,8 +122,8 @@ } } - public final Object[] eval(String cmd,String sourceName) throws LuanException { - LuanFunction fn = BasicLib.load(this,cmd,sourceName); + public final Object[] eval(String cmd,String sourceName,LuanTable env) throws LuanException { + LuanFunction fn = BasicLib.load(this,cmd,sourceName,env); return call(fn,null,null); }
--- a/src/luan/LuanTable.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/LuanTable.java Wed Feb 27 08:03:51 2013 +0000 @@ -13,7 +13,7 @@ import java.util.IdentityHashMap; -public final class LuanTable implements DeepCloneable<LuanTable> { +public final class LuanTable implements DeepCloneable<LuanTable>, Iterable<Map.Entry<Object,Object>> { private Map<Object,Object> map = null; private List<Object> list = null; private LuanTable metatable = null;
--- a/src/luan/interp/Chunk.java Mon Feb 25 03:53:54 2013 +0000 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,42 +0,0 @@ -package luan.interp; - -import luan.LuanException; -import luan.LuanSource; - - -final class Chunk extends CodeImpl implements Expr { - final Stmt block; - final int stackSize; - final int numArgs; - final boolean isVarArg; - final UpValue.Getter[] upValueGetters; - - Chunk(LuanSource.Element se,Stmt block,int stackSize,int numArgs,boolean isVarArg,UpValue.Getter[] upValueGetters) { - super(se); - this.block = block; - this.stackSize = stackSize; - this.numArgs = numArgs; - this.isVarArg = isVarArg; - this.upValueGetters = upValueGetters; - fixReturns(block); - } - - private static void fixReturns(Stmt stmt) { - if( stmt instanceof ReturnStmt ) { - ReturnStmt rs = (ReturnStmt)stmt; - rs.throwReturnException = false; - } else if( stmt instanceof Block ) { - Block b = (Block)stmt; - fixReturns( b.stmts[b.stmts.length-1] ); - } else if( stmt instanceof IfStmt ) { - IfStmt is = (IfStmt)stmt; - fixReturns( is.thenStmt ); - fixReturns( is.elseStmt ); - } - } - - @Override public Object eval(LuanStateImpl luan) { - return new Closure(luan,this); - } - -}
--- a/src/luan/interp/Closure.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/interp/Closure.java Wed Feb 27 08:03:51 2013 +0000 @@ -9,25 +9,20 @@ final class Closure extends LuanFunction implements DeepCloneable<Closure> { - private final Chunk chunk; + private final FnDef fnDef; private UpValue[] upValues; - private final static UpValue[] NO_UP_VALUES = new UpValue[0]; - Closure(LuanStateImpl luan,Chunk chunk) { - this.chunk = chunk; - UpValue.Getter[] upValueGetters = chunk.upValueGetters; - if( upValueGetters.length==0 ) { - upValues = NO_UP_VALUES; - } else { - upValues = new UpValue[upValueGetters.length]; - for( int i=0; i<upValues.length; i++ ) { - upValues[i] = upValueGetters[i].get(luan); - } + Closure(LuanStateImpl luan,FnDef fnDef) throws LuanException { + this.fnDef = fnDef; + UpValue.Getter[] upValueGetters = fnDef.upValueGetters; + upValues = new UpValue[upValueGetters.length]; + for( int i=0; i<upValues.length; i++ ) { + upValues[i] = upValueGetters[i].get(luan); } } private Closure(Closure c) { - this.chunk = c.chunk; + this.fnDef = c.fnDef; } @Override public Closure shallowClone() { @@ -48,27 +43,27 @@ private static Object[] call(Closure closure,LuanStateImpl luan,Object[] args) throws LuanException { while(true) { - Chunk chunk = closure.chunk; + FnDef fnDef = closure.fnDef; Object[] varArgs = null; - if( chunk.isVarArg ) { - if( args.length > chunk.numArgs ) { - varArgs = new Object[ args.length - chunk.numArgs ]; + if( fnDef.isVarArg ) { + if( args.length > fnDef.numArgs ) { + varArgs = new Object[ args.length - fnDef.numArgs ]; for( int i=0; i<varArgs.length; i++ ) { - varArgs[i] = args[chunk.numArgs+i]; + varArgs[i] = args[fnDef.numArgs+i]; } } else { - varArgs = LuanFunction.EMPTY_RTN; + varArgs = LuanFunction.EMPTY; } } - Object[] stack = luan.newFrame(closure,chunk.stackSize,varArgs); - final int n = Math.min(args.length,chunk.numArgs); + Object[] stack = luan.newFrame(closure,fnDef.stackSize,varArgs); + final int n = Math.min(args.length,fnDef.numArgs); for( int i=0; i<n; i++ ) { stack[i] = args[i]; } Object[] returnValues; Closure tailFn; try { - chunk.block.eval(luan); + fnDef.block.eval(luan); } catch(ReturnException e) { } finally { returnValues = luan.returnValues;
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/luan/interp/FnDef.java Wed Feb 27 08:03:51 2013 +0000 @@ -0,0 +1,42 @@ +package luan.interp; + +import luan.LuanException; +import luan.LuanSource; + + +final class FnDef extends CodeImpl implements Expr { + final Stmt block; + final int stackSize; + final int numArgs; + final boolean isVarArg; + final UpValue.Getter[] upValueGetters; + + FnDef(LuanSource.Element se,Stmt block,int stackSize,int numArgs,boolean isVarArg,UpValue.Getter[] upValueGetters) { + super(se); + this.block = block; + this.stackSize = stackSize; + this.numArgs = numArgs; + this.isVarArg = isVarArg; + this.upValueGetters = upValueGetters; + fixReturns(block); + } + + private static void fixReturns(Stmt stmt) { + if( stmt instanceof ReturnStmt ) { + ReturnStmt rs = (ReturnStmt)stmt; + rs.throwReturnException = false; + } else if( stmt instanceof Block ) { + Block b = (Block)stmt; + fixReturns( b.stmts[b.stmts.length-1] ); + } else if( stmt instanceof IfStmt ) { + IfStmt is = (IfStmt)stmt; + fixReturns( is.thenStmt ); + fixReturns( is.elseStmt ); + } + } + + @Override public Object eval(LuanStateImpl luan) throws LuanException { + return new Closure(luan,this); + } + +}
--- a/src/luan/interp/LuanCompiler.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/interp/LuanCompiler.java Wed Feb 27 08:03:51 2013 +0000 @@ -10,20 +10,21 @@ import luan.LuanException; import luan.LuanSource; import luan.LuanElement; +import luan.LuanTable; public final class LuanCompiler { private LuanCompiler() {} // never - public static LuanFunction compile(LuanState luan,LuanSource src) throws LuanException { - LuanParser parser = Parboiled.createParser(LuanParser.class); - parser.source = src; + public static LuanFunction compile(LuanState luan,LuanSource src,LuanTable env) throws LuanException { + UpValue.Getter envGetter = env!=null ? new UpValue.ValueGetter(env) : new UpValue.EnvGetter(); + LuanParser parser = Parboiled.createParser(LuanParser.class,src,envGetter); ParsingResult<?> result = new ReportingParseRunner(parser.Target()).run(src.text); // ParsingResult<?> result = new TracingParseRunner(parser.Target()).run(src); if( result.hasErrors() ) throw new LuanException( luan, LuanElement.COMPILER, ErrorUtils.printParseErrors(result) ); - Chunk chunk = (Chunk)result.resultValue; - return new Closure((LuanStateImpl)luan,chunk); + FnDef fnDef = (FnDef)result.resultValue; + return new Closure((LuanStateImpl)luan,fnDef); } public static LuanState newLuanState() {
--- a/src/luan/interp/LuanParser.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/interp/LuanParser.java Wed Feb 27 08:03:51 2013 +0000 @@ -24,14 +24,6 @@ class LuanParser extends BaseParser<Object> { - LuanSource source; - - LuanSource.Element se(int start) { - return new LuanSource.Element(source,start,currentIndex()); - } - - static final String _ENV = "_ENV"; - static final class Frame { final Frame parent; final List<String> symbols = new ArrayList<String>(); @@ -41,14 +33,16 @@ final List<String> upValueSymbols = new ArrayList<String>(); final List<UpValue.Getter> upValueGetters = new ArrayList<UpValue.Getter>(); - Frame() { + Frame(UpValue.Getter envGetter) { this.parent = null; upValueSymbols.add(_ENV); - upValueGetters.add(UpValue.globalGetter); + upValueGetters.add(envGetter); } Frame(Frame parent) { this.parent = parent; + if( upValueIndex(_ENV) != 0 ) + throw new RuntimeException(); } int stackIndex(String name) { @@ -82,10 +76,22 @@ } } + static final String _ENV = "_ENV"; static final UpValue.Getter[] NO_UP_VALUE_GETTERS = new UpValue.Getter[0]; +// UpValue.Getter envGetter = new UpValue.EnvGetter(); + final LuanSource source; + Frame frame; int nEquals; - Frame frame = new Frame(); + + LuanParser(LuanSource source,UpValue.Getter envGetter) { + this.source = source; + this.frame = new Frame(envGetter); + } + + LuanSource.Element se(int start) { + return new LuanSource.Element(source,start,currentIndex()); + } boolean nEquals(int n) { nEquals = n; @@ -140,8 +146,8 @@ return true; } - Chunk newChunk(int start) { - return new Chunk( se(start), (Stmt)pop(), frame.stackSize, symbolsSize(), frame.isVarArg, frame.upValueGetters.toArray(NO_UP_VALUE_GETTERS) ); + FnDef newFnDef(int start) { + return new FnDef( se(start), (Stmt)pop(), frame.stackSize, symbolsSize(), frame.isVarArg, frame.upValueGetters.toArray(NO_UP_VALUE_GETTERS) ); } Rule Target() { @@ -153,13 +159,13 @@ Sequence( ExpList(false), push( new ReturnStmt( se(start.get()), (Expressions)pop() ) ), - push( newChunk(start.get()) ), + push( newFnDef(start.get()) ), EOI ), Sequence( action( frame.isVarArg = true ), Block(), - push( newChunk(start.get()) ), + push( newFnDef(start.get()) ), EOI ) ) @@ -648,7 +654,7 @@ ) ), ')', Spaces(inParens), Block(), Keyword("end",inParens), - push( newChunk(start.get()) ), + push( newFnDef(start.get()) ), action( frame = frame.parent ) ); }
--- a/src/luan/interp/LuanStateImpl.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/interp/LuanStateImpl.java Wed Feb 27 08:03:51 2013 +0000 @@ -2,6 +2,8 @@ import java.util.List; import java.util.ArrayList; +import java.util.Map; +import java.util.HashMap; import luan.Luan; import luan.LuanState; import luan.LuanTable; @@ -53,10 +55,11 @@ } private Frame frame = null; - Object[] returnValues = LuanFunction.EMPTY_RTN; + Object[] returnValues = LuanFunction.EMPTY; Closure tailFn; + Map<UpValue.EnvGetter,UpValue> envs = new HashMap<UpValue.EnvGetter,UpValue>(); - public LuanStateImpl() {} + LuanStateImpl() {} private LuanStateImpl(LuanStateImpl luan) { super(luan); @@ -68,6 +71,15 @@ return new LuanStateImpl(this); } + @Override public void deepenClone(LuanState clone,DeepCloner cloner) { + super.deepenClone(clone,cloner); + LuanStateImpl cloneImpl = (LuanStateImpl)clone; + cloneImpl.envs = new HashMap<UpValue.EnvGetter,UpValue>(); + for( Map.Entry<UpValue.EnvGetter,UpValue> entry : envs.entrySet() ) { + cloneImpl.envs.put( entry.getKey(), cloner.deepClone(entry.getValue()) ); + } + } + // returns stack Object[] newFrame(Closure closure, int stackSize, Object[] varArgs) { frame = new Frame(frame,closure,stackSize,varArgs); @@ -75,7 +87,7 @@ } void popFrame() { - returnValues = LuanFunction.EMPTY_RTN; + returnValues = LuanFunction.EMPTY; tailFn = null; frame = frame.previousFrame; } @@ -104,6 +116,22 @@ return frame.getUpValue(index); } + UpValue getUpValue(UpValue.EnvGetter getter) throws LuanException { + UpValue uv = envs.get(getter); + if( uv == null ) { + LuanTable env = newEnvironment(); + uv = new UpValue(env); + envs.put(getter,uv); + } + return uv; + } + + @Override public LuanTable currentEnvironment() { + if( frame==null ) + return null; + return (LuanTable)frame.closure.upValues()[0].get(); + } + final Object arithmetic(LuanElement el,String op,Object o1,Object o2) throws LuanException { LuanFunction fn = getBinHandler(el,op,o1,o2);
--- a/src/luan/interp/UpValue.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/interp/UpValue.java Wed Feb 27 08:03:51 2013 +0000 @@ -2,6 +2,7 @@ import luan.DeepCloner; import luan.DeepCloneable; +import luan.LuanException; final class UpValue implements DeepCloneable<UpValue> { @@ -55,7 +56,7 @@ } static interface Getter { - public UpValue get(LuanStateImpl luan); + public UpValue get(LuanStateImpl luan) throws LuanException; } static final class StackGetter implements Getter { @@ -82,10 +83,23 @@ } } - static final Getter globalGetter = new Getter() { + static final class EnvGetter implements Getter { + + public UpValue get(LuanStateImpl luan) throws LuanException { + return luan.getUpValue(this); + } + } + + static final class ValueGetter implements Getter { + private final UpValue upValue; + + ValueGetter(Object value) { + this.upValue = new UpValue(value); + } + public UpValue get(LuanStateImpl luan) { - return new UpValue(luan.global()); + return upValue; } - }; + } }
--- a/src/luan/lib/BasicLib.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/lib/BasicLib.java Wed Feb 27 08:03:51 2013 +0000 @@ -10,6 +10,7 @@ import luan.LuanState; import luan.LuanTable; import luan.LuanFunction; +import luan.LuanLoader; import luan.LuanJavaFunction; import luan.LuanException; import luan.LuanSource; @@ -21,10 +22,11 @@ public static final String NAME = "basic"; - public static final LuanFunction LOADER = new LuanFunction() { - public Object[] call(LuanState luan,Object[] args) throws LuanException { - LuanTable global = luan.global(); - global.put( "_G", global ); + public static final LuanLoader LOADER = new LuanLoader() { + @Override protected void load(LuanState luan) { + LuanTable module = new LuanTable(); + LuanTable global = new LuanTable(); + module.put( LuanState._G, global ); try { global.put( "assert", new LuanJavaFunction(BasicLib.class.getMethod("assert_",LuanState.class,Object.class,String.class),null) ); add( global, "assert_boolean", LuanState.class, Boolean.TYPE ); @@ -32,13 +34,13 @@ add( global, "assert_number", LuanState.class, Number.class ); add( global, "assert_string", LuanState.class, String.class ); add( global, "assert_table", LuanState.class, LuanTable.class ); - add( global, "do_file", LuanState.class, String.class ); + add( global, "do_file", LuanState.class, String.class, LuanTable.class ); add( global, "error", LuanState.class, Object.class ); add( global, "get_metatable", LuanState.class, Object.class ); - add( global, "ipairs", LuanTable.class ); - add( global, "load", LuanState.class, String.class, String.class ); - add( global, "load_file", LuanState.class, String.class ); - add( global, "pairs", LuanTable.class ); + add( global, "ipairs", LuanState.class, LuanTable.class ); + add( global, "load", LuanState.class, String.class, String.class, LuanTable.class ); + add( global, "load_file", LuanState.class, String.class, LuanTable.class ); + add( global, "pairs", LuanState.class, LuanTable.class ); add( global, "print", LuanState.class, new Object[0].getClass() ); add( global, "raw_equal", Object.class, Object.class ); add( global, "raw_get", LuanTable.class, Object.class ); @@ -49,12 +51,10 @@ add( global, "to_string", LuanState.class, Object.class ); add( global, "type", Object.class ); global.put( "_VERSION", Luan.version ); - - add( global, "make_standard", LuanState.class ); } catch(NoSuchMethodException e) { throw new RuntimeException(e); } - return LuanFunction.EMPTY_RTN; + luan.loaded().put(NAME,module); } }; @@ -62,20 +62,6 @@ t.put( method, new LuanJavaFunction(BasicLib.class.getMethod(method,parameterTypes),null) ); } - public static void make_standard(LuanState luan) { - LuanTable global = luan.global(); - global.put( "dofile", global.get("do_file") ); - global.put( "getmetatable", global.get("get_metatable") ); - global.put( "loadfile", global.get("load_file") ); - global.put( "rawequal", global.get("raw_equal") ); - global.put( "rawget", global.get("raw_get") ); - global.put( "rawlen", global.get("raw_len") ); - global.put( "rawset", global.get("raw_set") ); - global.put( "setmetatable", global.get("set_metatable") ); - global.put( "tonumber", global.get("to_number") ); - global.put( "tostring", global.get("to_string") ); - } - public static void print(LuanState luan,Object... args) throws LuanException { for( int i=0; i<args.length; i++ ) { if( i > 0 ) @@ -89,22 +75,22 @@ return Luan.type(obj); } - public static LuanFunction load(LuanState luan,String text,String sourceName) throws LuanException { - return LuanCompiler.compile(luan,new LuanSource(sourceName,text)); + public static LuanFunction load(LuanState luan,String text,String sourceName,LuanTable env) throws LuanException { + return LuanCompiler.compile(luan,new LuanSource(sourceName,text),env); } - public static LuanFunction load_file(LuanState luan,String fileName) throws LuanException { + public static LuanFunction load_file(LuanState luan,String fileName,LuanTable env) throws LuanException { try { String src = fileName==null ? Utils.readAll(new InputStreamReader(System.in)) : Utils.read(new File(fileName)); - return load(luan,src,fileName); + return load(luan,src,fileName,env); } catch(IOException e) { throw new LuanException(luan,LuanElement.JAVA,e); } } - public static Object[] do_file(LuanState luan,String fileName) throws LuanException { - LuanFunction fn = load_file(luan,fileName); + public static Object[] do_file(LuanState luan,String fileName,LuanTable env) throws LuanException { + LuanFunction fn = load_file(luan,fileName,env); return luan.call(fn,LuanElement.JAVA,null); } @@ -112,18 +98,20 @@ return new LuanFunction() { public Object[] call(LuanState luan,Object[] args) { if( !iter.hasNext() ) - return LuanFunction.EMPTY_RTN; + return LuanFunction.EMPTY; Map.Entry<Object,Object> entry = iter.next(); return new Object[]{entry.getKey(),entry.getValue()}; } }; } - public static LuanFunction pairs(LuanTable t) { + public static LuanFunction pairs(LuanState luan,LuanTable t) throws LuanException { + Utils.checkNotNull(luan,t,"table"); return pairs( t.iterator() ); } - public static LuanFunction ipairs(LuanTable t) { + public static LuanFunction ipairs(LuanState luan,LuanTable t) throws LuanException { + Utils.checkNotNull(luan,t,"table"); return pairs( t.listIterator() ); } @@ -181,23 +169,18 @@ throw new LuanException( luan, LuanElement.JAVA, msg ); } - private static void checkNotNull(LuanState luan,Object v,String expected) throws LuanException { - if( v == null ) - throw new LuanException(luan,LuanElement.JAVA,"bad argument #1 ("+expected+" expected, got nil)"); - } - public static String assert_string(LuanState luan,String v) throws LuanException { - checkNotNull(luan,v,"string"); + Utils.checkNotNull(luan,v,"string"); return v; } public static Number assert_number(LuanState luan,Number v) throws LuanException { - checkNotNull(luan,v,"number"); + Utils.checkNotNull(luan,v,"number"); return v; } public static LuanTable assert_table(LuanState luan,LuanTable v) throws LuanException { - checkNotNull(luan,v,"table"); + Utils.checkNotNull(luan,v,"table"); return v; }
--- a/src/luan/lib/HtmlLib.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/lib/HtmlLib.java Wed Feb 27 08:03:51 2013 +0000 @@ -3,6 +3,7 @@ import luan.LuanState; import luan.LuanTable; import luan.LuanFunction; +import luan.LuanLoader; import luan.LuanJavaFunction; @@ -10,16 +11,15 @@ public static final String NAME = "html"; - public static final LuanFunction LOADER = new LuanFunction() { - public Object[] call(LuanState luan,Object[] args) { + public static final LuanLoader LOADER = new LuanLoader() { + @Override protected void load(LuanState luan) { LuanTable module = new LuanTable(); - LuanTable global = luan.global(); try { add( module, "encode", String.class ); } catch(NoSuchMethodException e) { throw new RuntimeException(e); } - return new Object[]{module}; + luan.loaded().put(NAME,module); } };
--- a/src/luan/lib/HttpLib.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/lib/HttpLib.java Wed Feb 27 08:03:51 2013 +0000 @@ -18,17 +18,23 @@ public final class HttpLib { public static final String NAME = "http"; - public static final String FN_NAME = "serve_http"; + public static final String FN_NAME = "http.server"; + + public static void load(LuanState luan) throws LuanException { + PackageLib.require(luan,NAME); + Object fn = luan.get(HttpLib.FN_NAME); + if( !(fn instanceof LuanFunction) ) + throw new LuanException( luan, LuanElement.JAVA, "function '"+HttpLib.FN_NAME+"' not defined" ); + } public static void service(LuanState luan,HttpServletRequest request,HttpServletResponse response) throws LuanException, IOException { - LuanFunction fn = (LuanFunction)luan.global().get(FN_NAME); + LuanFunction fn = (LuanFunction)luan.get(FN_NAME); ServletOutputStream sout = response.getOutputStream(); luan.out = new PrintStream(sout); - LuanTable module = new LuanTable(); - luan.global().put(NAME,module); + LuanTable module = (LuanTable)luan.loaded().get(NAME); LuanTable parameters = new LuanTable(); LuanTable parameter_lists = new LuanTable();
--- a/src/luan/lib/JavaLib.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/lib/JavaLib.java Wed Feb 27 08:03:51 2013 +0000 @@ -21,6 +21,7 @@ import luan.MetatableGetter; import luan.LuanException; import luan.LuanFunction; +import luan.LuanLoader; import luan.LuanJavaFunction; import luan.LuanElement; @@ -29,11 +30,12 @@ public static final String NAME = "java"; - public static final LuanFunction LOADER = new LuanFunction() { - public Object[] call(LuanState luan,Object[] args) throws LuanException { + public static final LuanLoader LOADER = new LuanLoader() { + @Override protected void load(LuanState luan) { luan.addMetatableGetter(mg); LuanTable module = new LuanTable(); - LuanTable global = luan.global(); + LuanTable global = new LuanTable(); + module.put( LuanState._G, global ); try { global.put( "import", new LuanJavaFunction(JavaLib.class.getMethod("importClass",LuanState.class,String.class),null) ); module.put( "class", new LuanJavaFunction(JavaLib.class.getMethod("getClass",LuanState.class,String.class),null) ); @@ -41,7 +43,7 @@ } catch(NoSuchMethodException e) { throw new RuntimeException(e); } - return new Object[]{module}; + luan.loaded().put(NAME,module); } }; @@ -298,7 +300,7 @@ } public static void importClass(LuanState luan,String name) throws LuanException { - luan.global().put( name.substring(name.lastIndexOf('.')+1), getClass(luan,name) ); + luan.currentEnvironment().put( name.substring(name.lastIndexOf('.')+1), getClass(luan,name) ); } static class AmbiguousJavaFunction extends LuanFunction {
--- a/src/luan/lib/MathLib.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/lib/MathLib.java Wed Feb 27 08:03:51 2013 +0000 @@ -3,6 +3,7 @@ import luan.LuanState; import luan.LuanTable; import luan.LuanFunction; +import luan.LuanLoader; import luan.LuanJavaFunction; @@ -10,16 +11,15 @@ public static final String NAME = "math"; - public static final LuanFunction LOADER = new LuanFunction() { - public Object[] call(LuanState luan,Object[] args) { + public static final LuanLoader LOADER = new LuanLoader() { + @Override protected void load(LuanState luan) { LuanTable module = new LuanTable(); - LuanTable global = luan.global(); try { add( module, "floor", Double.TYPE ); } catch(NoSuchMethodException e) { throw new RuntimeException(e); } - return new Object[]{module}; + luan.loaded().put(NAME,module); } };
--- a/src/luan/lib/PackageLib.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/lib/PackageLib.java Wed Feb 27 08:03:51 2013 +0000 @@ -4,10 +4,12 @@ import java.io.IOException; import java.net.URL; import java.util.Arrays; +import java.util.Collections; import luan.Luan; import luan.LuanState; import luan.LuanTable; import luan.LuanFunction; +import luan.LuanLoader; import luan.LuanJavaFunction; import luan.LuanElement; import luan.LuanException; @@ -17,22 +19,23 @@ public static final String NAME = "package"; - public static final LuanFunction LOADER = new LuanFunction() { - public Object[] call(LuanState luan,Object[] args) throws LuanException { - LuanTable global = luan.global(); + public static final LuanLoader LOADER = new LuanLoader() { + @Override protected void load(LuanState luan) { LuanTable module = new LuanTable(); + LuanTable global = new LuanTable(); + module.put( LuanState._G, global ); module.put("loaded",luan.loaded()); module.put("preload",luan.preload()); -// module.put("path","?.lua"); + module.put("path","?.lua"); try { add( global, "require", LuanState.class, String.class ); - add( module, "module", LuanState.class, String.class ); + add( global, "module", LuanState.class, String.class ); add( module, "search_path", String.class, String.class ); } catch(NoSuchMethodException e) { throw new RuntimeException(e); } module.put("searchers",new LuanTable(Arrays.<Object>asList(preloadSearcher,fileSearcher,javaFileSearcher))); - return new Object[]{module}; + luan.loaded().put(NAME,module); } }; @@ -40,32 +43,39 @@ t.put( method, new LuanJavaFunction(PackageLib.class.getMethod(method,parameterTypes),null) ); } - public static void require(LuanState luan,String modName) throws LuanException { - Object mod = module(luan,modName); - if( mod instanceof LuanTable ) - luan.global().put(modName,mod); + public static void module(LuanState luan,String modName) throws LuanException { + LuanTable module = new LuanTable(); + luan.currentEnvironment().put(modName,module); + luan.loaded().put(modName,module); } - public static Object module(LuanState luan,String modName) throws LuanException { - Object mod = luan.loaded().get(modName); + public static void require(LuanState luan,String modName) throws LuanException { + require(luan,modName,luan.currentEnvironment()); + } + + public static void require(LuanState luan,String modName,LuanTable env) throws LuanException { + LuanTable mod = (LuanTable)luan.loaded().get(modName); if( mod == null ) { LuanTable searchers = (LuanTable)luan.get("package.searchers"); + if( searchers == null ) + searchers = new LuanTable(Collections.<Object>singletonList(preloadSearcher)); for( Object s : searchers.asList() ) { LuanFunction searcher = (LuanFunction)s; - Object[] a = luan.call(searcher,LuanElement.JAVA,"searcher",modName); + Object[] a = luan.call(searcher,LuanElement.JAVA,"<searcher>",modName); if( a.length >= 1 && a[0] instanceof LuanFunction ) { LuanFunction loader = (LuanFunction)a[0]; - Object extra = a.length >= 2 ? a[1] : null; - mod = Luan.first(luan.call(loader,LuanElement.JAVA,"loader",modName,extra)); - if( mod == null ) - mod = true; - luan.loaded().put(modName,mod); + luan.call(loader,LuanElement.JAVA,"<loader>"); + mod = (LuanTable)luan.loaded().get(modName); + if( mod==null ) + throw new LuanException( luan, LuanElement.JAVA, "module '"+modName+"' didn't define its module" ); + break; } } if( mod == null ) throw new LuanException( luan, LuanElement.JAVA, "module '"+modName+"' not found" ); } - return mod; + if( env != null ) + env.put(modName,mod); } public static String search_path(String name,String path) { @@ -77,12 +87,16 @@ return null; } - private static final LuanFunction fileLoader = new LuanFunction() { - public Object[] call(LuanState luan,Object[] args) throws LuanException { - String modName = (String)args[0]; - String fileName = (String)args[1]; - LuanFunction fn = BasicLib.load_file(luan,fileName); - return luan.call(fn,LuanElement.JAVA,modName,args); + private static final class FileLoader extends LuanLoader { + private final String fileName; + + FileLoader(String fileName) { + this.fileName = fileName; + } + + @Override protected void load(LuanState luan) throws LuanException { + LuanFunction fn = BasicLib.load_file(luan,fileName,null); + fn.call(luan,EMPTY); } }; @@ -91,9 +105,9 @@ String modName = (String)args[0]; String path = (String)luan.get("package.path"); if( path==null ) - return LuanFunction.EMPTY_RTN; + return LuanFunction.EMPTY; String file = search_path(modName,path); - return file==null ? LuanFunction.EMPTY_RTN : new Object[]{fileLoader,file}; + return file==null ? LuanFunction.EMPTY : new Object[]{new FileLoader(file)}; } }; @@ -108,14 +122,18 @@ - private static final LuanFunction javaFileLoader = new LuanFunction() { - public Object[] call(LuanState luan,Object[] args) throws LuanException { + private static final class JavaFileLoader extends LuanLoader { + private final URL url; + + JavaFileLoader(URL url) { + this.url = url; + } + + @Override protected void load(LuanState luan) throws LuanException { try { - String modName = (String)args[0]; - URL url = (URL)args[1]; String src = Utils.read(url); - LuanFunction fn = BasicLib.load(luan,src,url.toString()); - return luan.call(fn,LuanElement.JAVA,modName,args); + LuanFunction fn = BasicLib.load(luan,src,url.toString(),null); + fn.call(luan,EMPTY); } catch(IOException e) { throw new LuanException(luan,LuanElement.JAVA,e); } @@ -127,15 +145,15 @@ String modName = (String)args[0]; String path = (String)luan.get("package.jpath"); if( path==null ) - return LuanFunction.EMPTY_RTN; + return LuanFunction.EMPTY; for( String s : path.split(";") ) { String file = s.replaceAll("\\?",modName); URL url = ClassLoader.getSystemResource(file); if( url != null ) { - return new Object[]{javaFileLoader,url}; + return new Object[]{new JavaFileLoader(url)}; } } - return LuanFunction.EMPTY_RTN; + return LuanFunction.EMPTY; } };
--- a/src/luan/lib/StringLib.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/lib/StringLib.java Wed Feb 27 08:03:51 2013 +0000 @@ -6,6 +6,7 @@ import luan.LuanState; import luan.LuanTable; import luan.LuanFunction; +import luan.LuanLoader; import luan.LuanJavaFunction; import luan.LuanElement; import luan.LuanException; @@ -15,10 +16,9 @@ public static final String NAME = "string"; - public static final LuanFunction LOADER = new LuanFunction() { - public Object[] call(LuanState luan,Object[] args) throws LuanException { + public static final LuanLoader LOADER = new LuanLoader() { + @Override protected void load(LuanState luan) { LuanTable module = new LuanTable(); - LuanTable global = luan.global(); try { module.put( "byte", new LuanJavaFunction(StringLib.class.getMethod("byte_",String.class,Integer.class,Integer.class),null) ); module.put( "char", new LuanJavaFunction(StringLib.class.getMethod("char_",new byte[0].getClass()),null) ); @@ -36,7 +36,7 @@ } catch(NoSuchMethodException e) { throw new RuntimeException(e); } - return new Object[]{module}; + luan.loaded().put(NAME,module); } }; @@ -134,7 +134,7 @@ return new LuanFunction() { public Object[] call(LuanState luan,Object[] args) { if( !m.find() ) - return LuanFunction.EMPTY_RTN; + return LuanFunction.EMPTY; final int n = m.groupCount(); if( n == 0 ) return new String[]{m.group()};
--- a/src/luan/lib/TableLib.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/lib/TableLib.java Wed Feb 27 08:03:51 2013 +0000 @@ -7,6 +7,7 @@ import luan.LuanState; import luan.LuanTable; import luan.LuanFunction; +import luan.LuanLoader; import luan.LuanJavaFunction; import luan.LuanElement; import luan.LuanException; @@ -17,10 +18,9 @@ public static final String NAME = "table"; - public static final LuanFunction LOADER = new LuanFunction() { - public Object[] call(LuanState luan,Object[] args) throws LuanException { + public static final LuanLoader LOADER = new LuanLoader() { + @Override protected void load(LuanState luan) { LuanTable module = new LuanTable(); - LuanTable global = luan.global(); try { add( module, "concat", LuanState.class, LuanTable.class, String.class, Integer.class, Integer.class ); add( module, "insert", LuanState.class, LuanTable.class, Integer.TYPE, Object.class ); @@ -32,7 +32,7 @@ } catch(NoSuchMethodException e) { throw new RuntimeException(e); } - return new Object[]{module}; + luan.loaded().put(NAME,module); } };
--- a/src/luan/lib/Utils.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/lib/Utils.java Wed Feb 27 08:03:51 2013 +0000 @@ -6,11 +6,19 @@ import java.io.Reader; import java.io.IOException; import java.net.URL; +import luan.LuanState; +import luan.LuanException; +import luan.LuanElement; public final class Utils { private Utils() {} // never + public static void checkNotNull(LuanState luan,Object v,String expected) throws LuanException { + if( v == null ) + throw new LuanException(luan,LuanElement.JAVA,"bad argument #1 ("+expected+" expected, got nil)"); + } + public static String readAll(Reader in) throws IOException {
--- a/src/luan/tools/CmdLine.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/tools/CmdLine.java Wed Feb 27 08:03:51 2013 +0000 @@ -14,7 +14,14 @@ public static void main(String[] args) { LuanState luan = LuanState.newStandard(); - BasicLib.make_standard(luan); + LuanTable env; + try { + env = luan.newEnvironment(); + } catch(LuanException e) { + System.err.println("command line error: "+e.getMessage()); + System.exit(-1); + throw new RuntimeException(); // never + } boolean interactive = false; boolean showVersion = false; int i = 0; @@ -35,7 +42,7 @@ error("'-e' needs argument"); String cmd = args[i]; try { - LuanFunction fn = BasicLib.load(luan,cmd,"(command line)"); + LuanFunction fn = BasicLib.load(luan,cmd,"(command line)",env); luan.call(fn,null,null); } catch(LuanException e) { System.err.println("command line error: "+e.getMessage()); @@ -43,7 +50,7 @@ } } else if( arg.equals("-") ) { try { - BasicLib.do_file(luan,"stdin"); + BasicLib.do_file(luan,"stdin",env); } catch(LuanException e) { System.err.println(e.getMessage()); System.exit(-1); @@ -65,9 +72,9 @@ for( int j=0; j<args.length; j++ ) { argsTable.put( j, args[j] ); } - luan.global().put("arg",argsTable); + env.put("arg",argsTable); try { - LuanFunction fn = BasicLib.load_file(luan,file); + LuanFunction fn = BasicLib.load_file(luan,file,env); luan.call(fn,null,null,varArgs); } catch(LuanException e) { // System.err.println("error: "+e.getMessage()); @@ -76,7 +83,7 @@ } } if( interactive ) - interactive(luan); + interactive(luan,env); } private static void error(String msg) { @@ -93,12 +100,12 @@ System.exit(-1); } - static void interactive(LuanState luan) { + static void interactive(LuanState luan,LuanTable env) { while( true ) { System.out.print("> "); String input = new Scanner(System.in).nextLine(); try { - Object[] rtn = luan.eval(input,"stdin"); + Object[] rtn = luan.eval(input,"stdin",env); if( rtn.length > 0 ) BasicLib.print(luan,rtn); } catch(LuanException e) {
--- a/src/luan/tools/WebRun.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/tools/WebRun.java Wed Feb 27 08:03:51 2013 +0000 @@ -10,6 +10,7 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; import luan.LuanState; +import luan.LuanTable; import luan.LuanException; import luan.lib.HtmlLib; @@ -33,9 +34,10 @@ try { LuanState luan = newLuanState(); luan.out = out; - luan.global().put("request",request); - luan.global().put("response",response); - luan.eval(code,"WebRun"); + LuanTable env = luan.newEnvironment(); + env.put("request",request); + env.put("response",response); + luan.eval(code,"WebRun",env); } catch(LuanException e) { logger.error(null,e); response.reset();
--- a/src/luan/tools/WebServlet.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/tools/WebServlet.java Wed Feb 27 08:03:51 2013 +0000 @@ -23,8 +23,6 @@ public class WebServlet extends HttpServlet { - public static final String HTTP_SERVER = "http_server"; - protected LuanState luanState = null; protected void loadLibs(LuanState luan) throws LuanException { @@ -39,10 +37,7 @@ protected LuanState newLuanState() throws LuanException { LuanState luan = LuanCompiler.newLuanState(); loadLibs(luan); - PackageLib.require(luan,HTTP_SERVER); - Object fn = luan.global().get(HttpLib.FN_NAME); - if( !(fn instanceof LuanFunction) ) - throw new LuanException( luan, LuanElement.JAVA, "function '"+HttpLib.FN_NAME+"' not defined" ); + HttpLib.load(luan); return luan; } @@ -54,7 +49,7 @@ return luanState.deepClone(); } - protected void service(HttpServletRequest request,HttpServletResponse response) + @Override protected void service(HttpServletRequest request,HttpServletResponse response) throws ServletException, IOException { try {
--- a/src/luan/tools/WebShell.java Mon Feb 25 03:53:54 2013 +0000 +++ b/src/luan/tools/WebShell.java Wed Feb 27 08:03:51 2013 +0000 @@ -15,8 +15,8 @@ import org.slf4j.LoggerFactory; import luan.LuanFunction; import luan.LuanState; +import luan.LuanTable; import luan.LuanException; -import luan.interp.LuanCompiler; import luan.lib.BasicLib; import luan.lib.HtmlLib; @@ -28,11 +28,15 @@ return LuanState.newStandard(); } - protected Object[] eval(LuanState luan,String cmd) throws LuanException { - return luan.eval(cmd,"WebShell"); + protected LuanTable newEnvironment(LuanState luan) throws LuanException { + return luan.newEnvironment(); } - protected void service(HttpServletRequest request,HttpServletResponse response) + protected Object[] eval(LuanState luan,String cmd,LuanTable env) throws LuanException { + return luan.eval(cmd,"WebShell",env); + } + + @Override protected void service(HttpServletRequest request,HttpServletResponse response) throws ServletException, IOException { PrintWriter out = response.getWriter(); @@ -58,10 +62,15 @@ luan = newLuanState(); session.putValue("luan",luan); } + LuanTable env = (LuanTable)session.getValue("env"); + if( env==null ) { + env = newEnvironment(luan); + session.putValue("env",env); + } luan.out = new PrintStream(history); - luan.global().put("request",request); - luan.global().put("response",response); - Object[] result = eval(luan,cmd); + env.put("request",request); + env.put("response",response); + Object[] result = eval(luan,cmd,env); if( result.length > 0 ) { for( int i=0; i<result.length; i++ ) { if( i > 0 )