Mercurial Hosting > luan
view core/src/luan/modules/PackageLuan.java @ 244:0a8e6fdb62f0
fix Math.random()
git-svn-id: https://luan-java.googlecode.com/svn/trunk@245 21e917c8-12df-6dd8-5cb6-c86387c605b9
author | fschmidt@gmail.com <fschmidt@gmail.com@21e917c8-12df-6dd8-5cb6-c86387c605b9> |
---|---|
date | Thu, 09 Oct 2014 22:44:07 +0000 |
parents | 2a54cb7d1cf4 |
children | 4dfa86dbca45 |
line wrap: on
line source
package luan.modules; import java.io.IOException; import java.util.Arrays; import java.util.Collections; import java.util.List; import luan.Luan; import luan.LuanState; import luan.LuanTable; import luan.LuanFunction; import luan.LuanJavaFunction; import luan.LuanElement; import luan.LuanException; public final class PackageLuan { private static final String jpath = "luan.modules.?Luan.LOADER"; public static final LuanFunction LOADER = new LuanFunction() { @Override public Object call(LuanState luan,Object[] args) { LuanTable module = Luan.newTable(); module.put( "loaded", loaded(luan) ); module.put( "preload", Luan.newTable() ); module.put( "path", "?.luan;java:luan/modules/?.luan" ); module.put( "jpath", jpath ); try { module.put("require",requireFn); add( module, "load", LuanState.class, String.class ); add( module, "load_lib", LuanState.class, String.class ); add( module, "search_path", String.class, String.class ); add( module, "search", LuanState.class, String.class ); } catch(NoSuchMethodException e) { throw new RuntimeException(e); } module.put( "searchers", searchers(luan) ); return module; } }; public static final LuanFunction requireFn; static { try { requireFn = new LuanJavaFunction(PackageLuan.class.getMethod("require",LuanState.class,String.class),null); } catch(NoSuchMethodException e) { throw new RuntimeException(e); } } private static void add(LuanTable t,String method,Class<?>... parameterTypes) throws NoSuchMethodException { t.put( method, new LuanJavaFunction(PackageLuan.class.getMethod(method,parameterTypes),null) ); } public static LuanTable loaded(LuanState luan) { return luan.registryTable("Package.loaded"); } private static LuanTable blocked(LuanState luan) { return luan.registryTable("Package.blocked"); } private static Object pkg(LuanState luan,String key) { LuanTable t = (LuanTable)loaded(luan).get("Package"); return t==null ? null : t.get(key); } private static LuanTable searchers(LuanState luan) { String key = "Package.searchers"; LuanTable tbl = (LuanTable)luan.registry().get(key); if( tbl == null ) { tbl = Luan.newTable(); tbl.add(preloadSearcher); tbl.add(fileSearcher); tbl.add(javaSearcher); tbl.add(JavaLuan.javaSearcher); luan.registry().put(key,tbl); } return tbl; } public static Object require(LuanState luan,String modName) throws LuanException { Object mod = load(luan,modName); if( mod==null ) throw luan.exception( "module '"+modName+"' not found" ); return mod; } public static Object load(LuanState luan,String modName) throws LuanException { LuanTable loaded = loaded(luan); Object mod = loaded.get(modName); if( mod == null ) { Object[] a = search(luan,modName); if( a == null ) return null; LuanFunction loader = (LuanFunction)a[0]; a[0] = modName; mod = Luan.first(luan.call(loader,"<require \""+modName+"\">",a)); if( mod != null ) { loaded.put(modName,mod); } else { mod = loaded.get(modName); if( mod==null ) { mod = true; loaded.put(modName,mod); } } } return mod; } public static Object[] search(LuanState luan,String modName) throws LuanException { for( Object s : searchers(luan).asList() ) { LuanFunction searcher = (LuanFunction)s; Object[] a = Luan.array(luan.call(searcher,"<searcher>",new Object[]{modName})); if( a.length >= 1 && a[0] instanceof LuanFunction ) return a; } return null; } public static final LuanFunction preloadSearcher = new LuanFunction() { @Override public Object call(LuanState luan,Object[] args) { String modName = (String)args[0]; LuanTable preload = (LuanTable)pkg(luan,"preload"); return preload==null ? LuanFunction.NOTHING : preload.get(modName); } }; public static String search_path(String name,String path) { for( String s : path.split(";") ) { String file = s.replaceAll("\\?",name); if( Utils.exists(file) ) return file; } return null; } public static final LuanFunction fileLoader = new LuanFunction() { @Override public Object call(LuanState luan,Object[] args) throws LuanException { String fileName = (String)args[1]; LuanFunction fn = BasicLuan.load_file(luan,fileName); return fn.call(luan,args); } }; public static final LuanFunction fileSearcher = new LuanFunction() { @Override public Object[] call(LuanState luan,Object[] args) { String modName = (String)args[0]; String path = (String)pkg(luan,"path"); if( path==null ) return LuanFunction.NOTHING; String file = search_path(modName,path); return file==null ? LuanFunction.NOTHING : new Object[]{fileLoader,file}; } }; public static final LuanFunction javaLoader = new LuanFunction() { @Override public Object call(LuanState luan,Object[] args) throws LuanException { try { String objName = (String)args[1]; LuanFunction fn = load_lib(luan,objName); return fn.call(luan,args); } catch(ClassNotFoundException e) { throw new RuntimeException(e); } catch(NoSuchFieldException e) { throw new RuntimeException(e); } catch(IllegalAccessException e) { throw new RuntimeException(e); } } }; public static final LuanFunction javaSearcher = new LuanFunction() { @Override public Object[] call(LuanState luan,Object[] args) throws LuanException { String modName = (String)args[0]; modName = modName.replace('/','.'); String path = (String)pkg(luan,"jpath"); if( path==null ) path = jpath; for( String s : path.split(";") ) { String objName = s.replaceAll("\\?",modName); try { load_lib(luan,objName); // throws exception if not found return new Object[]{javaLoader,objName}; } catch(ClassNotFoundException e) { } catch(NoSuchFieldException e) { } catch(IllegalAccessException e) { } } return LuanFunction.NOTHING; } }; public static void block(LuanState luan,String key) { blocked(luan).put(key,true); } public static boolean is_blocked(LuanState luan,String key) { return blocked(luan).get(key) != null; } public static LuanFunction load_lib(LuanState luan,String path) throws ClassNotFoundException, NoSuchFieldException, IllegalAccessException, LuanException { int i = path.lastIndexOf('.'); String clsPath = path.substring(0,i); String fld = path.substring(i+1); Class cls = Class.forName(clsPath); return (LuanFunction)cls.getField(fld).get(null); } }