/* * Copyright (C) 2015 The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * http://www.apache.org/licenses/LICENSE-2.0 * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.databinding.tool.reflection.java; import com.google.common.base.Splitter; import com.google.common.base.Strings; import org.apache.commons.io.FileUtils; import android.databinding.tool.reflection.ModelAnalyzer; import android.databinding.tool.reflection.ModelClass; import android.databinding.tool.reflection.SdkUtil; import android.databinding.tool.reflection.TypeUtil; import android.databinding.tool.util.L; import java.io.File; import java.io.IOException; import java.net.MalformedURLException; import java.net.URL; import java.net.URLClassLoader; import java.util.HashMap; import java.util.List; import java.util.Map; public class JavaAnalyzer extends ModelAnalyzer { public static final Map PRIMITIVE_TYPES; static { PRIMITIVE_TYPES = new HashMap(); PRIMITIVE_TYPES.put("boolean", boolean.class); PRIMITIVE_TYPES.put("byte", byte.class); PRIMITIVE_TYPES.put("short", short.class); PRIMITIVE_TYPES.put("char", char.class); PRIMITIVE_TYPES.put("int", int.class); PRIMITIVE_TYPES.put("long", long.class); PRIMITIVE_TYPES.put("float", float.class); PRIMITIVE_TYPES.put("double", double.class); } private HashMap mClassCache = new HashMap(); private final ClassLoader mClassLoader; public JavaAnalyzer(ClassLoader classLoader) { setInstance(this); mClassLoader = classLoader; } @Override public JavaClass loadPrimitive(String className) { Class clazz = PRIMITIVE_TYPES.get(className); if (clazz == null) { return null; } else { return new JavaClass(clazz); } } @Override protected ModelClass[] getObservableFieldTypes() { return new ModelClass[0]; } @Override public ModelClass findClass(String className, Map imports) { // TODO handle imports JavaClass loaded = mClassCache.get(className); if (loaded != null) { return loaded; } L.d("trying to load class %s from %s", className, mClassLoader.toString()); loaded = loadPrimitive(className); if (loaded == null) { try { if (className.startsWith("[") && className.contains("L")) { int indexOfL = className.indexOf('L'); JavaClass baseClass = (JavaClass) findClass( className.substring(indexOfL + 1, className.length() - 1), null); String realClassName = className.substring(0, indexOfL + 1) + baseClass.mClass.getCanonicalName() + ';'; loaded = new JavaClass(Class.forName(realClassName, false, mClassLoader)); mClassCache.put(className, loaded); } else { loaded = loadRecursively(className); mClassCache.put(className, loaded); } } catch (Throwable t) { // L.e(t, "cannot load class " + className); } } // expr visitor may call this to resolve statics. Sometimes, it is OK not to find a class. if (loaded == null) { return null; } L.d("loaded class %s", loaded.mClass.getCanonicalName()); return loaded; } @Override public ModelClass findClass(Class classType) { return new JavaClass(classType); } @Override public TypeUtil createTypeUtil() { return new JavaTypeUtil(); } private JavaClass loadRecursively(String className) throws ClassNotFoundException { try { L.d("recursively checking %s", className); return new JavaClass(mClassLoader.loadClass(className)); } catch (ClassNotFoundException ex) { int lastIndexOfDot = className.lastIndexOf("."); if (lastIndexOfDot == -1) { throw ex; } return loadRecursively(className.substring(0, lastIndexOfDot) + "$" + className .substring(lastIndexOfDot + 1)); } } private static String loadAndroidHome() { Map env = System.getenv(); for (Map.Entry entry : env.entrySet()) { L.d("%s %s", entry.getKey(), entry.getValue()); } if(env.containsKey("ANDROID_HOME")) { return env.get("ANDROID_HOME"); } // check for local.properties file File folder = new File(".").getAbsoluteFile(); while (folder != null && folder.exists()) { File f = new File(folder, "local.properties"); if (f.exists() && f.canRead()) { try { for (String line : FileUtils.readLines(f)) { List keyValue = Splitter.on('=').splitToList(line); if (keyValue.size() == 2) { String key = keyValue.get(0).trim(); if (key.equals("sdk.dir")) { return keyValue.get(1).trim(); } } } } catch (IOException ignored) {} } folder = folder.getParentFile(); } return null; } public static void initForTests() { String androidHome = loadAndroidHome(); if (Strings.isNullOrEmpty(androidHome) || !new File(androidHome).exists()) { throw new IllegalStateException( "you need to have ANDROID_HOME set in your environment" + " to run compiler tests"); } File androidJar = new File(androidHome + "/platforms/android-21/android.jar"); if (!androidJar.exists() || !androidJar.canRead()) { throw new IllegalStateException( "cannot find android jar at " + androidJar.getAbsolutePath()); } // now load android data binding library as well try { ClassLoader classLoader = new URLClassLoader(new URL[]{androidJar.toURI().toURL()}, ModelAnalyzer.class.getClassLoader()); new JavaAnalyzer(classLoader); } catch (MalformedURLException e) { throw new RuntimeException("cannot create class loader", e); } SdkUtil.initialize(8, new File(androidHome)); } }