diff --git a/core/src/main/java/hudson/Util.java b/core/src/main/java/hudson/Util.java index d2a0fab5ca0c78c3e805fcfbec31ebf0d9080e43..448457740059f58db6dbb274588d878ab85da823 100644 --- a/core/src/main/java/hudson/Util.java +++ b/core/src/main/java/hudson/Util.java @@ -862,6 +862,10 @@ public class Util { return l!=null ? l : Collections.emptySet(); } + public static Collection fixNull(Collection l) { + return l!=null ? l : Collections.emptySet(); + } + /** * Cuts all the leading path portion and get just the file name. */ diff --git a/core/src/main/java/hudson/model/AbstractProject.java b/core/src/main/java/hudson/model/AbstractProject.java index 92b244627bd6724232f635c4140862c7479be6aa..f8c489d541b97723554a9ae25d40da1fe1c15790 100644 --- a/core/src/main/java/hudson/model/AbstractProject.java +++ b/core/src/main/java/hudson/model/AbstractProject.java @@ -482,6 +482,9 @@ public abstract class AbstractProject

,R extends A if(a!=null) transientActions.add(a); } + + for (TransientProjectActionFactory tpaf : TransientProjectActionFactory.all()) + transientActions.addAll(Util.fixNull(tpaf.createFor(this))); // be defensive against null } } @@ -814,6 +817,8 @@ public abstract class AbstractProject

,R extends A * Note that this method returns a read-only view of {@link Action}s. * {@link BuildStep}s and others who want to add a project action * should do so by implementing {@link BuildStep#getProjectAction(AbstractProject)}. + * + * @see TransientProjectActionFactory */ @Override public synchronized List getActions() { diff --git a/core/src/main/java/hudson/model/TransientProjectActionFactory.java b/core/src/main/java/hudson/model/TransientProjectActionFactory.java new file mode 100644 index 0000000000000000000000000000000000000000..c885dc6c7c70db6b7b71080d910274004b1beb8e --- /dev/null +++ b/core/src/main/java/hudson/model/TransientProjectActionFactory.java @@ -0,0 +1,70 @@ +/* + * The MIT License + * + * Copyright (c) 2004-2009, Sun Microsystems, Inc. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package hudson.model; + +import hudson.Extension; +import hudson.ExtensionList; +import hudson.ExtensionPoint; +import hudson.tasks.BuildStep; + +import java.util.Collection; + +/** + * Extension point for inserting transient {@link Action}s into {@link AbstractProject}s. + * + *

+ * Actions of projects are primarily determined by {@link BuildStep}s that are associated by configurations, + * but sometimes it's convenient to be able to add actions across all or many projects, without being invoked + * through configuration. This extension point provides such a mechanism. + * + * Actions of {@link AbstractProject}s are transient — they will not be persisted, and each time Hudson starts + * or the configuration of the job changes, they'll be recreated. Therefore, to maintain persistent data + * per project, you'll need to do data serialization by yourself. Do so by storing a file + * under {@link AbstractProject#getRootDir()}. + * + *

+ * To register your implementation, put {@link Extension} on your subtype. + * + * @author Kohsuke Kawaguchi + * @since 1.327 + * @see Action + */ +public abstract class TransientProjectActionFactory implements ExtensionPoint { + /** + * Creates actions for the given project. + * + * @param target + * The project for which the action objects are requested. Never null. + * @return + * Can be empty but must not be null. + */ + public abstract Collection createFor(AbstractProject target); + + /** + * Returns all the registered {@link TransientProjectActionFactory}s. + */ + public static ExtensionList all() { + return Hudson.getInstance().getExtensionList(TransientProjectActionFactory.class); + } +}