All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.netbeans.api.java.source.JavaSource Maven / Gradle / Ivy

There is a newer version: RELEASE240
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.netbeans.api.java.source;

import com.sun.source.tree.Tree;
import com.sun.tools.javac.api.JavacTaskImpl;
import com.sun.tools.javac.code.Symbol.CompletionFailure;
import com.sun.tools.javac.tree.JCTree;
import com.sun.tools.javac.util.Log;
import java.io.IOException;
import java.lang.ref.Reference;
import java.lang.ref.WeakReference;
import java.net.URL;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.WeakHashMap;
import java.util.concurrent.Future;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.text.Document;
import javax.swing.text.Position;
import javax.tools.JavaFileObject;
import org.netbeans.api.annotations.common.CheckForNull;
import org.netbeans.api.annotations.common.NonNull;
import org.netbeans.api.annotations.common.NullUnknown;
import org.netbeans.api.java.classpath.ClassPath;
import org.netbeans.api.java.classpath.JavaClassPathConstants;
import org.netbeans.api.java.platform.JavaPlatformManager;
import org.netbeans.modules.java.source.JavaSourceAccessor;
import org.netbeans.modules.java.source.NoJavacHelper;
import org.netbeans.modules.java.source.parsing.CachingArchiveProvider;
import org.netbeans.modules.java.source.parsing.ClassParser;
import org.netbeans.modules.java.source.parsing.ClasspathInfoTask;
import org.netbeans.modules.java.source.parsing.CompilationInfoImpl;
import org.netbeans.modules.java.source.parsing.FileObjects;
import org.netbeans.modules.java.source.parsing.JavacParser;
import org.netbeans.modules.java.source.parsing.JavacParserFactory;
import org.netbeans.modules.java.source.parsing.MimeTask;
import org.netbeans.modules.java.source.parsing.NewComilerTask;
import org.netbeans.modules.java.source.parsing.ParameterNameProviderImpl;
import org.netbeans.modules.java.source.save.ElementOverlay;
import org.netbeans.modules.java.source.usages.ClasspathInfoAccessor;
import org.netbeans.modules.parsing.api.Embedding;
import org.netbeans.modules.parsing.api.ParserManager;
import org.netbeans.modules.parsing.api.ResultIterator;
import org.netbeans.modules.parsing.api.Snapshot;
import org.netbeans.modules.parsing.api.Source;
import org.netbeans.modules.parsing.api.UserTask;
import org.netbeans.modules.parsing.impl.Utilities;
import org.netbeans.modules.parsing.spi.ParseException;
import org.netbeans.modules.parsing.spi.Parser;
import org.netbeans.spi.java.classpath.support.ClassPathSupport;
import org.openide.filesystems.FileObject;
import org.openide.filesystems.FileStateInvalidException;
import org.openide.filesystems.FileUtil;
import org.openide.util.Exceptions;
import org.openide.util.NbBundle;
import org.openide.util.Parameters;

/** Class representing Java source file opened in the editor.
 *
 * @author Petr Hrebejk, Tomas Zezula
 */
public final class JavaSource {


    public static enum Phase {
        MODIFIED,

        PARSED,

        ELEMENTS_RESOLVED,

        RESOLVED,

        UP_TO_DATE;

    };

    public static enum Priority {
        MAX,
        HIGH,
        ABOVE_NORMAL,
        NORMAL,
        BELOW_NORMAL,
        LOW,
        MIN
    };



    /**
     * This specialization of {@link IOException} signals that a {@link JavaSource#runUserActionTask}
     * or {@link JavaSource#runModificationTask} failed due to lack of memory. The {@link InsufficientMemoryException#getFile}
     * method returns a file which cannot be processed.
     */
    public static final class InsufficientMemoryException extends IOException {

        private FileObject fo;

        private InsufficientMemoryException (final String message, final FileObject fo) {
            super (message);
            this.fo = fo;
        }

        private InsufficientMemoryException (FileObject fo) {
            this (NbBundle.getMessage(JavaSource.class, "MSG_UnsufficientMemoryException", FileUtil.getFileDisplayName (fo)),fo);
        }


        /**
         * Returns file which cannot be processed due to lack of memory.
         * @return {@link FileObject}
         */
        public FileObject getFile () {
            return this.fo;
        }
    }

    static {
        JavaSourceAccessor.setINSTANCE (new JavaSourceAccessorImpl ());
    }

    //Source files being processed, may be empty
    private final Collection sources;
    //Files being processed, may be empty
    private final Collection files;
    //Classpath info when explicitely given, may be null
    private final ClasspathInfo classpathInfo;
    //Cached classpath info
    //@GuardedBy(this)
    private ClasspathInfo cachedCpInfo;

    private static final Logger LOGGER = Logger.getLogger(JavaSource.class.getName());

    /**
     * Returns a {@link JavaSource} instance representing given {@link org.openide.filesystems.FileObject}s
     * and classpath represented by given {@link ClasspathInfo}.
     * @param cpInfo the classpaths to be used.
     * @param files for which the {@link JavaSource} should be created
     * @return a new {@link JavaSource}
     * @throws IllegalArgumentException if fileObject or cpInfo is null
     */
    public static @NullUnknown JavaSource create(final @NonNull ClasspathInfo cpInfo, final @NonNull FileObject... files) throws IllegalArgumentException {
        if (files == null || cpInfo == null) {
            throw new IllegalArgumentException ();
        }
        return _create(cpInfo, Arrays.asList(files));
    }

    /**
     * Returns a {@link JavaSource} instance representing given {@link org.openide.filesystems.FileObject}s
     * and classpath represented by given {@link ClasspathInfo}.
     * @param cpInfo the classpaths to be used.
     * @param files for which the {@link JavaSource} should be created
     * @return a new {@link JavaSource} or {@code null} if the essential infrastructure is missing
     * @throws IllegalArgumentException if fileObject or cpInfo is null
     */
    public static @NullUnknown JavaSource create(final @NonNull ClasspathInfo cpInfo, final @NonNull Collection files) throws IllegalArgumentException {
        return _create(cpInfo, files);
    }

    //where

    private static @NullUnknown JavaSource _create(final ClasspathInfo cpInfo, final @NonNull Collection files) throws IllegalArgumentException {
        if (files == null) {
            throw new IllegalArgumentException ();
        }
        if (!NoJavacHelper.hasWorkingJavac())
            return null;
        try {
            return new JavaSource(cpInfo, files);
// TODO: Split
//        } catch (DataObjectNotFoundException donf) {
//            Logger.getLogger("global").warning("Ignoring non existent file: " + FileUtil.getFileDisplayName(donf.getFileObject()));     //NOI18N
        } catch (IOException ex) {
            Exceptions.printStackTrace(ex);
        }
        return null;
    }

    private static Map> file2JavaSource = new WeakHashMap>();
    private static final String[] supportedMIMETypes = new String[] {ClassParser.MIME_TYPE, JavacParser.MIME_TYPE};

    /**
     * Returns a {@link JavaSource} instance associated to given {@link org.openide.filesystems.FileObject},
     * it returns null if the {@link Document} is not associated with data type providing the {@link JavaSource}.
     * @param fileObject for which the {@link JavaSource} should be found/created.
     * @return {@link JavaSource} or null
     * @throws IllegalArgumentException if fileObject is null
     */
    public static @CheckForNull JavaSource forFileObject(@NonNull FileObject fileObject) throws IllegalArgumentException {
        if (fileObject == null) {
            throw new IllegalArgumentException ("fileObject == null");  //NOI18N
        }
        if (!fileObject.isValid()) {
            LOGGER.log(Level.FINE, "FileObject ({0}) passed to JavaSource.forFileObject is invalid", fileObject.toURI().toString());
            return null;
        }

        String mimeType = null;

        try {
            if (   fileObject.getFileSystem().isDefault()
                && fileObject.getAttribute("javax.script.ScriptEngine") != null
                && fileObject.getAttribute("template") == Boolean.TRUE) {
                LOGGER.log(Level.FINE, "FileObject ({0}) passed to JavaSource.forFileObject is a template", fileObject.toURI().toString());
                return null;
            }
        } catch (FileStateInvalidException ex) {
            LOGGER.log(Level.FINE, null, ex);
            return null;
        }

        Reference ref = file2JavaSource.get(fileObject);
        JavaSource js = ref != null ? ref.get() : null;
        if (js == null) {
            mimeType = mimeType == null ? FileUtil.getMIMEType(fileObject, supportedMIMETypes) : mimeType;
            if (ClassParser.MIME_TYPE.equals(mimeType) || FileObjects.CLASS.equals(fileObject.getExt())) {
                ClassPath bootPath = ClassPath.getClassPath(fileObject, ClassPath.BOOT);
                if (bootPath == null) {
                    //javac requires at least java.lang
                    bootPath = JavaPlatformManager.getDefault().getDefaultPlatform().getBootstrapLibraries();
                }
                boolean hasModulePath = true;
                ClassPath moduleBootPath = ClassPath.getClassPath(fileObject, JavaClassPathConstants.MODULE_BOOT_PATH);
                if (moduleBootPath == null) {
                    moduleBootPath = bootPath;
                    hasModulePath = false;
                }
                ClassPath compilePath = ClassPath.getClassPath(fileObject, ClassPath.COMPILE);
                if (compilePath == null) {
                    compilePath = ClassPath.EMPTY;
                }
                ClassPath moduleCompilePath = ClassPath.getClassPath(fileObject, JavaClassPathConstants.MODULE_COMPILE_PATH);
                if (moduleCompilePath == null) {
                    moduleCompilePath = ClassPath.EMPTY;
                }
                ClassPath moduleClassPath = ClassPath.getClassPath(fileObject, JavaClassPathConstants.MODULE_CLASS_PATH);
                if (moduleClassPath == null) {
                    moduleClassPath = ClassPath.EMPTY;
                }
                ClassPath srcPath = ClassPath.getClassPath(fileObject, ClassPath.SOURCE);
                if (srcPath == null) {
                    srcPath = ClassPath.EMPTY;
                }
                ClassPath moduleSrcPath = ClassPath.getClassPath(fileObject, JavaClassPathConstants.MODULE_SOURCE_PATH);
                if (moduleSrcPath == null) {
                    moduleSrcPath = ClassPath.EMPTY;
                }
                ClassPath execPath = ClassPath.getClassPath(fileObject, ClassPath.EXECUTE);
                if (execPath != null) {
                    if (hasModulePath) {
                        //Todo: Upgrade module path should be here.
                        moduleClassPath = moduleClassPath == ClassPath.EMPTY ?
                                execPath :
                                ClassPathSupport.createProxyClassPath(moduleClassPath, execPath);
                        compilePath = ClassPathSupport.createProxyClassPath(compilePath, execPath);
                    } else {
                        bootPath = ClassPathSupport.createProxyClassPath(execPath, bootPath);
                    }
                }
                final ClasspathInfo info = ClasspathInfoAccessor.getINSTANCE().create(
                    bootPath,
                    moduleBootPath,
                    compilePath,
                    moduleCompilePath,
                    moduleClassPath,
                    srcPath,
                    moduleSrcPath,
                    null,
                    false, false, false, true, false, null);
                FileObject root = ClassPathSupport.createProxyClassPath(
                    ClassPathSupport.createClassPath(CachingArchiveProvider.getDefault().ctSymRootsFor(bootPath)),
                    bootPath,
                    compilePath,
                    srcPath).findOwnerRoot(fileObject);
                if (root == null) {
                    LOGGER.log(Level.FINE, "FileObject ({0}) passed to JavaSource.forFileObject of mimeType classfile does not have a corresponding root", fileObject.toURI().toString());
                    return null;
                }
                try {
                    Source classSource = Source.create(fileObject);
                    // avoid creation for invalid class files
                    if (classSource != null) {
                        js = new JavaSource (info, fileObject, classSource, root);
                    }
                } catch (IOException ioe) {
                    Exceptions.printStackTrace(ioe);
                }
            }
            else {
                if (!"text/x-java".equals(mimeType) && !"java".equals(fileObject.getExt())) {  //NOI18N
                    LOGGER.log(Level.FINE, "FileObject ({0}) passed to JavaSource.forFileObject is not a Java source file (mimetype: {1})", new Object[] {fileObject.toURI().toString(), mimeType});
                    return null;
                }
                js = _create(null, Collections.singletonList(fileObject));
            }
            file2JavaSource.put(fileObject, new WeakReference(js));
        }
        return js;
    }

    /**
     * Returns a {@link JavaSource} instance associated to the given {@link javax.swing.Document},
     * it returns null if the {@link Document} is not
     * associated with data type providing the {@link JavaSource}.
     * @param doc {@link Document} for which the {@link JavaSource} should be found/created.
     * @return {@link JavaSource} or null
     * @throws IllegalArgumentException if doc is null
     */
    public static @CheckForNull JavaSource forDocument(@NonNull Document doc) throws IllegalArgumentException {
        if (doc == null) {
            throw new IllegalArgumentException ("doc == null");  //NOI18N
        }
        Reference ref = (Reference) doc.getProperty(JavaSource.class);
        JavaSource js = ref != null ? (JavaSource) ref.get() : null;
        if (js == null) {
            FileObject fo = Utilities.getFileObject(doc);
            if (fo != null) {
                js = forFileObject(fo);
            }
        }
        return js;
    }

    /**
     * Creates a new instance of JavaSource
     * @param files to create JavaSource for
     * @param cpInfo classpath info
     */
    private JavaSource (final ClasspathInfo cpInfo,
                        final @NonNull Collection files) throws IOException {
// TODO: Split
//        boolean multipleSources = files.size() > 1;
        final List sources = new ArrayList(files.size());
        final List fl = new ArrayList(files.size());
        for (Iterator it = files.iterator(); it.hasNext();) {
            FileObject file = it.next();
            Logger.getLogger("TIMER").log(Level.FINE, "JavaSource",
                new Object[] {file, this});
            if (!file.isValid()) {
// TODO: Split
//                if (multipleSources) {
                    LOGGER.warning("Ignoring non existent file: " + FileUtil.getFileDisplayName(file));     //NOI18N
// TODO: Split
//                }
//                else {
//                    DataObject.find(file);  //throws IOE
//                }
            }
            else {
                fl.add (file);
                sources.add(Source.create(file));
            }
        }
        this.files = Collections.unmodifiableList(fl);
        this.sources = Collections.unmodifiableList(sources);
        this.classpathInfo = cpInfo;
    }

    /**
     * Creates a new instance of JavaSource for an class file
     * @param info the compilation info to be used
     * @param classFileObject the class file
     * @param root owning the class file
     * @throws IOException
     */
    private JavaSource (final @NonNull ClasspathInfo info,
                        final @NonNull FileObject classFileObject,
                        final @NonNull Source classSource,
                        final @NonNull FileObject root) throws IOException {
        assert info != null;
        assert classFileObject != null;
        assert classSource != null;
        assert root != null;
        this.files = Collections.singletonList(classFileObject);
        this.sources = Collections.singletonList(classSource);
        this.classpathInfo =  info;
    }

    /** Runs a task which permits for controlling phases of the parsing process.
     * You probably do not want to call this method unless you are reacting to
     * some user's GUI input which requires immediate action (e.g. code completion popup).
     * In all other cases use {@link JavaSourceTaskFactory}.
* Call to this method will cancel processing of all the phase completion tasks until * this task does not finish.
* @see org.netbeans.api.java.source.CancellableTask for information about implementation requirements * @param task The task which. * @param shared if true the java compiler may be reused by other {@link org.netbeans.api.java.source.CancellableTasks}, * the value false may have negative impact on the IDE performance. *
*

* It's legal to nest the {@link JavaSource#runUserActionTask} into another {@link JavaSource#runUserActionTask}. * It's also legal to nest the {@link JavaSource#runModificationTask} into {@link JavaSource#runUserActionTask}, * the outer {@link JavaSource#runUserActionTask} does not see changes caused by nested {@link JavaSource#runModificationTask}, * but the following nested task see them. *

*
*/ public void runUserActionTask( final @NonNull Task task, final boolean shared) throws IOException { runUserActionTaskImpl(task, shared); } private long runUserActionTaskImpl ( final Task task, final boolean shared) throws IOException { Parameters.notNull("task", task); long currentId = -1; if (sources.isEmpty()) { try { ParserManager.parse(JavacParser.MIME_TYPE,new MimeTask(this, task, this.classpathInfo)); } catch (final ParseException pe) { final Throwable rootCase = pe.getCause(); if (rootCase instanceof CompletionFailure) { IOException ioe = new IOException (); ioe.initCause(rootCase); throw ioe; } else if (rootCase instanceof RuntimeException) { throw (RuntimeException) rootCase; } else { IOException ioe = new IOException (); ioe.initCause(rootCase); throw ioe; } } } else { try { final UserTask _task = new MultiTask(this, task, this.classpathInfo); ParserManager.parse(sources, _task); } catch (final ParseException pe) { final Throwable rootCase = pe.getCause(); if (rootCase instanceof CompletionFailure) { IOException ioe = new IOException (); ioe.initCause(rootCase); throw ioe; } else if (rootCase instanceof RuntimeException) { throw (RuntimeException) rootCase; } else { IOException ioe = new IOException (); ioe.initCause(rootCase); throw ioe; } } } return currentId; } //where private static class MultiTask extends ClasspathInfoTask { private final JavaSource js; private final Task task; public MultiTask (final JavaSource js, final Task task, final ClasspathInfo cpInfo) { super (cpInfo); assert js != null; assert task != null; this.js = js; this.task = task; } @Override public void run(ResultIterator resultIterator) throws Exception { final Snapshot snapshot = resultIterator.getSnapshot(); if (JavacParser.MIME_TYPE.equals(snapshot.getMimeType()) || ClassParser.MIME_TYPE.equals(snapshot.getMimeType())) { Parser.Result result = resultIterator.getParserResult(); if (result == null) { //Deleted file of other parser critical issue return; } final CompilationController cc = CompilationController.get(result); assert cc != null; cc.setJavaSource(this.js); task.run (cc); final JavacTaskImpl jt = cc.impl.getJavacTask(); Log.instance(jt.getContext()).nerrors = 0; } else { Parser.Result result = findEmbeddedJava (resultIterator); if (result == null) { //No embedded java return; } final CompilationController cc = CompilationController.get(result); assert cc != null; cc.setJavaSource(this.js); task.run (cc); final JavacTaskImpl jt = cc.impl.getJavacTask(); Log.instance(jt.getContext()).nerrors = 0; } } @Override public String toString () { return this.getClass().getName()+"["+task.getClass().getName()+"]"; //NOI18N } private Parser.Result findEmbeddedJava (final ResultIterator theMess) throws ParseException { final Collection todo = new ArrayList(); //BFS should perform better than DFS in this dark. for (Embedding embedding : theMess.getEmbeddings()) { if (JavacParser.MIME_TYPE.equals(embedding.getMimeType())) { return theMess.getResultIterator(embedding).getParserResult(); } else { todo.add(embedding); } } for (Embedding embedding : todo) { Parser.Result result = findEmbeddedJava(theMess.getResultIterator(embedding)); if (result != null) { return result; } } return null; } } static long createTaggedController(final FileObject fo, int position, final long timestamp, final Object[] controller) throws IOException { ClasspathInfo cpi = ClasspathInfo.create(fo); Collection sources = Collections.singleton(Source.create(fo)); CompilationController cc = (CompilationController) controller[0]; final NewComilerTask _task = new NewComilerTask(cpi, position, cc, timestamp); try { ParserManager.parse(sources, _task); controller[0] = _task.getCompilationController(); return _task.getTimeStamp(); } catch (final ParseException pe) { final Throwable rootCase = pe.getCause(); if (rootCase instanceof CompletionFailure) { throw new IOException (rootCase); } else if (rootCase instanceof RuntimeException) { throw (RuntimeException) rootCase; } else { throw new IOException (rootCase); } } } long createTaggedController (final long timestamp, final Object[] controller) throws IOException { assert controller.length == 1; assert controller[0] == null || controller[0] instanceof CompilationController; try { CompilationController cc = (CompilationController) controller[0]; final NewComilerTask _task = new NewComilerTask(this.classpathInfo, -1, cc, timestamp); ParserManager.parse(sources, _task); controller[0] = _task.getCompilationController(); return _task.getTimeStamp(); } catch (final ParseException pe) { final Throwable rootCase = pe.getCause(); if (rootCase instanceof CompletionFailure) { throw new IOException (rootCase); } else if (rootCase instanceof RuntimeException) { throw (RuntimeException) rootCase; } else { throw new IOException (rootCase); } } } /** * Performs the given task when the scan finished. When no background scan is running * it performs the given task synchronously. When the background scan is active it queues * the given task and returns, the task is performed when the background scan completes by * the thread doing the background scan. * @param task to be performed * @param shared if true the java compiler may be reused by other {@link org.netbeans.api.java.source.CancellableTasks}, * the value false may have negative impact on the IDE performance. * @return {@link Future} which can be used to find out the sate of the task {@link Future#isDone} or {@link Future#isCancelled}. * The caller may cancel the task using {@link Future#cancel} or wait until the task is performed {@link Future#get}. * @throws IOException encapsulating the exception thrown by {@link CancellableTasks#run} * @since 0.12 */ public @NonNull Future runWhenScanFinished (final @NonNull Task task, final boolean shared) throws IOException { Parameters.notNull("task", task); if (sources.isEmpty()) { try { return ParserManager.parseWhenScanFinished(JavacParser.MIME_TYPE,new MimeTask(this, task, this.classpathInfo)); } catch (final ParseException pe) { final Throwable rootCase = pe.getCause(); if (rootCase instanceof CompletionFailure) { IOException ioe = new IOException (); ioe.initCause(rootCase); throw ioe; } else if (rootCase instanceof RuntimeException) { throw (RuntimeException) rootCase; } else { IOException ioe = new IOException (); ioe.initCause(rootCase); throw ioe; } } } else { try { final UserTask _task = new MultiTask(this, task, this.classpathInfo); return ParserManager.parseWhenScanFinished(sources, _task); } catch (final ParseException pe) { final Throwable rootCase = pe.getCause(); if (rootCase instanceof CompletionFailure) { IOException ioe = new IOException (); ioe.initCause(rootCase); throw ioe; } else if (rootCase instanceof RuntimeException) { throw (RuntimeException) rootCase; } else { IOException ioe = new IOException (); ioe.initCause(rootCase); throw ioe; } } } } /** Runs a task which permits for modifying the sources. * Call to this method will cancel processing of all the phase completion tasks until * this task does not finish.
* @see Task for information about implementation requirements * @param task The task which. */ public @NonNull ModificationResult runModificationTask(final @NonNull Task task) throws IOException { if (task == null) { throw new IllegalArgumentException ("Task cannot be null"); //NOI18N } final ModificationResult result = new ModificationResult(); final ElementOverlay overlay = ElementOverlay.getOrCreateOverlay(); long start = System.currentTimeMillis(); Task inner = new Task() { @Override public void run(CompilationController cc) throws Exception { final WorkingCopy copy = new WorkingCopy(cc.impl, overlay); copy.setJavaSource(JavaSource.this); if (sources.isEmpty()) { //runUserActionTask for source-less JavaSources does not require toPhase //so automatically initializing also for runModificationTask: copy.toPhase(Phase.PARSED); } task.run(copy); final JavacTaskImpl jt = copy.impl.getJavacTask(); Log.instance(jt.getContext()).nerrors = 0; final List diffs = copy.getChanges(result.tag2Span); if (diffs != null && diffs.size() > 0) { final FileObject file = copy.getFileObject(); result.diffs.put(file != null ? file : FileUtil.createMemoryFileSystem().getRoot().createData("temp", "java"), diffs); } } }; runUserActionTask(inner, true); if (sources.size() == 1) { Logger.getLogger("TIMER").log(Level.FINE, "Modification Task", //NOI18N new Object[] {sources.iterator().next().getFileObject(), System.currentTimeMillis() - start}); } return result; } /** * Returns the classpaths ({@link ClasspathInfo}) used by this * {@link JavaSource} * @return {@link ClasspathInfo}, never returns null. */ public @NonNull ClasspathInfo getClasspathInfo() { synchronized (this) { if (this.cachedCpInfo != null) { return this.cachedCpInfo; } } ClasspathInfo _tmp = this.classpathInfo; if (_tmp == null) { assert ! this.files.isEmpty(); _tmp = ClasspathInfo.create(this.files.iterator().next()); } synchronized (this) { if (this.cachedCpInfo == null) { this.cachedCpInfo = _tmp; } return this.cachedCpInfo; } } /** * Returns unmodifiable {@link Collection} of {@link FileObject}s used by this {@link JavaSource} * @return the {@link FileObject}s */ public @NonNull Collection getFileObjects() { return files; } private static class JavaSourceAccessorImpl extends JavaSourceAccessor { @Override public JavacTaskImpl getJavacTask (final CompilationInfo compilationInfo) { assert compilationInfo != null; return compilationInfo.impl.getJavacTask(); } public JavaSource create(ClasspathInfo cpInfo, PositionConverter binding, Collection files) throws IllegalArgumentException { return JavaSource.create(cpInfo, files); } @Override public CompilationController createCompilationController (final Source s, final ClasspathInfo cpInfo) throws IOException, ParseException { final JavacParserFactory factory = JavacParserFactory.getDefault(); final Snapshot snapshot = s != null ? s.createSnapshot() : null; final JavacParser parser = factory.createPrivateParser(snapshot); if (parser == null) return null; final ClasspathInfoTask dummy = new ClasspathInfoTask(cpInfo) { @Override public void run(ResultIterator resultIterator) throws Exception { } }; parser.parse(snapshot, dummy, null); return CompilationController.get(parser.getResult(dummy)); } @Override public long createTaggedCompilationController (FileObject f, int position, long currentTag, Object[] out) throws IOException { return JavaSource.createTaggedController(f, position, currentTag, out); } public long createTaggedCompilationController (JavaSource js, long currentTag, Object[] out) throws IOException { return js.createTaggedController(currentTag, out); } public CompilationInfo createCompilationInfo (final CompilationInfoImpl impl) { return new CompilationInfo(impl); } public CompilationController createCompilationController (final CompilationInfoImpl impl) { return new CompilationController(impl); } public void invalidate (final CompilationInfo info) { assert info != null; info.invalidate(); } @Override public Collection getSources(JavaSource js) { assert js != null; return js.sources; } @Override public void setJavaSource(final CompilationInfo info, final JavaSource js) { assert info != null; assert js != null; info.setJavaSource(js); } @Override public void invalidateCachedClasspathInfo(FileObject file) { assert file != null; final Reference ref = file2JavaSource.get(file); JavaSource js; if (ref != null && (js=ref.get())!=null) { synchronized (js) { js.cachedCpInfo = null; } } } @Override public CompilationInfoImpl getCompilationInfoImpl(CompilationInfo info) { assert info != null; return info.impl; } @Override public @NonNull String generateReadableParameterName (@NonNull String typeName, @NonNull Set used) { return ParameterNameProviderImpl.generateReadableParameterName(typeName, used); } @Override public ModificationResult.Difference createDifference(ModificationResult.Difference.Kind kind, Position startPos, Position endPos, String oldText, String newText, String description, Source src) { return new ModificationResult.Difference(kind, startPos, endPos, oldText, newText, description, src); } @Override public ModificationResult.Difference createNewFileDifference(JavaFileObject fileObject, String text) { return new ModificationResult.CreateChange(fileObject, text); } @Override public ModificationResult createModificationResult(Map> diffs, Map tag2Span) { ModificationResult result = new ModificationResult(); result.diffs = diffs; result.tag2Span = tag2Span; return result; } @Override public ElementUtilities createElementUtilities(@NonNull JavacTaskImpl jt) { return new ElementUtilities(jt); } @Override public Map> getDiffsFromModificationResult(ModificationResult mr) { return mr.diffs; } @Override public Map getTagsFromModificationResult(ModificationResult mr) { return mr.tag2Span; } @Override public ClassIndex createClassIndex(ClassPath bootPath, ClassPath classPath, ClassPath sourcePath, boolean supportsChanges) { return new ClassIndex(bootPath, classPath, sourcePath, supportsChanges); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy