blob: a5ec483bc0075b3a22f23455f05c18f8151afac3 [file] [log] [blame]
/*
* 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.modules.java.j2seproject;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.io.IOException;
import java.net.URL;
import java.util.Collection;
import java.util.concurrent.Future;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.lang.model.element.TypeElement;
import javax.swing.SwingUtilities;
import org.netbeans.api.java.classpath.ClassPath;
import org.netbeans.api.java.classpath.JavaClassPathConstants;
import org.netbeans.api.java.platform.JavaPlatformManager;
import org.netbeans.api.java.source.*;
import org.netbeans.api.project.Project;
import org.netbeans.api.project.ProjectManager;
import org.netbeans.modules.java.api.common.ant.UpdateHelper;
import org.netbeans.modules.java.api.common.project.ProjectProperties;
import org.netbeans.spi.java.classpath.support.ClassPathSupport;
import org.netbeans.spi.project.support.ant.AntProjectHelper;
import org.netbeans.spi.project.support.ant.EditableProperties;
import org.netbeans.spi.project.support.ant.PropertyEvaluator;
import org.openide.filesystems.FileChangeAdapter;
import org.openide.filesystems.FileChangeListener;
import org.openide.filesystems.FileObject;
import org.openide.filesystems.FileRenameEvent;
import org.openide.filesystems.FileUtil;
import org.openide.loaders.DataObject;
import org.openide.util.Exceptions;
import org.openide.util.Mutex;
import org.openide.util.MutexException;
import org.openide.util.RequestProcessor;
import org.openide.util.WeakListeners;
/**
*
* @author Tomas Zezula
*/
public final class MainClassUpdater extends FileChangeAdapter implements PropertyChangeListener {
private static final RequestProcessor RP = new RequestProcessor ("main-class-updater",1); //NOI18N
private static final Logger LOG = Logger.getLogger(MainClassUpdater.class.getName());
private static final int NEW = 0;
private static final int STARTED = 1;
private static final int FINISHED = 2;
private final Project project;
private final PropertyEvaluator eval;
private final UpdateHelper helper;
private final ClassPath sourcePath;
private final String mainClassPropName;
private final AtomicInteger state;
//@GuardedBy("this")
private FileObject currentFo;
//@GuardedBy("this")
private DataObject currentDo;
//@GuardedBy("this")
private FileChangeListener foListener;
//@GuardedBy("this")
private PropertyChangeListener doListener;
//@GuardedBy("this")
private long lc = 0;
/** Creates a new instance of MainClassUpdater */
public MainClassUpdater(final Project project, final PropertyEvaluator eval,
final UpdateHelper helper, final ClassPath sourcePath, final String mainClassPropName) {
assert project != null;
assert eval != null;
assert helper != null;
assert sourcePath != null;
assert mainClassPropName != null;
this.project = project;
this.eval = eval;
this.helper = helper;
this.sourcePath = sourcePath;
this.mainClassPropName = mainClassPropName;
this.state = new AtomicInteger(NEW);
}
public void start () {
RP.submit(new Runnable () {
@Override
public void run() {
if (state.compareAndSet(NEW, STARTED)) {
eval.addPropertyChangeListener(MainClassUpdater.this);
addFileChangeListener ();
} else {
throw new IllegalStateException("Current State: " + state.get()); //NOI18N
}
}
});
}
public void stop() {
RP.submit(new Runnable() {
@Override
public void run() {
if (state.compareAndSet(STARTED, FINISHED)) {
synchronized (MainClassUpdater.this) {
if (currentFo != null && foListener != null) {
currentFo.removeFileChangeListener(foListener);
}
if (currentDo != null && doListener != null) {
currentDo.removePropertyChangeListener(doListener);
}
}
} else {
throw new IllegalStateException("Current State: " + state.get()); //NOI18N
}
}
});
}
@Override
public void propertyChange(PropertyChangeEvent evt) {
if (DataObject.PROP_PRIMARY_FILE.equals(evt.getPropertyName())) {
final FileObject newFile = (FileObject) evt.getNewValue();
final FileObject oldFile = (FileObject) evt.getOldValue();
handleMainClassMoved(oldFile, newFile);
} else if (this.mainClassPropName.equals(evt.getPropertyName())) {
//Go out of the ProjectManager.MUTEX, see #118722
RP.post(new Runnable () {
@Override
public void run() {
MainClassUpdater.this.addFileChangeListener ();
}
});
}
}
@Override
public void fileRenamed (final FileRenameEvent evt) {
handleMainClassMoved(evt.getFile(), evt.getFile());
}
private void handleMainClassMoved(final FileObject oldFile, final FileObject newFile) {
if (!project.getProjectDirectory().isValid()) {
return;
}
final FileObject _current;
synchronized (this) {
_current = this.currentFo;
}
if (oldFile == _current) {
Runnable r = new Runnable () {
@Override
public void run () {
try {
final String oldMainClass = ProjectManager.mutex().readAccess(new Mutex.ExceptionAction<String>() {
@Override
public String run() throws Exception {
return eval.getProperty(mainClassPropName);
}
});
Collection<ElementHandle<TypeElement>> main = SourceUtils.getMainClasses(newFile);
String newMainClass = null;
if (!main.isEmpty()) {
ElementHandle<TypeElement> mainHandle = main.iterator().next();
newMainClass = mainHandle.getQualifiedName();
}
if (newMainClass != null && !newMainClass.equals(oldMainClass) && helper.requestUpdate() &&
// XXX ##84806: ideally should update nbproject/configs/*.properties in this case:
eval.getProperty(ProjectProperties.PROP_PROJECT_CONFIGURATION_CONFIG) == null) {
final String newMainClassFinal = newMainClass;
ProjectManager.mutex().writeAccess(new Mutex.ExceptionAction<Void>() {
@Override
public Void run() throws Exception {
EditableProperties props = helper.getProperties(AntProjectHelper.PROJECT_PROPERTIES_PATH);
props.put (mainClassPropName, newMainClassFinal);
helper.putProperties(AntProjectHelper.PROJECT_PROPERTIES_PATH, props);
ProjectManager.getDefault().saveProject (project);
return null;
}
});
}
} catch (IOException e) {
Exceptions.printStackTrace(e);
}
catch (MutexException e) {
Exceptions.printStackTrace(e);
}
}
};
if (SwingUtilities.isEventDispatchThread()) {
r.run();
}
else {
SwingUtilities.invokeLater(r);
}
}
}
private void addFileChangeListener () {
final long clc;
synchronized (MainClassUpdater.this) {
if (currentFo != null && foListener != null) {
currentFo.removeFileChangeListener(foListener);
foListener = null;
currentFo = null;
}
if (currentDo != null && doListener != null) {
currentDo.removePropertyChangeListener(doListener);
doListener = null;
currentDo = null;
}
clc = ++lc;
}
final String mainClassName = org.netbeans.modules.java.j2seproject.MainClassUpdater.this.eval.getProperty(mainClassPropName);
if (mainClassName != null) {
FileObject[] roots = sourcePath.getRoots();
if (roots.length>0) {
ClassPath bootCp = ClassPath.getClassPath(roots[0], ClassPath.BOOT);
if (bootCp == null) {
LOG.log(Level.WARNING, "No bootpath for: {0}", FileUtil.getFileDisplayName(roots[0])); //NOI18N
bootCp = JavaPlatformManager.getDefault().getDefaultPlatform().getBootstrapLibraries();
}
ClassPath compileCp = ClassPath.getClassPath(roots[0], ClassPath.COMPILE);
if (compileCp == null) {
LOG.log(Level.WARNING, "No classpath for: {0}", FileUtil.getFileDisplayName(roots[0])); //NOI18N
compileCp = ClassPathSupport.createClassPath(new URL[0]);
}
final ClassPath systemModules = ClassPath.getClassPath(roots[0], JavaClassPathConstants.MODULE_BOOT_PATH);
final ClassPath modulePath = ClassPath.getClassPath(roots[0], JavaClassPathConstants.MODULE_COMPILE_PATH);
final ClassPath allUnnamed = ClassPath.getClassPath(roots[0], JavaClassPathConstants.MODULE_CLASS_PATH);
final ClasspathInfo cpInfo = new ClasspathInfo.Builder(bootCp)
.setClassPath(compileCp)
.setSourcePath(sourcePath)
.setModuleBootPath(systemModules)
.setModuleCompilePath(modulePath)
.setModuleClassPath(allUnnamed)
.build();
final JavaSource js = JavaSource.create(cpInfo);
// execute immediately, or delay if cannot find main class
ScanUtils.postUserActionTask(js, new Task<CompilationController>() {
@Override
public void run(CompilationController c) throws Exception {
c.toPhase(JavaSource.Phase.ELEMENTS_RESOLVED);
TypeElement te = ScanUtils.checkElement(c, c.getElements().getTypeElement(mainClassName));
if (te != null) {
final FileObject fo = SourceUtils.getFile(te, cpInfo);
synchronized (MainClassUpdater.this) {
if (lc == clc && fo != null && sourcePath.contains(fo)) {
currentFo = fo;
foListener = WeakListeners.create(FileChangeListener.class, MainClassUpdater.this, currentFo);
currentFo.addFileChangeListener(foListener);
currentDo = DataObject.find(currentFo);
doListener = org.openide.util.WeakListeners.propertyChange(MainClassUpdater.this, currentDo);
currentDo.addPropertyChangeListener(doListener);
}
}
}
}
});
}
}
}
}