view core/src/luan/impl/LuanCompiler.java @ 195:24ede40ee0aa

make MetatableGetter DeepCloneable, scoped, and secure git-svn-id: https://luan-java.googlecode.com/svn/trunk@196 21e917c8-12df-6dd8-5cb6-c86387c605b9
author fschmidt@gmail.com <fschmidt@gmail.com@21e917c8-12df-6dd8-5cb6-c86387c605b9>
date Thu, 03 Jul 2014 08:19:48 +0000
parents 08df375e2e5f
children 5ba136769034
line wrap: on
line source

package luan.impl;

import luan.LuanFunction;
import luan.LuanState;
import luan.LuanException;
import luan.LuanSource;
import luan.LuanElement;
import luan.LuanTable;
import java.util.Map;


public final class LuanCompiler {
	private LuanCompiler() {}  // never

	public static LuanFunction compile(LuanState luan,LuanSource src,LuanTable env,boolean allowExpr) throws LuanException {
		boolean passedEnv = env != null;
		if( !passedEnv )
			env = new LuanTable();
		UpValue.Getter envGetter = new UpValue.ValueGetter(env);
		LuanParser parser = new LuanParser(src,envGetter);
		for( Map.Entry<Object,Object> entry : luan.global() ) {
			Object key = entry.getKey();
			if( key instanceof String )
				parser.addVar( (String)key, entry.getValue() );
		}
		FnDef fnDef = parse(luan,parser,allowExpr);
		final LuanStateImpl luanImpl = (LuanStateImpl)luan;
		MtGetterLink mtGetterLink = (MtGetterLink)env.get("_MTG");
		final Closure c = new Closure(luanImpl,fnDef,mtGetterLink);
		final LuanTable ENV = env;
		if( passedEnv ) {
			return new LuanFunction() {
				@Override public Object call(LuanState luan,Object[] args) throws LuanException {
					Object rtn = c.call(luan,args);
					ENV.put("_MTG",luanImpl.mtGetterLink);
					return rtn;
				}
			};
		} else {
			return new LuanFunction() {
				@Override public Object call(LuanState luan,Object[] args) throws LuanException {
					Object rtn = c.call(luan,args);
					if( rtn instanceof Object[] && ((Object[])rtn).length==0 )
						rtn = ENV;
					return rtn;
				}
			};
		}
	}

	private static FnDef parse(LuanState luan,LuanParser parser,boolean allowExpr) throws LuanException {
		try {
			if( allowExpr ) {
				FnDef fnDef = parser.Expression();
				if( fnDef != null )
					return fnDef;
			}
			return parser.RequiredModule();
		} catch(ParseException e) {
//e.printStackTrace();
			LuanElement le = new LuanSource.CompilerElement(parser.source);
			throw luan.bit(le).exception( e.getFancyMessage() );
		}
	}

	public static LuanState newLuanState() {
		return new LuanStateImpl();
	}
}