Package groovyx.gpars.dataflow.stream

Source Code of groovyx.gpars.dataflow.stream.DataflowStreamWriteAdapter$BindDataflowStream

// GPars - Groovy Parallel Systems
//
// Copyright © 2008-11, 2014  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 groovyx.gpars.dataflow.stream;

import groovyx.gpars.actor.impl.MessageStream;
import groovyx.gpars.dataflow.DataflowReadChannel;
import groovyx.gpars.dataflow.DataflowWriteChannel;
import groovyx.gpars.remote.RemoteConnection;
import groovyx.gpars.remote.RemoteHost;
import groovyx.gpars.serial.SerialMsg;
import groovyx.gpars.serial.WithSerialId;

import java.util.List;

/**
* Adapts a DataflowStream to accommodate for the DataflowWriteChannel interface.
* It also synchronizes all changes to the underlying stream allowing multiple threads accessing the stream concurrently.
*
* @param <T> The type of messages to pass through the stream
* @author Vaclav Pech
*/
@SuppressWarnings({"SynchronizedMethod"})
public class DataflowStreamWriteAdapter<T> extends WithSerialId implements DataflowWriteChannel<T> {

    private StreamCore<T> head;

    /**
     * Creates a new adapter
     *
     * @param stream The stream to wrap
     */
    public DataflowStreamWriteAdapter(final StreamCore<T> stream) {
        this.head = stream;
    }

    @Override
    public final DataflowWriteChannel<T> leftShift(final T value) {
        updateHead().leftShift(value);
        notifyRemote(value);
        return this;
    }

    @Override
    public final DataflowWriteChannel<T> leftShift(final DataflowReadChannel<T> ref) {
        updateHead().leftShift(ref);
        ref.getValAsync(new MessageStream() {
            @Override
            public MessageStream send(Object message) {
                if (!(message instanceof Throwable))
                    notifyRemote((T)message);
                return this;
            }
        });
        return this;
    }

    @Override
    public final void bind(final T value) {
        updateHead().leftShift(value);
        notifyRemote(value);
    }

    /**
     * Moves head
     *
     * @return The old head
     */
    private synchronized StreamCore<T> updateHead() {
        final StreamCore<T> oldHead = head;
        head = (StreamCore<T>) head.getRest();
        return oldHead;
    }

    @Override
    public synchronized String toString() {
        return head.toString();
    }

    protected final synchronized StreamCore<T> getHead() {
        return head;
    }

    private void notifyRemote(T value) {
        if (serialHandle != null) {
            // TODO schedule this job?
            final Object sub = serialHandle.getSubscribers();
            if (sub instanceof RemoteHost) {
                RemoteHost remoteHost = (RemoteHost)sub;
                remoteHost.write(new BindDataflowStream(this, value));
            }
            if (sub instanceof List) {
                List<RemoteHost> subs = (List<RemoteHost>) sub;
                for (RemoteHost subscriber : subs) {
                    subscriber.write(new BindDataflowStream<T>(this, value));
                }
            }
        }
    }

    private class BindDataflowStream<T> extends SerialMsg {
        private DataflowStreamWriteAdapter<T> stream;
        private T value;

        public BindDataflowStream(DataflowStreamWriteAdapter<T> stream, T value) {
            this.stream = stream;
            this.value = value;
        }

        @Override
        public void execute(RemoteConnection conn) {
            stream.leftShift(value);
        }
    }
}
TOP

Related Classes of groovyx.gpars.dataflow.stream.DataflowStreamWriteAdapter$BindDataflowStream

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.