/* * Copyright 2009 the original author or authors. * * 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 org.gradle.groovy.scripts.internal; import org.codehaus.groovy.ast.ClassNode; import org.codehaus.groovy.ast.ImportNode; import org.codehaus.groovy.ast.MethodNode; import org.codehaus.groovy.ast.ModuleNode; import org.codehaus.groovy.control.CompilationFailedException; import org.codehaus.groovy.control.Phases; import org.codehaus.groovy.control.SourceUnit; import org.gradle.internal.UncheckedException; import java.lang.reflect.Field; import java.util.ArrayList; import java.util.Iterator; import java.util.Map; /** * Excludes everything from a script except statements that are satisfied by a given predicate, and imports for accessible classes. * <p> * All other kinds of constructs are filtered, including classes and methods etc. */ public class SubsetScriptTransformer extends AbstractScriptTransformer { private final StatementTransformer transformer; public SubsetScriptTransformer(StatementTransformer transformer) { this.transformer = transformer; } protected int getPhase() { return Phases.CONVERSION; } public void call(SourceUnit source) throws CompilationFailedException { AstUtils.filterAndTransformStatements(source, transformer); // Filter imported classes which are not available yet Iterator<ImportNode> iter = source.getAST().getImports().iterator(); while (iter.hasNext()) { ImportNode importedClass = iter.next(); if (!AstUtils.isVisible(source, importedClass.getClassName())) { try { Field field = ModuleNode.class.getDeclaredField("imports"); field.setAccessible(true); Map value = (Map) field.get(source.getAST()); value.remove(importedClass.getAlias()); } catch (Exception e) { throw UncheckedException.throwAsUncheckedException(e); } } } iter = source.getAST().getStaticImports().values().iterator(); while (iter.hasNext()) { ImportNode importedClass = iter.next(); if (!AstUtils.isVisible(source, importedClass.getClassName())) { iter.remove(); } } iter = source.getAST().getStaticStarImports().values().iterator(); while (iter.hasNext()) { ImportNode importedClass = iter.next(); if (!AstUtils.isVisible(source, importedClass.getClassName())) { iter.remove(); } } ClassNode scriptClass = AstUtils.getScriptClass(source); // Remove all the classes other than the main class Iterator<ClassNode> classes = source.getAST().getClasses().iterator(); while (classes.hasNext()) { ClassNode classNode = classes.next(); if (classNode != scriptClass) { classes.remove(); } } // Remove all the methods from the main class if (scriptClass != null) { for (MethodNode methodNode : new ArrayList<MethodNode>(scriptClass.getMethods())) { if (!methodNode.getName().equals("run")) { AstUtils.removeMethod(scriptClass, methodNode); } } } source.getAST().getMethods().clear(); } }