/**
* 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.
*
* See the NOTICE file distributed with this work for additional
* information regarding copyright ownership.
*/
/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
package org.thingml.compilers.checker.genericRules;
import org.eclipse.emf.ecore.util.EcoreUtil;
import org.sintef.thingml.*;
import org.sintef.thingml.constraints.ThingMLHelpers;
import org.sintef.thingml.constraints.Types;
import org.sintef.thingml.helpers.*;
import org.thingml.compilers.checker.Checker;
import org.thingml.compilers.checker.Rule;
/**
*
* @author sintef
*/
public class MessagesUsage extends Rule {
public MessagesUsage() {
super();
}
@Override
public Checker.InfoType getHighestLevel() {
return Checker.InfoType.NOTICE;
}
@Override
public String getName() {
return "Messages Usage";
}
@Override
public String getDescription() {
return "Check that each message declared as to be sent in port declaration can be sent by the state machine.";
}
@Override
public void check(ThingMLModel model, Checker checker) {
for (Thing t : ThingMLHelpers.allThings(model)) {
check(t, checker);
}
}
@Override
public void check(Configuration cfg, Checker checker) {
for (Thing t : ConfigurationHelper.allThings(cfg)) {
check(t, checker);
}
}
private void check(Thing t, Checker checker) {
for (Port p : ThingMLHelpers.allPorts(t)) {
for (Message m : p.getSends()) {
boolean found = false;
for (Action b : ActionHelper.getAllActions(t, SendAction.class)) {
SendAction a = (SendAction) b;
if (EcoreUtil.equals(a.getMessage(), m)) {
found = true;
if (m.getParameters().size() != a.getParameters().size()) {
checker.addGenericError("Message " + m.getName() + " of Thing " + t.getName() + " is sent with wrong number of parameters. Expected " + m.getParameters().size() + ", called with " + a.getParameters().size(), a);
} else {
for (Parameter pa : m.getParameters()) {
Expression e = a.getParameters().get(m.getParameters().indexOf(pa));
Type expected = TyperHelper.getBroadType(pa.getType());
Type actual = checker.typeChecker.computeTypeOf(e);
if (actual != null) {
if (actual.equals(Types.ERROR_TYPE)) {
checker.addGenericError("Message " + m.getName() + " of Thing " + t.getName() + " is sent with an erroneous parameter. Expected " + TyperHelper.getBroadType(expected).getName() + ", called with " + TyperHelper.getBroadType(actual).getName(), a);
} else if (actual.equals(Types.ANY_TYPE)) {
checker.addGenericWarning("Message " + m.getName() + " of Thing " + t.getName() + " is sent with a parameter which cannot be typed.", a);
} else if (!TyperHelper.isA(actual, expected)) {
checker.addGenericError("Message " + m.getName() + " of Thing " + t.getName() + " is sent with an erroneous parameter. Expected " + TyperHelper.getBroadType(expected).getName() + ", called with " + TyperHelper.getBroadType(actual).getName(), a);
}
}
}
}
}
}
if (!found)
checker.addGenericNotice("Port " + p.getName() + " of Thing " + t.getName() + " defines a Message " + m.getName() + " that is never sent.", m);
else {//check if message is serializable
for (Parameter pa : m.getParameters()) {
if ((pa.getType() instanceof ObjectType) && !AnnotatedElementHelper.isDefined(pa, "serializable", "true")) {
checker.addGenericWarning("Message " + m.getName() + " of Thing " + t.getName() + " is not serializable. Parameter " + pa.getName() + " (at least) is not a primitive datatype. If this message is to be sent out on the network, please use only primitive datatypes.", pa);
break;
}
}
}
}
for (Message m : p.getReceives()) {
for (StateMachine sm : ThingMLHelpers.allStateMachines(t)) {
if (StateHelper.allMessageHandlers(sm).get(p) == null || StateHelper.allMessageHandlers(sm).get(p).get(m) == null) {
checker.addGenericNotice("Port " + p.getName() + " of Thing " + t.getName() + " defines a Message " + m.getName() + " that is never received.", m);
}
}
for (Parameter pa : m.getParameters()) {
if ((pa.getType() instanceof ObjectType) && !AnnotatedElementHelper.isDefined(pa, "serializable", "true")) {
checker.addGenericWarning("Message " + m.getName() + " of Thing " + t.getName() + " is not serializable. Parameter " + pa.getName() + " (at least) is not a primitive datatype. If this message is to be received from the network, please use only primitive datatypes.", pa);
break;
}
}
}
}
}
}