org.gradle.groovy.scripts.internal.SubsetScriptTransformer Maven / Gradle / Ivy
Show all versions of gradle-api Show documentation
/*
* 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.List;
/**
* Excludes everything from a script except statements that are satisfied by a given predicate, and imports for accessible classes.
*
* 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;
}
@Override
protected int getPhase() {
return Phases.CONVERSION;
}
@Override
public void call(SourceUnit source) throws CompilationFailedException {
AstUtils.filterAndTransformStatements(source, transformer);
// Filter imported classes which are not available yet
Iterator 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);
@SuppressWarnings("unchecked") List value = (List) field.get(source.getAST());
value.removeIf(i -> i.getAlias().equals(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
source.getAST().getClasses().removeIf(classNode -> classNode != scriptClass);
// Remove all the methods from the main class
if (scriptClass != null) {
for (MethodNode methodNode : new ArrayList<>(scriptClass.getMethods())) {
if (!methodNode.getName().equals("run")) {
AstUtils.removeMethod(scriptClass, methodNode);
}
}
}
source.getAST().getMethods().clear();
}
}