/* Alloy Analyzer 4 -- Copyright (c) 2006-2009, Felix Chang
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files
* (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify,
* merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
* OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
* OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
package edu.mit.csail.sdg.alloy4compiler.ast;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import edu.mit.csail.sdg.alloy4.Pos;
import edu.mit.csail.sdg.alloy4.Err;
import edu.mit.csail.sdg.alloy4.ErrorType;
import edu.mit.csail.sdg.alloy4.ErrorWarning;
import edu.mit.csail.sdg.alloy4.JoinableList;
import static edu.mit.csail.sdg.alloy4compiler.ast.Type.EMPTY;
/** Immutable; represents an illegal relation join.
*
* <p> <b>Invariant:</b> this.type==EMPTY && this.errors.size()>0
*/
public final class ExprBadJoin extends Expr {
/** The left-hand-side expression. */
public final Expr left;
/** The right-hand-side expression. */
public final Expr right;
/** Caches the span() result. */
private Pos span=null;
/** {@inheritDoc} */
@Override public Pos span() {
Pos p=span;
if (p==null) span = (p = pos.merge(closingBracket).merge(right.span()).merge(left.span()));
return p;
}
/** {@inheritDoc} */
@Override public void toString(StringBuilder out, int indent) {
if (indent<0) {
left.toString(out,-1);
out.append('.');
right.toString(out,-1);
} else {
for(int i=0; i<indent; i++) { out.append(' '); }
out.append("ExprBadJoin:\n");
left.toString(out, indent+2);
right.toString(out, indent+2);
}
}
/** Constructs an ExprBadJoin node. */
private ExprBadJoin(Pos pos, Pos closingBracket, Expr left, Expr right, JoinableList<Err> errors) {
super(pos, closingBracket, (left.ambiguous || right.ambiguous), EMPTY, 0, 0, errors);
this.left=left;
this.right=right;
}
/** Constructs an ExprBadJoin node. */
public static Expr make(Pos pos, Pos closingBracket, Expr left, Expr right) {
JoinableList<Err> errors = left.errors.make(right.errors);
if (errors.isEmpty()) {
StringBuilder sb=new StringBuilder("This cannot be a legal relational join where\nleft hand side is ");
left.toString(sb,-1);
sb.append(" (type = ").append(left.type).append(")\nright hand side is ");
right.toString(sb,-1);
sb.append(" (type = ").append(right.type).append(")\n");
errors = errors.make(new ErrorType(pos, sb.toString()));
}
return new ExprBadJoin(pos, closingBracket, left, right, errors);
}
/** {@inheritDoc} */
public int getDepth() {
int a=left.getDepth(), b=right.getDepth();
if (a>=b) return 1+a; else return 1+b;
}
/** {@inheritDoc} */
@Override public Expr resolve(Type t, Collection<ErrorWarning> warns) { return this; }
/** {@inheritDoc} */
@Override final<T> T accept(VisitReturn<T> visitor) throws Err { return visitor.visit(this); }
/** {@inheritDoc} */
@Override public String getDescription() { return "<b>error</b> (parser or typechecker failed)"; }
/** {@inheritDoc} */
@Override public List<? extends Browsable> getSubnodes() { return new ArrayList<Browsable>(0); }
}