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

org.gradle.util.ConfigureUtil Maven / Gradle / Ivy

/*
 * Copyright 2010 the original author or authors.
 *
 * Licensed 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.gradle.util;

import groovy.lang.Closure;
import org.codehaus.groovy.runtime.GeneratedClosure;
import org.gradle.api.Action;
import org.gradle.api.Nullable;
import org.gradle.api.internal.ClosureBackedAction;
import org.gradle.api.internal.DynamicObjectUtil;
import org.gradle.internal.Actions;
import org.gradle.internal.metaobject.ConfigureDelegate;
import org.gradle.internal.metaobject.DynamicObject;
import org.gradle.internal.metaobject.InvokeMethodResult;
import org.gradle.internal.metaobject.SetPropertyResult;

import java.util.Collection;
import java.util.Map;

import static org.gradle.util.CollectionUtils.toStringList;

public class ConfigureUtil {

    public static  T configureByMap(Map properties, T delegate) {
        DynamicObject dynamicObject = DynamicObjectUtil.asDynamicObject(delegate);

        for (Map.Entry entry : properties.entrySet()) {
            String name = entry.getKey().toString();
            Object value = entry.getValue();

            SetPropertyResult setterResult = new SetPropertyResult();
            dynamicObject.setProperty(name, value, setterResult);
            if (setterResult.isFound()) {
                continue;
            }

            InvokeMethodResult invokeResult = new InvokeMethodResult();
            dynamicObject.invokeMethod(name, invokeResult, value);
            if (!invokeResult.isFound()) {
                throw dynamicObject.setMissingProperty(name);
            }
        }

        return delegate;
    }

    public static  T configureByMap(Map properties, T delegate, Collection mandatoryKeys) {
        if (!mandatoryKeys.isEmpty()) {
            Collection missingKeys = toStringList(mandatoryKeys);
            missingKeys.removeAll(toStringList(properties.keySet()));
            if (!missingKeys.isEmpty()) {
                throw new IncompleteInputException("Input configuration map does not contain following mandatory keys: " + missingKeys, missingKeys);
            }
        }
        return configureByMap(properties, delegate);
    }

    public static class IncompleteInputException extends RuntimeException {
        private final Collection missingKeys;

        public IncompleteInputException(String message, Collection missingKeys) {
            super(message);
            this.missingKeys = missingKeys;
        }

        public Collection getMissingKeys() {
            return missingKeys;
        }
    }

    /**
     * 

Configures {@code target} with {@code configureClosure}, via the {@link Configurable} interface if necessary.

* *

If {@code target} does not implement {@link Configurable} interface, it is set as the delegate of a clone of * {@code configureClosure} with a resolve strategy of {@code DELEGATE_FIRST}.

* *

If {@code target} does implement the {@link Configurable} interface, the {@code configureClosure} will be passed to * {@code delegate}'s {@link Configurable#configure(Closure)} method.

* * @param configureClosure The configuration closure * @param target The object to be configured * @return The delegate param */ public static T configure(@Nullable Closure configureClosure, T target) { if (configureClosure == null) { return target; } if (target instanceof Configurable) { ((Configurable) target).configure(configureClosure); } else { Action action = configureActionFor(configureClosure, target, new ConfigureDelegate(configureClosure, target)); action.execute(target); } return target; } /** * Creates an action that uses the given closure to configure objects of type T. */ public static Action configureUsing(@Nullable final Closure configureClosure) { if (configureClosure == null) { return Actions.doNothing(); } return new Action() { @Override public void execute(T t) { configure(configureClosure, t); } }; } /** * Called from an object's {@link Configurable#configure} method. */ public static T configureSelf(@Nullable Closure configureClosure, T target) { if (configureClosure == null) { return target; } configureActionFor(configureClosure, target, new ConfigureDelegate(configureClosure, target)).execute(target); return target; } /** * Called from an object's {@link Configurable#configure} method. */ public static T configureSelf(@Nullable Closure configureClosure, T target, ConfigureDelegate closureDelegate) { if (configureClosure == null) { return target; } configureActionFor(configureClosure, target, closureDelegate).execute(target); return target; } private static Action configureActionFor(Closure configureClosure, T target, ConfigureDelegate closureDelegate) { if (!(configureClosure instanceof GeneratedClosure)) { return new ClosureBackedAction(configureClosure, Closure.DELEGATE_FIRST, false); } // Hackery to make closure execution faster, by short-circuiting the expensive property and method lookup on Closure Closure withNewOwner = configureClosure.rehydrate(target, closureDelegate, configureClosure.getThisObject()); return new ClosureBackedAction(withNewOwner, Closure.OWNER_ONLY, false); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy