blob: ee9b441d9c6a292040204ceed5df962e97958516 [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.php.phing;
import java.util.Iterator;
import java.util.concurrent.CancellationException;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import org.netbeans.api.annotations.common.CheckForNull;
import org.netbeans.api.annotations.common.NonNull;
import org.netbeans.api.annotations.common.NullAllowed;
import org.netbeans.api.project.Project;
import org.netbeans.modules.php.phing.exec.PhingExecutable;
import org.netbeans.modules.php.phing.file.PhingTargets;
import org.netbeans.modules.php.phing.file.BuildXml;
import org.netbeans.modules.php.phing.preferences.PhingPreferences;
import org.netbeans.modules.php.phing.ui.customizer.PhingCustomizerProvider;
import org.netbeans.modules.php.phing.util.PhingUtils;
import org.netbeans.modules.web.clientproject.spi.build.BuildToolImplementation;
import org.netbeans.spi.project.ProjectServiceProvider;
import org.netbeans.spi.project.ui.CustomizerProvider2;
import org.openide.DialogDisplayer;
import org.openide.NotifyDescriptor;
import org.openide.filesystems.FileObject;
import org.openide.util.NbBundle;
import org.openide.util.WeakListeners;
@ProjectServiceProvider(
service = {
BuildToolImplementation.class,
PhingBuildTool.class,
}, projectType = "org-netbeans-modules-php-project" // NOI18N
)
public final class PhingBuildTool implements BuildToolImplementation {
private static final Logger LOGGER = Logger.getLogger(PhingBuildTool.class.getName());
public static final String IDENTIFIER = "Phing"; // NOI18N
private final Project project;
private final BuildXml projectBuildXml;
private final PhingTargets projectPhingTargets;
private final PhingPreferences phingPreferences;
final ConcurrentMap<FileObject, PhingTargets> phingTargets = new ConcurrentHashMap<>();
private final ChangeListener cleanupListener = new CleanupListener();
public PhingBuildTool(Project project) {
assert project != null;
this.project = project;
projectBuildXml = new BuildXml(project.getProjectDirectory());
projectPhingTargets = PhingTargets.create(project, projectBuildXml);
phingPreferences = new PhingPreferences(project);
}
@NonNull
public static PhingBuildTool forProject(Project project) {
PhingBuildTool buildTool = inProject(project);
assert buildTool != null : "PhingBuildTool should be found in project " + project.getClass().getName() + " (lookup: " + project.getLookup() + ")";
return buildTool;
}
@CheckForNull
public static PhingBuildTool inProject(Project project) {
assert project != null;
return project.getLookup().lookup(PhingBuildTool.class);
}
@Override
public String getIdentifier() {
return IDENTIFIER;
}
@NbBundle.Messages("PhingBuildTool.name=Phing")
@Override
public String getDisplayName() {
return Bundle.PhingBuildTool_name();
}
public BuildXml getProjectBuildXml() {
return projectBuildXml;
}
public PhingTargets getProjectPhingTargets() {
return projectPhingTargets;
}
public PhingTargets getPhingTargets(@NullAllowed FileObject buildXml) {
if (buildXml == null
|| buildXml.getParent().equals(project.getProjectDirectory())) {
return getProjectPhingTargets();
}
PhingTargets targets = phingTargets.get(buildXml);
if (targets != null) {
return targets;
}
BuildXml file = new BuildXml(buildXml.getParent());
targets = PhingTargets.create(project, file);
PhingTargets currentTargets = phingTargets.putIfAbsent(buildXml, targets);
if (currentTargets != null) {
return currentTargets;
}
// register listener
file.addChangeListener(WeakListeners.change(cleanupListener, file));
return targets;
}
public PhingPreferences getPhingPreferences() {
return phingPreferences;
}
@Override
public boolean isEnabled() {
return projectBuildXml.exists();
}
@NbBundle.Messages("PhingBuildTool.configure=Do you want to configure project actions to call Phing targets?")
@Override
public boolean run(String commandId, boolean waitFinished, boolean warnUser) {
assert isEnabled() : project.getProjectDirectory().getNameExt();
assert projectBuildXml.exists() : project.getProjectDirectory().getNameExt();
String phingBuild = phingPreferences.getTarget(commandId);
if (phingBuild != null) {
PhingExecutable phing = PhingExecutable.getDefault(project, warnUser);
if (phing != null) {
PhingUtils.logUsagePhingBuild();
Future<Integer> result = phing.run(phingBuild.split(" ")); // NOI18N
if (waitFinished) {
try {
result.get();
} catch (InterruptedException ex) {
LOGGER.log(Level.INFO, null, ex);
} catch (CancellationException ex) {
// cancelled by user
LOGGER.log(Level.FINE, null, ex);
} catch (ExecutionException ex) {
LOGGER.log(Level.INFO, null, ex);
if (warnUser) {
// XXX open customizer? show error dialog?
}
}
}
}
} else if (warnUser) {
Object option = DialogDisplayer.getDefault().notify(
new NotifyDescriptor.Confirmation(Bundle.PhingBuildTool_configure(), NotifyDescriptor.YES_NO_OPTION));
if (option == NotifyDescriptor.YES_OPTION) {
project.getLookup().lookup(CustomizerProvider2.class).showCustomizer(PhingCustomizerProvider.CUSTOMIZER_IDENT, null);
}
}
return true;
}
//~ Inner classes
private final class CleanupListener implements ChangeListener {
@Override
public void stateChanged(ChangeEvent e) {
Iterator<FileObject> iterator = phingTargets.keySet().iterator();
while (iterator.hasNext()) {
FileObject buildXml = iterator.next();
if (!buildXml.isValid()) {
LOGGER.log(Level.FINE, "Removing invalid phing file {0}", buildXml);
iterator.remove();
}
}
}
}
}