/* * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. * * Copyright 1997-2010 Oracle and/or its affiliates. All rights reserved. * * Oracle and Java are registered trademarks of Oracle and/or its affiliates. * Other names may be trademarks of their respective owners. * * The contents of this file are subject to the terms of either the GNU * General Public License Version 2 only ("GPL") or the Common * Development and Distribution License("CDDL") (collectively, the * "License"). You may not use this file except in compliance with the * License. You can obtain a copy of the License at * http://www.netbeans.org/cddl-gplv2.html * or nbbuild/licenses/CDDL-GPL-2-CP. See the License for the * specific language governing permissions and limitations under the * License. When distributing the software, include this License Header * Notice in each file and include the License file at * nbbuild/licenses/CDDL-GPL-2-CP. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the GPL Version 2 section of the License file that * accompanied this code. If applicable, add the following below the * License Header, with the fields enclosed by brackets [] replaced by * your own identifying information: * "Portions Copyrighted [year] [name of copyright owner]" * * Contributor(s): * * The Original Software is NetBeans. The Initial Developer of the Original * Software is Sun Microsystems, Inc. Portions Copyright 1997-2008 Sun * Microsystems, Inc. All Rights Reserved. * * If you wish your version of this file to be governed by only the CDDL * or only the GPL Version 2, indicate your decision by adding * "[Contributor] elects to include this software in this distribution * under the [CDDL or GPL Version 2] license." If you do not indicate a * single choice of license, a recipient has the option to distribute * your version of this file under either the CDDL, the GPL Version 2 or * to extend the choice of license to its licensees as provided above. * However, if you add GPL Version 2 code and therefore, elected the GPL * Version 2 license, then the option applies only if the new code is * made subject to such option by the copyright holder. */ package org.netbeans.modules.ruby.rubyproject; import java.io.File; import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.Set; import java.util.regex.Matcher; import java.util.regex.Pattern; import javax.swing.text.BadLocationException; import org.jrubyparser.ast.ClassNode; import org.jrubyparser.ast.Node; import org.netbeans.api.project.FileOwnerQuery; import org.netbeans.api.project.Project; import org.netbeans.api.ruby.platform.RubyPlatform; import org.netbeans.editor.BaseDocument; import org.netbeans.editor.Utilities; import org.netbeans.modules.csl.api.DeclarationFinder.DeclarationLocation; import org.netbeans.modules.parsing.api.ParserManager; import org.netbeans.modules.parsing.api.ResultIterator; import org.netbeans.modules.parsing.api.Source; import org.netbeans.modules.parsing.api.UserTask; import org.netbeans.modules.parsing.spi.ParseException; import org.netbeans.modules.parsing.spi.Parser; import org.netbeans.modules.parsing.spi.indexing.support.QuerySupport; import org.netbeans.modules.ruby.AstUtilities; import org.netbeans.modules.ruby.RubyIndex; import org.netbeans.modules.ruby.RubyParseResult; import org.netbeans.modules.ruby.RubyUtils; import org.netbeans.modules.ruby.elements.IndexedClass; import org.netbeans.modules.ruby.platform.gems.GemManager; import org.netbeans.spi.gototest.TestLocator; import org.netbeans.spi.gototest.TestLocator.LocationResult; import org.openide.filesystems.FileObject; import org.openide.filesystems.FileUtil; import org.openide.util.Exceptions; import org.openide.util.NbBundle; /** * Action for jumping from a testfile to its original file or vice versa. * * Some of the file-based rules are based on toggle.el by Ryan Davis: * http://www.emacswiki.org/cgi-bin/emacs/toggle.el * * @author Tor Norbye */ @org.openide.util.lookup.ServiceProvider(service=org.netbeans.spi.gototest.TestLocator.class) public class GotoTest implements TestLocator { private static final String FILE = "(.+)"; // NOI18N private static final String EXT = "(.+)"; // NOI18N private final String[] ZENTEST_PATTERNS = { "app/controllers/" + FILE + "\\." + EXT, "test/controllers/" + FILE + "_test\\." + EXT, // NOI18N "app/views/" + FILE + "\\." + EXT, "test/views/" + FILE + "_test\\." + EXT, // NOI18N "app/models/" + FILE + "\\." + EXT, "test/unit/" + FILE + "_test\\." + EXT, // NOI18N "lib/" + FILE + "\\." + EXT, "test/unit/test_" + FILE + "\\." + EXT, // NOI18N }; private final String[] RSPEC_PATTERNS = { "app/models/" + FILE + "\\." + EXT, "spec/models/" + FILE + "_spec\\." + EXT, // NOI18N "app/controllers/" + FILE + "\\." + EXT, "spec/controllers/" + FILE + "_spec\\." + EXT, // NOI18N "app/views/" + FILE + "\\." + EXT, "spec/views/" + FILE + "_spec\\." + EXT, // NOI18N "app/helpers/" + FILE + "\\." + EXT, "spec/helpers/" + FILE + "_spec\\." + EXT, // NOI18N "lib/" + FILE + "\\." + EXT, "spec/" + FILE + "_spec\\." + EXT, // NOI18N "lib/" + FILE + "\\." + EXT, "spec/lib/" + FILE + "_spec\\." + EXT, // NOI18N "/" + FILE + "\\." + EXT, "/" + FILE + "_spec\\." + EXT, // NOI18N }; private final String[] RAILS_PATTERNS = { "app/controllers/" + FILE + "\\." + EXT, "test/functional/" + FILE + "_test\\." + EXT, // NOI18N "app/models/" + FILE + "\\." + EXT, "test/unit/" + FILE + "_test\\." + EXT, // NOI18N "lib/" + FILE + "\\." + EXT, "test/unit/test_" + FILE + "\\." + EXT, // NOI18N "lib/" + FILE + "\\." + EXT, "test/lib/test_" + FILE + "\\." + EXT, // NOI18N "lib/" + FILE + "\\." + EXT, "test/lib/" + FILE + "_test\\." + EXT, // NOI18N "app/" + FILE + "\\." + EXT, "test/" + FILE + "_test\\." + EXT, // NOI18N }; private final String[] RUBYTEST_PATTERNS = { "lib/" + FILE + "\\." + EXT, "test/test_" + FILE + "\\." + EXT, // NOI18N "lib/" + FILE + "\\." + EXT, "test/" + FILE + "_test\\." + EXT, // NOI18N "lib/" + FILE + "\\." + EXT, "test/tc_" + FILE + "\\." + EXT, // NOI18N "/" + FILE + "\\." + EXT, "/test_" + FILE + "\\." + EXT, // NOI18N "/" + FILE + "\\." + EXT, "/" + FILE + "_test\\." + EXT, // NOI18N "/" + FILE + "\\." + EXT, "/tc_" + FILE + "\\." + EXT, // NOI18N }; private String[] getProjectSourceRootPatterns(Project project) { RubyBaseProject rubyBaseProject = project.getLookup().lookup(RubyBaseProject.class); if (rubyBaseProject == null) { return new String[0]; } List<String> result = new ArrayList<String>(); for (FileObject sourceRoot : rubyBaseProject.getSourceRootFiles()) { for (FileObject testRoot : rubyBaseProject.getTestSourceRootFiles()) { addPatternPairs(sourceRoot, testRoot, result); } } return result.toArray(new String[result.size()]); } private void addPatternPairs(FileObject sourceRoot, FileObject testRoot, List<String> result) { result.add(sourceRoot.getName() + "/" + FILE + "\\." + EXT); result.add(testRoot.getName() + "/" + "test_" + FILE + "\\." + EXT); result.add(sourceRoot.getName() + "/" + FILE + "\\." + EXT); result.add(testRoot.getName() + "/" + FILE + "_test\\." + EXT); result.add(sourceRoot.getName() + "/" + FILE + "\\." + EXT); result.add(testRoot.getName() + "/" + FILE + "_spec\\." + EXT); } public GotoTest() { } private boolean isZenTestInstalled(final Project project) { GemManager gemManager = RubyPlatform.gemManagerFor(project); return gemManager != null && gemManager.getLatestVersion("ZenTest") != null; // NOI18N } private boolean isRSpecInstalled(final Project project) { return new RSpecSupport(project).isRSpecInstalled(); } private boolean isRailsInstalled() { // Uhm no, it could be in the vendor gems too! //return RubyInstallation.getInstance().getVersion("rails") != null; // NOI18N return true; } private void appendRegexp(StringBuilder sb, String s) { // Append chars: If they are regexp escapes, insert literal. // Also do file separator conversion (/ to \ on Windows) for (int i = 0; i < s.length(); i++) { char c = s.charAt(i); if ((c == '/') && (File.separatorChar != '/')) { sb.append(File.separatorChar); } else if (c == '\\') { // Don't insert - strip these puppies out } else { sb.append(c); } } } /* * See if the given file matches pattern1, and if so, check if the * corresponding file matched by pattern2 exists. */ private File findMatching(File file, String pattern1, String pattern2) { assert file.getPath().equals(file.getAbsolutePath()) : "This method requires absolute paths"; String path = slashifyPathForRE(file.getPath()); // Do suffix matching Pattern pattern = Pattern.compile("(.*)" + pattern1); // NOI18N Matcher matcher = pattern.matcher(path); if (matcher.matches()) { String prefix = matcher.group(1); String name = matcher.group(2); String ext = matcher.group(3); int nameIndex = pattern2.indexOf(FILE); assert nameIndex != -1; int extIndex = pattern2.indexOf(EXT,nameIndex+FILE.length()); assert extIndex != -1; StringBuilder sb = new StringBuilder(); appendRegexp(sb, prefix); appendRegexp(sb, File.separator); appendRegexp(sb, pattern2.substring(0, nameIndex)); appendRegexp(sb, name); appendRegexp(sb, pattern2.substring(nameIndex + FILE.length(), extIndex)); appendRegexp(sb, ext); String otherPath = slashifyPathForRE(sb.toString()); File otherFile = new File(otherPath); if (otherFile.exists()) { return otherFile; } // Try looking for arbitrary extensions // (but only for patterns with matches in different // directories - see #119106) if (pattern2.indexOf('/') != -1) { int fileIndex = pattern2.indexOf(FILE); String newPattern = pattern2.substring(0, fileIndex) + name + pattern2.substring(fileIndex+FILE.length()); Pattern p2 = Pattern.compile("(.*)" + newPattern); // NOI18N File parent = otherFile.getParentFile(); File[] children = parent.listFiles(); if (children != null) { for (File f : children) { if (p2.matcher(slashifyPathForRE(f.getPath())).matches()) { return f; } } } } } return null; } private File findMatching(String[] patternPairs, File file, boolean findTest) { int index = 0; while (index < patternPairs.length) { String pattern1 = patternPairs[index]; String pattern2 = patternPairs[index + 1]; File matching = null; if (findTest) { matching = findMatching(file, pattern1, pattern2); } else { matching = findMatching(file, pattern2, pattern1); } if (matching != null) { return matching; } index += 2; } return null; } private FileObject findMatchingFile(FileObject fo, boolean findTest) { // Test zen test paths File file = FileUtil.toFile(fo); // Absolute paths are needed to do prefix path matching file = file.getAbsoluteFile(); File matching = findMatchingFile(file, findTest); if (matching != null) { return FileUtil.toFileObject(matching); } return null; } private File findMatchingFile(File file, boolean findTest) { Project project = FileOwnerQuery.getOwner(FileUtil.toFileObject(file)); if (project != null) { if (isZenTestInstalled(project)) { File matching = findMatching(ZENTEST_PATTERNS, file, findTest); if (matching != null) { return matching; } } } if (isRailsInstalled()) { File matching = findMatching(RAILS_PATTERNS, file, findTest); if (matching != null) { return matching; } } File matching = findMatching(RUBYTEST_PATTERNS, file, findTest); if (matching != null) { return matching; } if (project != null) { if (isRSpecInstalled(project)) { matching = findMatching(RSPEC_PATTERNS, file, findTest); if (matching != null) { return matching; } } // try source/test roots defined in project properties matching = findMatching(getProjectSourceRootPatterns(project), file, findTest); if (matching != null) { return matching; } } return null; } private DeclarationLocation find(FileObject fileObject, int caretOffset, boolean findTest) { FileObject matching = findMatchingFile(fileObject, findTest); if (matching != null) { // TODO - look up file offsets by peeking inside the file // so that we can jump to the test declaration itself? // Or better yet, the test case method corresponding to // the method you're in, or vice versa return new DeclarationLocation(matching, -1); } else { if (caretOffset != -1) { DeclarationLocation location = findTestPair(fileObject, caretOffset, findTest); if (location != DeclarationLocation.NONE) { matching = location.getFileObject(); int offset = location.getOffset(); return new DeclarationLocation(matching, offset); } } } return DeclarationLocation.NONE; } /** * Find the test for the given file, if any * @param fileObject The file whose test we want to find * @param caretOffset The current caret offset, or -1 if not known. The caret offset * can be used to look into the file and see if we're inside a class, and if so * look for a class that is named say Test+name or name+Test. * @return The declaration location for the test, or {@link DeclarationLocation.NONE} if * not found. */ public DeclarationLocation findTest(FileObject fileObject, int caretOffset) { return find(fileObject, caretOffset, true); } /** * Find the file being tested by the given test, if any * @param fileObject The test file whose tested file we want to find * @param caretOffset The current caret offset, or -1 if not known. The caret offset * can be used to look into the file and see if we're inside a class, and if so * look for a class that is named say Test+name or name+Test. * @return The declaration location for the tested file, or {@link DeclarationLocation.NONE} if * not found. */ public DeclarationLocation findTested(FileObject fileObject, int caretOffset) { return find(fileObject, caretOffset, false); } private DeclarationLocation findTestPair(FileObject fo, final int offset, final boolean findTest) { Source source = Source.create(fo); if (source == null) { return DeclarationLocation.NONE; } // XXX Parsing API // if (js.isScanInProgress()) { // return DeclarationLocation.NONE; // } final DeclarationLocation[] locationResult = new DeclarationLocation[1]; locationResult[0] = DeclarationLocation.NONE; try { ParserManager.parse(Collections.singleton(source), new UserTask() { @Override public void run(ResultIterator resultIterator) throws Exception { Parser.Result parserResult = resultIterator.getParserResult(); // for .erb files we have FakeRhtmlParserResult, with which // we can't do much if (!(parserResult instanceof RubyParseResult)) { return; } Node root = AstUtilities.getRoot(parserResult); if (root == null) { return; } ClassNode cls = AstUtilities.findClassAtOffset(root, offset); if (cls == null) { // It's possible the user had the caret on a line // that includes a method that isn't actually inside // the method block - such as the beginning of the // "def" line, or the end of a line after "end". // The latter isn't very likely, but the former can // happen, so let's check the method bodies at the // end of the current line try { BaseDocument doc = RubyUtils.getDocument(parserResult); int endOffset = Utilities.getRowEnd(doc, offset); if (endOffset != offset) { cls = AstUtilities.findClassAtOffset(root, endOffset); } } catch (BadLocationException ble) { // do nothing - see #154991 } } // TODO - look up the specific method at the caret and use it // to pick a corresponding test method! // MethodDefNode method = AstUtilities.findMethodAtOffset(root, endOffset); if (cls != null) { RubyIndex index = RubyIndex.get(parserResult); if (index != null) { String className = AstUtilities.getClassOrModuleName(cls); String TEST = "Test"; // NOI18N if (findTest) { // Foo => FooTest String name = className + TEST; DeclarationLocation location = findClass(name, index); if (location != DeclarationLocation.NONE) { locationResult[0] = location; return; } // Foo => TestFoo name = TEST + className; location = findClass(name, index); if (location != DeclarationLocation.NONE) { locationResult[0] = location; return; } } else { // FooTest => Foo if (className.endsWith(TEST)) { String name = className.substring(0, className.length() - TEST.length()); DeclarationLocation location = findClass(name, index); if (location != DeclarationLocation.NONE) { locationResult[0] = location; return; } } // TestFoo => Foo if (className.startsWith(TEST)) { String name = className.substring(TEST.length()); DeclarationLocation location = findClass(name, index); if (location != DeclarationLocation.NONE) { locationResult[0] = location; return; } } } } } } }); } catch (ParseException pe) { Exceptions.printStackTrace(pe); } return locationResult[0]; } private DeclarationLocation findClass(String className, RubyIndex index) { Set<IndexedClass> classes = index.getClasses(className, QuerySupport.Kind.EXACT, true, false, false /*?*/, null); // First look for candidates whose filenames contain test or tc // Second look for candidates whose paths contain test // Third look for candidates in the same module for (IndexedClass c : classes) { // TODO - pick the -best- candidate. First try in SOURCE scope, then in DEPENDENCIES. // Look for classes that extend superclass FileObject fo = c.getFileObject(); if (fo != null) { int offset = 0; Node node = AstUtilities.getForeignNode(c); if (node != null) { offset = node.getPosition().getStartOffset(); } return new DeclarationLocation(fo, offset); } } return DeclarationLocation.NONE; } public boolean appliesTo(FileObject fo) { return RubyUtils.isRubyFile(fo) || RubyUtils.isRhtmlFile(fo); } public boolean asynchronous() { return false; } public LocationResult findOpposite(FileObject fileObject, int caretOffset) { DeclarationLocation location = findTest(fileObject, caretOffset); if (location == DeclarationLocation.NONE) { location = findTested(fileObject, caretOffset); } if (location != DeclarationLocation.NONE) { return new LocationResult(location.getFileObject(), location.getOffset()); } else { return new LocationResult(NbBundle.getMessage(GotoTest.class, "OppositeNotFound")); } } public void findOpposite(FileObject fo, int caretOffset, LocationListener callback) { throw new UnsupportedOperationException("GotoTest is synchronous"); } public FileType getFileType(FileObject fo) { String name = fo.getName(); return name.indexOf("_test") != -1 || name.indexOf("test_") != -1 || name.indexOf("_spec") != -1 ? // NOI18N TestLocator.FileType.TEST : TestLocator.FileType.TESTED; } /** Strips out backslashes (windows path separators). */ private static String slashifyPathForRE(String path) { return (File.separatorChar == '/') ? path : path.replace(File.separatorChar, '/'); } }