8149328: remove the dependency on java.logging from java.compiler
Reviewed-by: jjg
#// Usage: jjs resourcetrysuggester.js -- <directory>/* * Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * - Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * - Neither the name of Oracle nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */// This example demonstrates Java subclassing by Java.extend// and javac Compiler and Tree API. This example looks for// finally clauses with "close" call and suggests "resource try"!if (arguments.length == 0) { print("Usage: jjs resourcetrysuggester.js -- <directory>"); exit(1);}// Java types usedvar ExpressionStatementTree = Java.type("com.sun.source.tree.ExpressionStatementTree");var File = Java.type("java.io.File");var Files = Java.type("java.nio.file.Files");var MemberSelectTree = Java.type("com.sun.source.tree.MemberSelectTree");var MethodInvocationTree = Java.type("com.sun.source.tree.MethodInvocationTree");var StringArray = Java.type("java.lang.String[]");var ToolProvider = Java.type("javax.tools.ToolProvider");var Tree = Java.type("com.sun.source.tree.Tree");var Trees = Java.type("com.sun.source.util.Trees");var TreeScanner = Java.type("com.sun.source.util.TreeScanner");// resourceTrySuggestionsfunction resourceTrySuggestions() { // get the system compiler tool var compiler = ToolProvider.systemJavaCompiler; // get standard file manager var fileMgr = compiler.getStandardFileManager(null, null, null); // Using Java.to convert script array (arguments) to a Java String[] var compUnits = fileMgr.getJavaFileObjects( Java.to(arguments, StringArray)); // create a new compilation task var task = compiler.getTask(null, fileMgr, null, null, null, compUnits); // SourcePositions object to get positions of AST nodes var sourcePositions = Trees.instance(task).sourcePositions; // subclass SimpleTreeVisitor - to print resource try suggestions var ResourceTrySuggester = Java.extend(TreeScanner); function hasOnlyEmptyStats(stats) { var itr = stats.iterator(); while (itr.hasNext()) { if (! (itr.next() instanceof EmptyStatementTree)) { return false; } } return true; } // does the given statement list has an expression statement which // calls "close" method (don't worry about types - just crude one will do) function hasCloseCall(stats) { var itr = stats.iterator(); while (itr.hasNext()) { var stat = itr.next(); if (stat instanceof ExpressionStatementTree) { var expr = stat.expression; if (expr instanceof MethodInvocationTree) { var method = expr.methodSelect; if (method instanceof MemberSelectTree) { return method.identifier.toString().equals("close"); } } } } return false; } var visitor = new ResourceTrySuggester() { // current CompilationUnitTree compUnit: null, // current LineMap (pos -> line, column) lineMap: null, // current compilation unit's file name fileName: null, // overrides of TreeScanner methods visitCompilationUnit: function(node, p) { // capture info about current Compilation unit this.compUnit = node; this.lineMap = node.lineMap; this.fileName = node.sourceFile.name; // Using Java.super API to call super class method here return Java.super(visitor).visitCompilationUnit(node, p); }, visitTry: function (node, p) { var finallyBlk = node.finallyBlock; if (finallyBlk != null && hasCloseCall(finallyBlk.statements)) { var pos = sourcePositions.getStartPosition(this.compUnit, node); var line = this.lineMap.getLineNumber(pos); var col = this.lineMap.getColumnNumber(pos); print("Consider resource try statement " + " @ " + this.fileName + ":" + line + ":" + col); // print(node); } } } for each (var cu in task.parse()) { cu.accept(visitor, null); }}// for each ".java" file in directory (recursively) and check it!function main(dir) { Files.walk(dir.toPath()). forEach(function(p) { var name = p.toFile().absolutePath; if (name.endsWith(".java")) { try { resourceTrySuggestions(p.toFile().getAbsolutePath()); } catch (e) { print(e); } } });}main(new File(arguments[0]));