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

com.sun.el.stream.Stream Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2012, 2020 Oracle and/or its affiliates and others.
 * All rights reserved.
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License v. 2.0, which is available at
 * http://www.eclipse.org/legal/epl-2.0.
 *
 * This Source Code may also be made available under the following Secondary
 * Licenses when the conditions for such availability set forth in the
 * Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
 * version 2 with the GNU Classpath Exception, which is available at
 * https://www.gnu.org/software/classpath/license.html.
 *
 * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
 */

package com.sun.el.stream;

import java.util.ArrayList;
import java.util.Comparator;
import java.util.HashSet;
import java.util.Iterator;
import java.util.PriorityQueue;
import java.util.Set;

import jakarta.el.ELException;
import jakarta.el.LambdaExpression;

import com.sun.el.lang.ELArithmetic;
import com.sun.el.lang.ELSupport;

/*
 */

public class Stream {

    private Iterator source;
    private Stream upstream;
    private Operator op;

    Stream(Iterator source) {
        this.source = source;
    }

    Stream(Stream upstream, Operator op) {
        this.upstream = upstream;
        this.op = op;
    }

    public Iterator iterator() {
        if (source != null) {
            return source;
        }

        return op.iterator(upstream.iterator());
    }

    public Stream filter(final LambdaExpression predicate) {
        return new Stream(this, new Operator() {
            @Override
            public Iterator iterator(final Iterator upstream) {
                return new Iterator2(upstream) {
                    @Override
                    public void doItem(Object item) {
                        if ((Boolean) predicate.invoke(item)) {
                            yield(item);
                        }
                    }
                };
            }
        });
    }

    public Stream map(final LambdaExpression mapper) {
        return new Stream(this, new Operator() {
            @Override
            public Iterator iterator(final Iterator up) {
                return new Iterator1(up) {
                    @Override
                    public Object next() {
                        return mapper.invoke(iter.next());
                    }
                };
            }
        });
    }

    public Stream peek(final LambdaExpression comsumer) {
        return new Stream(this, new Operator() {
            @Override
            public Iterator iterator(final Iterator up) {
                return new Iterator2(up) {
                    @Override
                    void doItem(Object item) {
                        comsumer.invoke(item);
                        yield(item);
                    }
                };
            }
        });
    }

    public Stream limit(final long n) {
        if (n < 0) {
            throw new IllegalArgumentException("limit must be non-negative");
        }
        return new Stream(this, new Operator() {
            @Override
            public Iterator iterator(final Iterator up) {
                return new Iterator0() {
                    long limit = n;

                    @Override
                    public boolean hasNext() {
                        return (limit > 0) ? up.hasNext() : false;
                    }

                    @Override
                    public Object next() {
                        limit--;
                        return up.next();
                    }
                };
            }
        });
    }

    public Stream substream(final long startIndex) {
        if (startIndex < 0) {
            throw new IllegalArgumentException("substream index must be non-negative");
        }
        return new Stream(this, new Operator() {
            long skip = startIndex;

            @Override
            public Iterator iterator(final Iterator up) {
                while (skip > 0 && up.hasNext()) {
                    up.next();
                    skip--;
                }
                return up;
            }
        });
    }

    public Stream substream(long startIndex, long endIndex) {
        return substream(startIndex).limit(endIndex - startIndex);
    }

    public Stream distinct() {
        return new Stream(this, new Operator() {
            @Override
            public Iterator iterator(final Iterator up) {
                return new Iterator2(up) {
                    private Set set = new HashSet();

                    @Override
                    public void doItem(Object item) {
                        if (set.add(item)) {
                            yield(item);
                        }
                    }
                };
            }
        });
    }

    public Stream sorted() {
        return new Stream(this, new Operator() {

            private PriorityQueue queue = null;

            @Override
            public Iterator iterator(final Iterator up) {
                if (queue == null) {
                    queue = new PriorityQueue(16, new Comparator() {
                        @Override
                        public int compare(Object o1, Object o2) {
                            @SuppressWarnings("unchecked")
                            int result = ((Comparable) o1).compareTo(o2); 
                            return result;
                        }
                    });

                    while (up.hasNext()) {
                        queue.add(up.next());
                    }
                }

                return new Iterator0() {
                    @Override
                    public boolean hasNext() {
                        return !queue.isEmpty();
                    }

                    @Override
                    public Object next() {
                        return queue.remove();
                    }
                };
            }
        });
    }

    public Stream sorted(final LambdaExpression comparator) {
        return new Stream(this, new Operator() {

            private PriorityQueue queue = null;

            @Override
            public Iterator iterator(final Iterator up) {
                if (queue == null) {
                    queue = new PriorityQueue(16, new Comparator() {
                        @Override
                        public int compare(Object o1, Object o2) {
                            return (Integer) ELSupport.coerceToType(comparator.invoke(o1, o2), Integer.class);
                        }
                    });

                    while (up.hasNext()) {
                        queue.add(up.next());
                    }
                }

                return new Iterator0() {
                    @Override
                    public boolean hasNext() {
                        return !queue.isEmpty();
                    }

                    @Override
                    public Object next() {
                        return queue.remove();
                    }
                };
            }
        });
    }

    public Stream flatMap(final LambdaExpression mapper) {
        return new Stream(this, new Operator() {
            @Override
            public Iterator iterator(final Iterator upstream) {
                return new Iterator0() {
                    Iterator iter = null;

                    @Override
                    public boolean hasNext() {
                        while (true) {
                            if (iter == null) {
                                if (!upstream.hasNext()) {
                                    return false;
                                }
                                Object mapped = mapper.invoke(upstream.next());
                                if (!(mapped instanceof Stream)) {
                                    throw new ELException("Expecting a Stream " + "from flatMap's mapper function.");
                                }
                                iter = ((Stream) mapped).iterator();
                            } else {
                                if (iter.hasNext()) {
                                    return true;
                                }
                                iter = null;
                            }
                        }
                    }

                    @Override
                    public Object next() {
                        if (iter == null) {
                            return null;
                        }
                        return iter.next();
                    }
                };
            }
        });
    }

    public Object reduce(Object base, LambdaExpression op) {
        Iterator iter = iterator();
        while (iter.hasNext()) {
            base = op.invoke(base, iter.next());
        }
        return base;
    }

    public Optional reduce(LambdaExpression op) {
        Iterator iter = iterator();
        if (iter.hasNext()) {
            Object base = iter.next();
            while (iter.hasNext()) {
                base = op.invoke(base, iter.next());
            }
            return new Optional(base);
        }
        return new Optional();
    }

    /*
     * public Map reduceBy(LambdaExpression classifier, LambdaExpression seed, LambdaExpression reducer) {
     * Map map = new HashMap(); Iterator iter = iterator(); while (iter.hasNext()) {
     * Object item = iter.next(); Object key = classifier.invoke(item); Object value = map.get(key); if (value == null) {
     * value = seed.invoke(); } map.put(key, reducer.invoke(value, item)); } return map; }
     */

    public void forEach(LambdaExpression comsumer) {
        Iterator iter = iterator();
        while (iter.hasNext()) {
            comsumer.invoke(iter.next());
        }
    }

    /*
     * public Map> groupBy(LambdaExpression classifier) { Map> map =
     * new HashMap>(); Iterator iter = iterator(); while (iter.hasNext()) { Object item =
     * iter.next(); Object key = classifier.invoke(item); if (key == null) { throw new ELException("null key"); }
     * Collection c = map.get(key); if (c == null) { c = new ArrayList(); map.put(key, c); } c.add(item); }
     * return map; }
     */
    public boolean anyMatch(LambdaExpression predicate) {
        Iterator iter = iterator();
        while (iter.hasNext()) {
            if ((Boolean) predicate.invoke(iter.next())) {
                return true;
            }
        }
        return false;
    }

    public boolean allMatch(LambdaExpression predicate) {
        Iterator iter = iterator();
        while (iter.hasNext()) {
            if (!(Boolean) predicate.invoke(iter.next())) {
                return false;
            }
        }
        return true;
    }

    public boolean noneMatch(LambdaExpression predicate) {
        Iterator iter = iterator();
        while (iter.hasNext()) {
            if ((Boolean) predicate.invoke(iter.next())) {
                return false;
            }
        }
        return true;
    }

    public Object[] toArray() {
        Iterator iter = iterator();
        ArrayList al = new ArrayList();
        while (iter.hasNext()) {
            al.add(iter.next());
        }
        return al.toArray();
    }

    public Object toList() {
        Iterator iter = iterator();
        ArrayList al = new ArrayList();
        while (iter.hasNext()) {
            al.add(iter.next());
        }
        return al;
    }

    /*
     * public Object into(Object target) { if (! (target instanceof Collection)) { throw new
     * ELException("The argument type for into operation mush be a Collection"); } Collection c =
     * (Collection) target; Iterator iter = iterator(); while (iter.hasNext()) { c.add(iter.next()); }
     * return c; }
     */

    public Optional findFirst() {
        Iterator iter = iterator();
        if (iter.hasNext()) {
            return new Optional(iter.next());
        } else {
            return new Optional();
        }
    }

    public Object sum() {
        Number sum = Long.valueOf(0);
        Iterator iter = iterator();
        while (iter.hasNext()) {
            sum = ELArithmetic.add(sum, iter.next());
        }
        return sum;
    }

    public Object count() {
        long count = 0;
        Iterator iter = iterator();
        while (iter.hasNext()) {
            count++;
            iter.next();
        }
        return Long.valueOf(count);
    }

    public Optional min() {
        Object min = null;
        Iterator iter = iterator();
        while (iter.hasNext()) {
            Object item = iter.next();
            if (min == null || ELSupport.compare(min, item) > 0) {
                min = item;
            }
        }
        if (min == null) {
            return new Optional();
        }
        return new Optional(min);
    }

    public Optional max() {
        Object max = null;
        Iterator iter = iterator();
        while (iter.hasNext()) {
            Object item = iter.next();
            if (max == null || ELSupport.compare(max, item) < 0) {
                max = item;
            }
        }
        if (max == null) {
            return new Optional();
        }
        return new Optional(max);
    }

    public Optional min(final LambdaExpression comparator) {
        Object min = null;
        Iterator iter = iterator();
        while (iter.hasNext()) {
            Object item = iter.next();
            if (min == null || ELSupport.compare(comparator.invoke(item, min), Long.valueOf(0)) < 0) {
                min = item;
            }
        }
        if (min == null) {
            return new Optional();
        }
        return new Optional(min);
    }

    public Optional max(final LambdaExpression comparator) {
        Object max = null;
        Iterator iter = iterator();
        while (iter.hasNext()) {
            Object item = iter.next();
            if (max == null || ELSupport.compare(comparator.invoke(max, item), Long.valueOf(0)) < 0) {
                max = item;
            }
        }
        if (max == null) {
            return new Optional();
        }
        return new Optional(max);
    }

    public Optional average() {
        Number sum = Long.valueOf(0);
        long count = 0;
        Iterator iter = iterator();
        while (iter.hasNext()) {
            count++;
            sum = ELArithmetic.add(sum, iter.next());
        }
        if (count == 0) {
            return new Optional();
        }
        return new Optional(ELArithmetic.divide(sum, count));
    }

    abstract class Iterator0 implements Iterator {
        @Override
        public void remove() {
            throw new UnsupportedOperationException();
        }
    }

    abstract class Iterator1 extends Iterator0 {

        Iterator iter;

        Iterator1(Iterator iter) {
            this.iter = iter;
        }

        @Override
        public boolean hasNext() {
            return iter.hasNext();
        }
    }

    abstract class Iterator2 extends Iterator1 {
        private Object current;
        private boolean yielded;

        Iterator2(Iterator upstream) {
            super(upstream);
        }

        @Override
        public Object next() {
            yielded = false;
            return current;
        }

        @Override
        public boolean hasNext() {
            while ((!yielded) && iter.hasNext()) {
                doItem(iter.next());
            }
            return yielded;
        }

        void yield(Object current) {
            this.current = current;
            yielded = true;
        }

        abstract void doItem(Object item);
    }
}