org.gradle.model.internal.registry.RuleBinder.java Source code

Java tutorial

Introduction

Here is the source code for org.gradle.model.internal.registry.RuleBinder.java

Source

/*
 * Copyright 2013 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.model.internal.registry;

import com.google.common.base.Function;
import com.google.common.base.Predicate;
import com.google.common.collect.FluentIterable;
import com.google.common.collect.Iterables;
import net.jcip.annotations.NotThreadSafe;
import org.gradle.api.Action;
import org.gradle.api.Nullable;
import org.gradle.model.internal.core.ModelBinding;
import org.gradle.model.internal.core.ModelPath;
import org.gradle.model.internal.core.ModelReference;
import org.gradle.model.internal.core.rule.describe.ModelRuleDescriptor;

import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Set;

/**
 * The progressive binding of the subject/inputs of the references of a model rule.
 *
 * This type is mutable.
 */
@NotThreadSafe
public class RuleBinder<T> {

    private final ModelReference<T> subjectReference;
    private final List<? extends ModelReference<?>> inputReferences;

    private final ModelRuleDescriptor descriptor;

    private Action<? super RuleBinder<T>> onBind;

    private int inputsBound;

    private ModelBinding<T> subjectBinding;
    private List<ModelBinding<?>> inputBindings;

    public RuleBinder(@Nullable ModelReference<T> subjectReference,
            List<? extends ModelReference<?>> inputReferences, ModelRuleDescriptor descriptor,
            Action<? super RuleBinder<T>> onBind) {
        this.subjectReference = subjectReference;
        this.inputReferences = inputReferences;
        this.descriptor = descriptor;
        this.onBind = onBind;

        this.inputBindings = inputReferences.isEmpty() ? Collections.<ModelBinding<?>>emptyList()
                : Arrays.asList(new ModelBinding<?>[inputReferences.size()]); // fix size

        maybeFire();
    }

    @Nullable
    public ModelReference<T> getSubjectReference() {
        return subjectReference;
    }

    public List<? extends ModelReference<?>> getInputReferences() {
        return inputReferences;
    }

    public ModelBinding<T> getSubjectBinding() {
        return subjectBinding;
    }

    public List<ModelBinding<?>> getInputBindings() {
        return inputBindings;
    }

    public ModelRuleDescriptor getDescriptor() {
        return descriptor;
    }

    public boolean bindSubject(ModelPath path) {
        assert this.subjectBinding == null;
        this.subjectBinding = bind(subjectReference, path);
        return maybeFire();
    }

    public boolean bindInput(int i, ModelPath path) {
        assert this.inputBindings.get(i) == null;
        this.inputBindings.set(i, bind(inputReferences.get(i), path));
        inputsBound += 1;
        return maybeFire();
    }

    private boolean maybeFire() {
        if (isBound()) {
            fire();
            return true;
        } else {
            return false;
        }
    }

    public boolean isBound() {
        return (subjectReference == null || subjectBinding != null) && inputsBound == inputReferences.size();
    }

    private void fire() {
        onBind.execute(this);
        onBind = null; // let go for gc
    }

    private static <I> ModelBinding<I> bind(ModelReference<I> reference, ModelPath path) {
        return ModelBinding.of(reference, path);
    }

    public Iterable<ModelPath> getUnboundPaths() {
        Set<ModelPath> subjectUnboundPath = Collections.emptySet();
        if (subjectReference != null && subjectReference.getPath() != null && subjectBinding == null) {
            subjectUnboundPath = Collections.singleton(subjectReference.getPath());
        }

        return Iterables.concat(subjectUnboundPath,
                FluentIterable.from(getInputReferences()).filter(new Predicate<ModelReference<?>>() {
                    public boolean apply(final ModelReference<?> reference) {
                        return reference.getPath() != null
                                && !Iterables.any(getInputBindings(), new Predicate<ModelBinding<?>>() {
                                    public boolean apply(@Nullable ModelBinding<?> binding) {
                                        return binding != null && binding.getReference() == reference;
                                    }
                                });
                    }
                }).transform(new Function<ModelReference<?>, ModelPath>() {
                    public ModelPath apply(ModelReference<?> input) {
                        return input.getPath();
                    }
                })

        );
    }

}