package org.jruby.ir.operands;

import org.jruby.Ruby;
import org.jruby.RubyHash;
import org.jruby.ir.IRVisitor;
import org.jruby.ir.persistence.IRReaderDecoder;
import org.jruby.ir.persistence.IRWriterEncoder;
import org.jruby.ir.transformations.inlining.CloneInfo;
import org.jruby.parser.StaticScope;
import org.jruby.runtime.DynamicScope;
import org.jruby.runtime.ThreadContext;
import org.jruby.runtime.builtin.IRubyObject;
import org.jruby.util.KeyValuePair;

import java.util.Arrays;
import java.util.List;
import java.util.Map;

// Represents a hash { _ =>_, _ => _ .. } in ruby
//
// NOTE: This operand is only used in the initial stages of optimization.
// Further down the line, this hash could get converted to calls
// that actually build the hash
public class Hash extends Operand {
    final public KeyValuePair<Operand, Operand>[] pairs;

    // Is this a hash used to represent a keyword hash to be setup for ZSuper?
    // SSS FIXME: Quick hack for now - this should probably be done with an overloaded operand.
    final public boolean isKWArgsHash;

    public Hash(List<KeyValuePair<Operand, Operand>> pairs, boolean isKWArgsHash) {
        this(pairs.toArray(new KeyValuePair[pairs.size()]), isKWArgsHash);
    }

    protected Hash(KeyValuePair<Operand, Operand>[] pairs, boolean isKWArgsHash) {
        super();

        this.pairs = pairs;
        this.isKWArgsHash = isKWArgsHash;
    }

    @Override
    public OperandType getOperandType() {
        return OperandType.HASH;
    }

    public Hash(List<KeyValuePair<Operand, Operand>> pairs) {
        this(pairs, false);
    }

    public boolean isBlank() {
        return pairs == null || pairs.length == 0;
    }

    @Override
    public boolean hasKnownValue() {
        for (KeyValuePair<Operand, Operand> pair : pairs) {
            if (!pair.getKey().hasKnownValue() || !pair.getValue().hasKnownValue())
                return false;
        }

        return true;
    }

    @Override
    public Operand getSimplifiedOperand(Map<Operand, Operand> valueMap, boolean force) {
        KeyValuePair<Operand, Operand>[] newPairs = Arrays.stream(pairs)
                .map(pair ->
                        new KeyValuePair<>(
                                pair.getKey().getSimplifiedOperand(valueMap, force),
                                pair.getValue().getSimplifiedOperand(valueMap, force)))
                .toArray(n -> new KeyValuePair[n]);

        return new Hash(newPairs, isKWArgsHash);
    }

    
Append the list of variables used in this operand to the input list
/** Append the list of variables used in this operand to the input list */
@Override public void addUsedVariables(List<Variable> l) { for (KeyValuePair<Operand, Operand> pair : pairs) { pair.getKey().addUsedVariables(l); pair.getValue().addUsedVariables(l); } } public Operand cloneForLVarDepth(int newDepth) { KeyValuePair<Operand, Operand>[] newPairs = Arrays.stream(pairs) .map(pair -> new KeyValuePair( pair.getKey(), ((DepthCloneable) pair.getValue()).cloneForDepth(newDepth))) .toArray(n -> new KeyValuePair[n]); return new Hash(newPairs, isKWArgsHash); } @Override public Operand cloneForInlining(CloneInfo ii) { if (hasKnownValue()) return this; KeyValuePair<Operand, Operand>[] newPairs = Arrays.stream(pairs) .map(pair -> new KeyValuePair( pair.getKey().cloneForInlining(ii), pair.getValue().cloneForInlining(ii))) .toArray(n -> new KeyValuePair[n]); return new Hash(newPairs, isKWArgsHash); } @Override public Object retrieve(ThreadContext context, IRubyObject self, StaticScope currScope, DynamicScope currDynScope, Object[] temp) { Ruby runtime = context.runtime; RubyHash hash; KeyValuePair<Operand, Operand>[] pairs = this.pairs; int index = 0; if (isKWArgsHash && pairs[0].getKey().equals(Symbol.KW_REST_ARG_DUMMY)) { // Dup the rest args hash and use that as the basis for inserting the non-rest args hash = ((RubyHash) pairs[0].getValue().retrieve(context, self, currScope, currDynScope, temp)).dupFast(context); // Skip the first pair index++; } else { hash = RubyHash.newHash(runtime); } for (int i = index; i < pairs.length; i++) { KeyValuePair<Operand, Operand> pair = pairs[i]; IRubyObject key = (IRubyObject) pair.getKey().retrieve(context, self, currScope, currDynScope, temp); IRubyObject value = (IRubyObject) pair.getValue().retrieve(context, self, currScope, currDynScope, temp); hash.fastASetCheckString(runtime, key, value); } return hash; } @Override public void visit(IRVisitor visitor) { visitor.Hash(this); } @Override public void encode(IRWriterEncoder e) { super.encode(e); e.encode(pairs.length); for (KeyValuePair<Operand, Operand> pair: pairs) { e.encode(pair.getKey()); e.encode(pair.getValue()); } e.encode(isKWArgsHash); } public static Hash decode(IRReaderDecoder d) { int size = d.decodeInt(); KeyValuePair<Operand, Operand> pairs[] = new KeyValuePair[size]; for (int i = 0; i < size; i++) { pairs[i] = new KeyValuePair(d.decodeOperand(), d.decodeOperand()); } return new Hash(pairs, d.decodeBoolean()); } @Override public String toString() { StringBuilder builder = new StringBuilder(); builder.append("{"); if (!isBlank()) { int pairCount = pairs.length; for (int i = 0; i < pairCount; i++) { if (i > 0) { builder.append(", "); } builder.append(pairs[i]); } } builder.append("}"); return builder.toString(); } public List<KeyValuePair<Operand, Operand>> getPairs() { return Arrays.asList(pairs); } }