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

it.unibo.alchemist.model.implementations.molecules.LsaMolecule Maven / Gradle / Ivy

/*
 * Copyright (C) 2010-2019, Danilo Pianini and contributors listed in the main project's alchemist/build.gradle file.
 *
 * This file is part of Alchemist, and is distributed under the terms of the
 * GNU General Public License, with a linking exception,
 * as described in the file LICENSE in the Alchemist distribution's top directory.
 */
package it.unibo.alchemist.model.implementations.molecules;

import it.unibo.alchemist.expressions.implementations.Expression;
import it.unibo.alchemist.expressions.implementations.ExpressionFactory;
import it.unibo.alchemist.expressions.implementations.ListTreeNode;
import it.unibo.alchemist.expressions.implementations.Type;
import it.unibo.alchemist.expressions.implementations.VarTreeNode;
import it.unibo.alchemist.expressions.interfaces.IExpression;
import it.unibo.alchemist.expressions.interfaces.ITreeNode;
import it.unibo.alchemist.model.interfaces.Dependency;
import it.unibo.alchemist.model.interfaces.ILsaMolecule;
import org.danilopianini.lang.HashString;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Spliterator;
import java.util.function.Consumer;

/**
 *         This class realizes an LsaMolecule, where arguments are of type
 *         Expression.
 * 
 */
public final class LsaMolecule extends SimpleMolecule implements ILsaMolecule {

    private static final String OPEN_SYMBOL = "<", CLOSE_SYMBOL = ">", SEPARATOR = ", ";
    private static final long serialVersionUID = -2727376723102146271L;
    private static final Map> SMAP =
            Collections.unmodifiableMap(new LinkedHashMap<>(0, 1f));
    /**
     * Synthetic property representing the distance.
     */
    public static final HashString SYN_D = new HashString("#D");
    /**
     * Synthetic property representing an LSA ID.
     */
    public static final HashString SYN_MOL_ID = new HashString("#ID");
    /**
     * Synthetic property representing the current neighborhood.
     */
    public static final HashString SYN_NEIGH = new HashString("#NEIGH");
    /**
     * Synthetic property representing the local node id.
     */
    public static final HashString SYN_NODE_ID = new HashString("#NODE");
    /**
     * Synthetic property representing the orientation.
     */
    public static final HashString SYN_O = new HashString("#O");
    /**
     * Synthetic property representing a random value.
     */
    public static final HashString SYN_RAND = new HashString("#RANDOM");
    /**
     * Synthetic property representing the distance. If the environment does not
     * support route computation, it falls back to SYN_D.
     */
    public static final HashString SYN_ROUTE = new HashString("#ROUTE");
    /**
     * Synthetic property representing the current selected neighbor ("+"
     * conditions on the left).
     */
    public static final HashString SYN_SELECTED = new HashString("#SELECTEDNEIGH");
    /**
     * Synthetic property representing the current simulation time.
     */
    public static final HashString SYN_T = new HashString("#T");

    private final List args;
    private final boolean duplicateVars, instance;
    @Nullable
    private HashString repr;

    /**
     * Empty molecule, no arguments.
     */
    public LsaMolecule() {
        this(new ArrayList<>(0));
    }

    /**
     * Builds a new LsaMolecule by interpreting a list of IExpressions.
     * Dramatically faster than parsing, slower than copy.
     * 
     * @param listArgs
     *            the list of IExpressions
     */
    public LsaMolecule(@Nonnull final List listArgs) {
        this(listArgs, buildString(listArgs));
    }

    private LsaMolecule(final List listArgs, @Nonnull final HashString hash) {
        this(listArgs, hash, selfVariableUsed(listArgs), computeInstance(listArgs));
    }

    private LsaMolecule(
            final List argumentList,
            @Nonnull final HashString hash,
            final boolean duplicateVariables,
            final boolean isInstance
    ) {
        super(hash);
        this.repr = hash;
        args = Collections.unmodifiableList(argumentList);
        duplicateVars = duplicateVariables;
        instance = isInstance;
    }

    /**
     * Very fast constructor, produces a copy of an LsaMolecule. Use whenever
     * possible.
     * 
     * @param m
     *            the LsaMolecule to copy
     */
    @SuppressWarnings("CopyConstructorMissesField")
    public LsaMolecule(final LsaMolecule m) {
        this(m.args, m.toHashString(), m.duplicateVars, m.instance);
    }

    /**
     * Builds a LsaMolecule by parsing the passed String. Slow, use only if
     * strictly needed.
     * 
     * @param argsString
     *            the String to parse
     */
    public LsaMolecule(final String argsString) {
        this(argsString, null);
    }

    /**
     * Builds a LsaMolecule by parsing the passed String. Slow, use only if
     * strictly needed.
     * 
     * @param argsString
     *            the String to parse
     * @param description
     *            a String to append at the end of the LSA. This is a special
     *            item, and can carry any type of String. It will be treated
     *            internally as a single literal or variable
     */
    public LsaMolecule(final String argsString, final String description) {
        super(buildString(buildArgsDesc(argsString, description)));
        args = Collections.unmodifiableList(buildArgsDesc(argsString, description));
        duplicateVars = selfVariableUsed(args);
        instance = computeInstance(args);
    }

    @Override
    public List allocateVar(final Map> matches) {
        if (matches == null) {
            return new ArrayList<>(args);
        }
        final List l = new ArrayList<>(args.size());
        for (final IExpression arg : args) {
            /*
             * Try to instance every part of the molecule
             */
            l.add(new Expression(arg.updateMatchedVar(matches)));
        }
        return l;
    }

    @Override
    public int argsNumber() {
        return args.size();
    }

    @Override
    public int compareTo(final ILsaMolecule o) {
        return args.size() - o.argsNumber();
    }

    @Override
    public boolean dependsOn(final Dependency m) {
        if (m instanceof ILsaMolecule) {
            final ILsaMolecule mol = (ILsaMolecule) m;
            if (mol.argsNumber() != argsNumber()) {
                return false;
            }
            for (int i = 0; i < argsNumber(); i++) {
                if (!args.get(i).mayMatch(mol.getArg(i))) {
                    return false;
                }
            }
            return true;
        } else {
            return false;
        }
    }

    @Override
    public boolean equals(final Object o) {
        return o instanceof LsaMolecule && super.equals(o);
    }

    @Override
    public void forEach(final Consumer action) {
        args.forEach(action);
    }

    @Override
    public ILsaMolecule generalize() {
        final List nl = new ArrayList<>(size());
        int i = 0;
        for (final IExpression e : this) {
            switch (e.getRootNodeType()) {
            case NUM:
            case VAR:
            case CONST:
                nl.add(e);
                break;
            default:
                nl.add(new Expression(new VarTreeNode(new HashString("VAR" + i++))));
            }
        }
        return new LsaMolecule(nl);
    }

    @Override
    public IExpression getArg(final int i) {
        return args.get(i);
    }

    /**
     * @return the list of the arguments. Warning: this backs the internal
     *         representation of the LsaMolecule (which should be stateless), so
     *         be sure that your subclass does not change the contents. Copying
     *         it is mandatory before modifying.
     */
    protected List getArgList() {
        return args;
    }

    @Override
    public boolean hasDuplicateVariables() {
        return duplicateVars;
    }

    @Override
    public int hashCode() {
        return ~super.hashCode();
    }

    @Override
    public boolean isIdenticalTo(final ILsaMolecule mol) {
        for (int i = 0; i < argsNumber(); i++) {
            final IExpression argument = args.get(i);
            final IExpression molArgument = mol.getArg(i);
            final boolean isComparator = argument.getRootNodeType() == Type.COMPARATOR;
            if (isComparator && argument.getLeftChildren().getData().equals(molArgument.getRootNodeData())
                    || molArgument.getRootNodeType() == Type.COMPARATOR
                        && molArgument.getLeftChildren().getData().equals(argument.getRootNodeData())
            ) {
                return true; // case  5> == 
            } else if (argument.getRootNodeType() != molArgument.getRootNodeType()
                    || !(argument.getRootNodeData().equals(molArgument.getRootNodeData()))
            ) {
                return false;
            }
        }
        return true;
    }

    @Override
    public boolean isIstance() {
        return instance;
    }

    @Override
    public Iterator iterator() {
        return args.iterator();
    }

    @Override
    public boolean matches(final ILsaMolecule mol) {
        if (this == mol // NOPMD: this comparison is intentional
                || mol instanceof LsaMolecule && ((LsaMolecule) mol).toHashString().equals(toHashString())) {
            return true;
        }
        return mol.matches(args, duplicateVars);
    }

    @Override
    public boolean matches(final List mol, final boolean duplicateVariables) {
        if (argsNumber() != mol.size()) {
            return false;
        }
        final Map> map = duplicateVars || duplicateVariables ? new HashMap<>(argsNumber(), 1f) : SMAP;
        for (int i = 0; i < argsNumber(); i++) {
            /*
             * Call matchwith of Expression
             */
            final IExpression a = args.get(i);
            final IExpression tomatch = mol.get(i);
            if (!a.syntacticMatch(tomatch)) {
                final IExpression tempinstance = new Expression(a.updateMatchedVar(map));
                if (!tempinstance.matches(tomatch, map)) {
                    return false;
                }
                if (duplicateVars || duplicateVariables) {
                    final Type tt = tomatch.getRootNodeType();
                    final Type at = a.getRootNodeType();
                    final boolean toMatchIsAssignable = tt.equals(Type.NUM) || tt.equals(Type.CONST) || tt.equals(Type.OPERATOR);
                    if (toMatchIsAssignable && at.equals(Type.VAR)) {
                        map.put((HashString) a.getRootNodeData(), tomatch.calculate(map));
                    } else {
                        final boolean aIsAssignable = at.equals(Type.NUM) || at.equals(Type.CONST) || at.equals(Type.OPERATOR);
                        if (aIsAssignable && tt.equals(Type.VAR)) {
                            map.put((HashString) tomatch.getRootNodeData(), a.calculate(map));
                        }
                    }
                }
            }
        }
        return true;
    }

    @Override
    public boolean moreGenericOf(final ILsaMolecule mol) {
        if (mol.argsNumber() != argsNumber()) {
            return false;
        }
        for (int i = 0; i < argsNumber(); i++) {
            final IExpression a = args.get(i);
            final IExpression tomatch = mol.getArg(i);
            /*
             * If the comparison between arguments make no sense (e.g. two
             * different atoms) or the passed is a variable (and consequently,
             * more generic), then return false
             */
            if (!args.get(i).mayMatch(mol.getArg(i))
                    || a.getRootNodeType() != Type.VAR && tomatch.getRootNodeType() == Type.VAR
            ) {
                return false;
            }
        }
        return true;
    }

    @Override
    public int size() {
        return args.size();
    }

    @Override
    public Spliterator spliterator() {
        return args.spliterator();
    }

    @Override
    public HashString toHashString() {
        if (repr == null) {
            repr = buildString(args);
        }
        return repr;
    }

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

    private static List buildArgsDesc(final String argsString, final String description) {
        final String[] listArgs = argsString.split(",");
        final boolean hasDescription = description != null && description.length() > 0;
        final List args = new ArrayList<>(listArgs.length + (hasDescription ? 1 : 0));
        for (final String listArg : listArgs) {
            args.add(new Expression(listArg));
        }
        if (hasDescription) {
            args.add(ExpressionFactory.buildComplexGroundExpression(description));
        }
        return args;
    }

    private static HashString buildString(final List expList) {
        final StringBuilder output = new StringBuilder(OPEN_SYMBOL);
        for (int i = 0; i < expList.size(); i++) {
            output.append(expList.get(i).toString());
            if (i < expList.size() - 1) {
                output.append(SEPARATOR);
            }
        }
        output.append(CLOSE_SYMBOL);
        return new HashString(output.toString());
    }

    private static boolean computeInstance(final List e) {
        for (final IExpression exp : e) {
            final Type t = exp.getRootNodeType();
            if (isVarType(t)) {
                return false;
            } else if (t == Type.LIST) {
                for (final ITreeNode ln : ((ListTreeNode) exp.getRootNode()).getData()) {
                    if (isVarType(ln.getType())) {
                        return false;
                    }
                }
            }
        }
        return true;
    }

    private static boolean containVars(final ITreeNode e, final List l) {
        if (e.getType() == Type.VAR) {
            final HashString fs = e.toHashString();
            if (l.contains(fs)) {
                return true;
            } else {
                l.add(e.toHashString());
                return false;
            }
        } else if (e.getLeftChild() != null) {
            if (e.getRightChild() != null) {
                return containVars(e.getRightChild(), l) || containVars(e.getLeftChild(), l);
            }
            return containVars(e.getLeftChild(), l);
        }
        return false;
    }

    private static boolean isVarType(final Type t) {
        return t == Type.VAR || t == Type.COMPARATOR || t == Type.LISTCOMPARATOR || t == Type.OPERATOR;
    }

    private static boolean selfVariableUsed(final List expList) {
        final List foundVars = new ArrayList<>(expList.size());
        for (final IExpression e : expList) {
            if (containVars(e.getRootNode(), foundVars)) {
                return true;
            }
        }
        return false;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy