package generators; import java.util.AbstractMap; import java.util.ArrayList; import java.util.HashMap; import java.util.HashSet; import java.util.Iterator; import java.util.List; import java.util.Map; import java.util.Map.Entry; import java.util.Set; import algorithms.TypeInference; import code.ast.CodeUtil; import code.ast.CompilationUnit; import code.ast.MethodDeclaration; import code.ast.TypeDeclaration; import code.ast.VariableDeclaration; import models.Edge; import models.algebra.Constant; import models.algebra.Expression; import models.algebra.Field; import models.algebra.InvalidMessage; import models.algebra.Parameter; import models.algebra.ParameterizedIdentifierIsFutureWork; import models.algebra.Position; import models.algebra.Symbol; import models.algebra.Term; import models.algebra.Type; import models.algebra.UnificationFailed; import models.algebra.ValueUndefined; import models.algebra.Variable; import models.dataConstraintModel.Channel; import models.dataConstraintModel.ChannelMember; import models.dataConstraintModel.DataConstraintModel; import models.dataConstraintModel.JsonAccessor; import models.dataConstraintModel.JsonTerm; import models.dataConstraintModel.ResourceHierarchy; import models.dataConstraintModel.ResourcePath; import models.dataConstraintModel.Selector; import models.dataFlowModel.DataTransferModel; import models.dataFlowModel.DataTransferChannel; import models.dataFlowModel.PushPullAttribute; import models.dataFlowModel.PushPullValue; import models.dataFlowModel.ResolvingMultipleDefinitionIsFutureWork; import models.dataFlowModel.ChannelNode; import models.dataFlowModel.DataFlowEdge; import models.dataFlowModel.DataFlowGraph; import models.dataFlowModel.ResourceNode; import models.dataFlowModel.StoreAttribute; import models.dataFlowModel.DataTransferChannel.IResourceStateAccessor; public class JerseyMethodBodyGenerator { private static String baseURL = "http://localhost:8080"; public static ArrayList<CompilationUnit> doGenerate(DataFlowGraph graph, DataTransferModel model, ArrayList<CompilationUnit> codes) { // Create a map from type names (lower case) to their types. Map<String, TypeDeclaration> componentMap = new HashMap<>(); for (CompilationUnit code: codes) { for (TypeDeclaration component: code.types()) { componentMap.put(component.getTypeName(), component); } } // Generate the body of each update or getter method. try { Set<MethodDeclaration> chainedCalls = new HashSet<>(); Map<MethodDeclaration, Set<ResourcePath>> referredResources = new HashMap<>(); for (Edge e: graph.getEdges()) { DataFlowEdge resToCh = (DataFlowEdge) e; if (!resToCh.isChannelToResource()) { PushPullAttribute pushPull = (PushPullAttribute) resToCh.getAttribute(); ResourceNode src = (ResourceNode) resToCh.getSource(); for (Edge chToRes: resToCh.getDestination().getOutEdges()) { ResourceNode dst = (ResourceNode) chToRes.getDestination(); String srcResourceName = JerseyCodeGenerator.getComponentName(src.getResourceHierarchy()); String dstResourceName = JerseyCodeGenerator.getComponentName(dst.getResourceHierarchy()); TypeDeclaration srcComponent = componentMap.get(srcResourceName); TypeDeclaration dstComponent = componentMap.get(dstResourceName); DataTransferChannel ch = ((ChannelNode) resToCh.getDestination()).getChannel(); for (ChannelMember out: ch.getOutputChannelMembers()) { if (dst.getInSideResources().contains(out.getResource())) { // Check if the input resource is outside of the channel scope. boolean outsideInputResource = false; ChannelMember in = null; for (ChannelMember cm: ch.getInputChannelMembers()) { if (src.getOutSideResources().contains(cm.getResource())) { in = cm; if (cm.isOutside()) { outsideInputResource = true; // Regarded as pull transfer. break; } } } // Check if the output resource is outside of the channel scope. boolean outsideOutputResource = out.isOutside(); if ((pushPull.getOptions().get(0) == PushPullValue.PUSH && !outsideInputResource) || outsideOutputResource) { // for push data transfer MethodDeclaration update = null; if (dstComponent == null) { String dstParentResourceName = JerseyCodeGenerator.getComponentName(dst.getResourceHierarchy().getParent()); dstComponent = componentMap.get(dstParentResourceName); update = getUpdateMethod(dstComponent, dstResourceName, srcResourceName); } else { update = getUpdateMethod(dstComponent, null, srcResourceName); } if (((StoreAttribute) dst.getAttribute()).isStored()) { // update stored state of dst side resource (when every incoming edge is in push style) Expression updateExp = null; if (ch.getReferenceChannelMembers().size() == 0) { updateExp = ch.deriveUpdateExpressionOf(out, JerseyCodeGenerator.pushAccessor).getKey(); } else { // if there exists one or more reference channel member. HashMap<ChannelMember, IResourceStateAccessor> inputResourceToStateAccessor = new HashMap<>(); for (Edge chToRes2: dst.getInEdges()) { DataTransferChannel ch2 = ((ChannelNode) chToRes2.getSource()).getChannel(); for (Edge resToCh2: chToRes2.getSource().getInEdges()) { DataFlowEdge dIn = (DataFlowEdge) resToCh2; ChannelMember in2 = null; for (ChannelMember cm: ch2.getInputChannelMembers()) { if (((ResourceNode) dIn.getSource()).getOutSideResources().contains(cm.getResource())) { in2 = cm; break; } } inputResourceToStateAccessor.put(in2, JerseyCodeGenerator.pushAccessor); } } for (ChannelMember c: ch.getReferenceChannelMembers()) { inputResourceToStateAccessor.put(c, JerseyCodeGenerator.refAccessor); } updateExp = ch.deriveUpdateExpressionOf(out, JerseyCodeGenerator.pushAccessor, inputResourceToStateAccessor).getKey(); } // Replace Json constructor with a constructor of a descendant resource. ResourceHierarchy outRes = out.getResource().getResourceHierarchy(); if (outRes.getChildren().size() == 1 && outRes.getChildren().iterator().next().getNumParameters() > 0) { ResourceHierarchy descendantRes = outRes.getChildren().iterator().next(); Set<ResourceHierarchy> children; do { if (JerseyCodeGenerator.generatesComponent(descendantRes)) break; children = descendantRes.getChildren(); } while (children != null && children.size() == 1 && (descendantRes = children.iterator().next()) != null); Type descendantStateType = descendantRes.getResourceStateType(); String descendantComponentName = JerseyCodeGenerator.getComponentName(descendantRes); TypeDeclaration descendantComponent = componentMap.get(descendantComponentName); if (DataConstraintModel.typeJson.isAncestorOf(descendantStateType)) { replaceJsonTermWithConstructorInvocation(updateExp, descendantStateType, descendantComponentName, descendantComponent); } } // Replace the type of the state field. Type fieldType = JerseyCodeGenerator.getImplStateType(outRes); if (updateExp instanceof Term) { ((Term) updateExp).setType(fieldType); for (Map.Entry<Position, Variable> varEnt: ((Term) updateExp).getVariables().entrySet()) { if (varEnt.getValue().getName().equals("value")) { varEnt.getValue().setType(fieldType); } } } else if (updateExp instanceof Variable) { ((Variable) updateExp).setType(fieldType); } // Add statements to the update method. String[] sideEffects = new String[] {""}; String newState = updateExp.toImplementation(sideEffects); int numOfOutResourcesWithTheSameHierarchy = 0; for (ResourcePath outResPath: ch.getOutputResources()) { if (outResPath.getResourceHierarchy().equals(outRes)) { numOfOutResourcesWithTheSameHierarchy++; } } String updateStatement = ""; if (JerseyCodeGenerator.generatesComponent(outRes)) { if (updateExp instanceof Term && ((Term) updateExp).getSymbol().isImplWithSideEffect()) { updateStatement = sideEffects[0]; } else { updateStatement = sideEffects[0] + "this.value = " + newState + ";"; } } else { if (sideEffects[0] != null) { updateStatement = sideEffects[0]; updateStatement = updateStatement.replace(".value", "." + JerseyCodeGenerator.toVariableName(JerseyCodeGenerator.getComponentName(outRes))); } if (DataConstraintModel.typeList.isAncestorOf(outRes.getParent().getResourceStateType())) { Term selector = new Term(DataConstraintModel.set); selector.addChild(new Field("value")); selector.addChild(new Variable(update.getParameters().get(update.getParameters().size() - 2).getName())); selector.addChild(new Constant(newState)); String[] sideEffects2 = new String[] {""}; String newList = selector.toImplementation(sideEffects2); updateStatement += sideEffects2[0]; } else if (DataConstraintModel.typeMap.isAncestorOf(outRes.getParent().getResourceStateType())) { Term selector = new Term(DataConstraintModel.insert); selector.addChild(new Field("value")); selector.addChild(new Variable(update.getParameters().get(update.getParameters().size() - 2).getName())); selector.addChild(new Constant(newState)); String[] sideEffects2 = new String[] {""}; String newMap = selector.toImplementation(sideEffects2); updateStatement += sideEffects2[0]; } else if (!(updateExp instanceof Term && ((Term) updateExp).getSymbol().isImplWithSideEffect())) { updateStatement += "this." + JerseyCodeGenerator.toVariableName(JerseyCodeGenerator.getComponentName(outRes)) + " = " + newState + ";"; } } // add an update statement of the state of dst side resource. if (numOfOutResourcesWithTheSameHierarchy == 1) { update.addFirstStatement(updateStatement); } else { Term conditions = null; int v = 1; Map<ChannelMember, Entry<ResourcePath, Set<ChannelMember>>> resourcePaths = ch.fillOutsideResourcePaths(out, JerseyCodeGenerator.pushAccessor); for (Expression pathParam: out.getResource().getPathParams()) { if (pathParam instanceof Variable) { String selfParamName = ((Variable) pathParam).getName(); Expression arg = null; for (Selector selector: ch.getAllSelectors()) { if (selector.getExpression() instanceof Variable) { Variable selVar = (Variable) selector.getExpression(); if (selVar.getName().equals(selfParamName)) { arg = selVar; break; } } } if (arg == null) { ResourcePath filledPath = resourcePaths.get(out).getKey(); arg = filledPath.getPathParams().get(v - 1); } Term condition = new Term(DataConstraintModel.eq, new Expression[] { new Parameter("self" + (v > 1 ? v : ""), DataConstraintModel.typeString), arg}); if (conditions == null) { conditions = condition; } else { conditions = new Term(DataConstraintModel.and, new Expression[] { conditions, condition}); } } v++; } String ifStatement = "if (" + conditions.toImplementation(new String[] {""})+ ") {\n"; update.addFirstStatement(ifStatement + "\t" + updateStatement.replace("\n", "\n\t") + "\n}"); } } // Calculate in-degree of the destination resource. Set<ResourceHierarchy> inResources = new HashSet<>(); for (ResourceNode rn: graph.getResourceNodes(out.getResource().getResourceHierarchy())) { // ResourceNodes that have the same ResourceHierarchy. for (Edge chToRes2: rn.getInEdges()) { for (Edge resToCh2: chToRes2.getSource().getInEdges()) { inResources.add(((ResourceNode) resToCh2.getSource()).getResourceHierarchy()); } } } int inDegree = inResources.size(); if (inDegree > 1 || (inDegree == 1 && ch.getInputChannelMembers().iterator().next().getStateTransition().isRightPartial())) { // update a cache of src side resource (when incoming edges are multiple) String cacheStatement = "this." + JerseyCodeGenerator.toVariableName(srcResourceName) + " = " + JerseyCodeGenerator.toVariableName(srcResourceName) + ";"; if (update.getBody() == null || !update.getBody().getStatements().contains(cacheStatement)) { update.addStatement(cacheStatement); } } // For a post/put REST API. if (outsideOutputResource || (in.getResource().getCommonPrefix(out.getResource()) == null && JerseyCodeGenerator.differentTreesAsDifferentServices)) { // Inter-services if (dst.getResourceHierarchy().getParent() != null) { // If not a root resource. TypeDeclaration rootComponent = componentMap.get(JerseyCodeGenerator.getComponentName(dst.getResourceHierarchy().getRoot())); MethodDeclaration update2 = update; update = getMethod(rootComponent, update2.getName()); // get the accessor to the update method. // To make the accessor call the update method. Expression resExp = JerseyCodeGenerator.pullAccessor.getDirectStateAccessorFor(out.getResource(), out.getResource().getRoot()); String args = ""; String delimiter = ""; if (resExp instanceof Term) { // to access the parent if (((Term) resExp).getChildren().size() > 1 && ((Term) resExp).getChild(1) instanceof Variable) { args += delimiter + ((Variable)((Term) resExp).getChild(1)).getName(); delimiter = ", "; } resExp = ((Term) resExp).getChild(0); } String resourceAccess = resExp.toImplementation(new String[] {""}); int v = 0; for (VariableDeclaration var: update2.getParameters()) { if (v < out.getResource().getPathParams().size()) { args += delimiter + ((Variable) out.getResource().getPathParams().get(v)).getName(); } else { args += delimiter + var.getName(); } delimiter = ", "; v++; } update.addStatement(resourceAccess + "." + update2.getName() + "(" + args + ");"); } // to convert a json param to a tuple, pair or map object. for (VariableDeclaration param: update.getParameters()) { Type paramType = param.getType(); String paramName = param.getName(); String paramConverter = ""; if (DataConstraintModel.typeList.isAncestorOf(paramType) && paramType != DataConstraintModel.typeList) { Type compType = TypeInference.getListComponentType(paramType); if (DataConstraintModel.typeTuple.isAncestorOf(compType)) { param.setType(DataConstraintModel.typeListStr); param.setName(paramName + "_json"); paramConverter += paramType.getInterfaceTypeName() + " " + paramName + " = new " + paramType.getImplementationTypeName() + "();\n"; paramConverter += "for (String str: " + param.getName() + ") {\n"; String mapTypeName = convertFromEntryToMapType(compType); paramConverter += "\t" + mapTypeName + " i = new ObjectMapper().readValue(str, HashMap.class);\n"; paramConverter += "\t" + paramName + ".add(" + getCodeForConversionFromMapToTuple(compType, "i") + ");\n"; paramConverter += "}"; update.addThrow("JsonProcessingException"); } else if (DataConstraintModel.typePair.isAncestorOf(compType)) { param.setType(DataConstraintModel.typeListStr); param.setName(paramName + "_json"); paramConverter += paramType.getInterfaceTypeName() + " " + paramName + " = new " + paramType.getImplementationTypeName() + "();\n"; paramConverter += "for (String str: " + param.getName() + ") {\n"; String mapTypeName = convertFromEntryToMapType(compType); paramConverter += "\t" + mapTypeName + " i = new ObjectMapper().readValue(str, HashMap.class);\n"; paramConverter += "\t" + paramName + ".add(" + getCodeForConversionFromMapToPair(compType, "i") + ");\n"; paramConverter += "}"; update.addThrow("JsonProcessingException"); } else if (DataConstraintModel.typeMap.isAncestorOf(compType)) { param.setType(DataConstraintModel.typeListStr); // To do. } } else if (DataConstraintModel.typeTuple.isAncestorOf(paramType)) { param.setType(DataConstraintModel.typeString); param.setName(paramName + "_json"); paramConverter += paramType.getInterfaceTypeName() + " " + paramName + ";\n"; paramConverter += "{\n"; String mapTypeName = convertFromEntryToMapType(paramType); paramConverter += "\t" + mapTypeName + " i = new ObjectMapper().readValue(" + paramName + "_json" + ", HashMap.class);\n"; paramConverter += "\t" + paramName + " = " + getCodeForConversionFromMapToTuple(paramType, "i") + ";\n"; paramConverter += "}"; update.addThrow("JsonProcessingException"); } else if (DataConstraintModel.typePair.isAncestorOf(paramType)) { param.setType(DataConstraintModel.typeString); param.setName(paramName + "_json"); paramConverter += paramType.getInterfaceTypeName() + " " + paramName + ";\n"; paramConverter += "{\n"; String mapTypeName = convertFromEntryToMapType(paramType); paramConverter += "\t" + mapTypeName + " i = new ObjectMapper().readValue(" + paramName + "_json" + ", HashMap.class);\n"; paramConverter += "\t" + paramName + " = " + getCodeForConversionFromMapToPair(paramType, "i") + ";\n"; paramConverter += "}"; update.addThrow("JsonProcessingException"); } else if (DataConstraintModel.typeMap.isAncestorOf(paramType)) { param.setType(DataConstraintModel.typeString); param.setName(paramName + "_json"); paramConverter += paramType.getInterfaceTypeName() + " " + paramName + " = " + "new " + paramType.getImplementationTypeName() + "();\n"; paramConverter += "{\n"; String mapTypeName = convertFromEntryToMapType(paramType); paramConverter += "\t" + mapTypeName + " i = new ObjectMapper().readValue(" + paramName + "_json" + ", HashMap.class);\n"; paramConverter += "\t" + getCodeForConversionFromMapToMap(paramType, "i", paramName) + "\n"; paramConverter += "}"; update.addThrow("JsonProcessingException"); } if (paramConverter.length() > 0 && !update.getBody().getStatements().contains(paramConverter)) { update.addFirstStatement(paramConverter); } } } if (((StoreAttribute) dst.getAttribute()).isStored()) { // returns the state stored in a field. MethodDeclaration getter = null; if (JerseyCodeGenerator.generatesComponent(dst.getResourceHierarchy())) { getter = getMethod(dstComponent, "getValue"); } else { getter = getGetterMethod(dstComponent, dstResourceName); } if (getter.getBody() == null || getter.getBody().getStatements().size() == 0) { if (dst.getResourceHierarchy().getNumParameters() == 0) { if (JerseyCodeGenerator.generatesComponent(dst.getResourceHierarchy())) { // dst has a component. getter.addStatement("return value;"); } else { // dst has no component. String dstResName = JerseyCodeGenerator.toVariableName(JerseyCodeGenerator.getComponentName(dst.getResourceHierarchy())); getter.addStatement("return " + dstResName + ";"); } } else { if (DataConstraintModel.typeList.isAncestorOf(dst.getParent().getResourceStateType())) { Term selector = new Term(DataConstraintModel.get); selector.addChild(new Field("value")); selector.addChild(dst.getSelectors().get(dst.getSelectors().size() - 1).getExpression()); getter.addStatement("return " + selector.toImplementation(new String[] {}) + ";"); } else if (DataConstraintModel.typeMap.isAncestorOf(dst.getParent().getResourceStateType())) { Term selector = new Term(DataConstraintModel.lookup); selector.addChild(new Field("value")); selector.addChild(dst.getSelectors().get(dst.getSelectors().size() - 1).getExpression()); getter.addStatement("return " + selector.toImplementation(new String[] {}) + ";"); } } } } // src side (for a chain of update method invocations) String httpMethod = null; if (out.getStateTransition().isRightUnary()) { httpMethod = "put"; } else { httpMethod = "post"; } String srcName = null; if (srcComponent == null) { String srcParentResourceName = JerseyCodeGenerator.getComponentName(src.getResourceHierarchy().getParent()); srcComponent = componentMap.get(srcParentResourceName); srcName = srcResourceName; } for (MethodDeclaration srcUpdate: getUpdateMethods(srcComponent, srcName)) { if (srcUpdate != null) { List<Map.Entry<Type, Map.Entry<String, String>>> params = new ArrayList<>(); ResourcePath dstRes = out.getResource(); // Values of channel parameters. for (Selector selector: ch.getAllSelectors()) { if (selector.getExpression() instanceof Variable) { Variable selVar = (Variable) selector.getExpression(); params.add(new AbstractMap.SimpleEntry<>(selVar.getType(), new AbstractMap.SimpleEntry<>(selVar.getName(), selVar.getName()))); } } // Value of the source side (input side) resource. String srcFieldName = "this.value"; if (!JerseyCodeGenerator.generatesComponent(src.getResourceHierarchy())) { srcFieldName = JerseyCodeGenerator.toVariableName(srcResourceName); } params.add(new AbstractMap.SimpleEntry<>(src.getResourceStateType(), new AbstractMap.SimpleEntry<>(JerseyCodeGenerator.toVariableName(srcResourceName), srcFieldName))); Set<ResourcePath> referredSet = referredResources.get(srcUpdate); if (ch.getReferenceChannelMembers().size() > 0) { for (ChannelMember rc: ch.getReferenceChannelMembers()) { // For each reference channel member, get the current state of the reference side resource by pull data transfer. ResourcePath ref = rc.getResource(); if (referredSet == null) { referredSet = new HashSet<>(); referredResources.put(srcUpdate, referredSet); } if (!dst.getInSideResources().contains(ref)) { String refResourceName = ref.getLeafResourceName(); Type refResourceType = ref.getResourceStateType(); if (!referredSet.contains(ref)) { referredSet.add(ref); String[] sideEffects = new String[] {""}; if (rc.isOutside()) { List<String> pathParams = new ArrayList<>(); for (Expression pathExp: ref.getPathParams()) { pathParams.add("\" + " + pathExp.toImplementation(sideEffects) + " + \""); } generatePullDataTransfer(srcUpdate, refResourceName, ref.getResourceHierarchy().toResourcePath(pathParams), refResourceType); } else { ResourcePath srcRes = in.getResource(); if (!JerseyCodeGenerator.generatesComponent(srcRes.getResourceHierarchy())) { srcRes = srcRes.getParent(); } Expression refGetter = JerseyCodeGenerator.pullAccessor.getDirectStateAccessorFor(ref, srcRes); String refExp = refGetter.toImplementation(sideEffects); String refTypeName = ref.getResourceStateType().getInterfaceTypeName(); srcUpdate.addFirstStatement(sideEffects[0] + refTypeName + " " + refResourceName + " = " + refExp + ";"); } } // Value of a reference side resource. params.add(new AbstractMap.SimpleEntry<>(refResourceType, new AbstractMap.SimpleEntry<>(refResourceName, refResourceName))); } } } if (outsideOutputResource || (in.getResource().getCommonPrefix(dstRes) == null && JerseyCodeGenerator.differentTreesAsDifferentServices)) { // Inter-servces String[] sideEffects = new String[] {""}; List<String> pathParams = new ArrayList<>(); for (Expression pathExp: dstRes.getPathParams()) { pathParams.add("\" + " + pathExp.toImplementation(sideEffects) + " + \""); } String srcResName = JerseyCodeGenerator.toVariableName(srcResourceName); if (inDegree <= 1) { srcResName = null; } if (!chainedCalls.contains(srcUpdate)) { // The first call to an update method in this method srcUpdate.addStatement(getHttpMethodParamsStatement(srcComponent.getTypeName(), params, true)); srcUpdate.addStatement("String result = " + getHttpMethodCallStatement(baseURL, dstRes.getResourceHierarchy().toResourcePath(pathParams), srcResName, httpMethod)); chainedCalls.add(srcUpdate); } else { // After the second time of call to update methods in this method srcUpdate.addStatement(getHttpMethodParamsStatement(srcComponent.getTypeName(), params, false)); srcUpdate.addStatement("result = " + getHttpMethodCallStatement(baseURL, dstRes.getResourceHierarchy().toResourcePath(pathParams), srcResName, httpMethod)); } srcUpdate.addThrow("JsonProcessingException"); } else { // Inner-service String updateMethodName = null; if (JerseyCodeGenerator.generatesComponent(dst.getResourceHierarchy())) { updateMethodName = "updateFrom" + srcResourceName; } else { updateMethodName = "update" + dstResourceName + "From" + srcResourceName; } String callParams = ""; String delimiter = ""; // Values of path parameters. for (Expression pathParam: dstRes.getPathParams()) { if (pathParam instanceof Variable) { Variable pathVar = (Variable) pathParam; callParams += delimiter + pathVar.getName(); delimiter = ", "; } } // Values of other parameters. for (Map.Entry<Type, Map.Entry<String, String>> paramEnt: params) { callParams += delimiter + paramEnt.getValue().getValue(); delimiter = ", "; } if (srcComponent != dstComponent) { srcUpdate.addStatement("this." + JerseyCodeGenerator.toVariableName(dstResourceName) + "." + updateMethodName + "(" + callParams + ");"); } else { srcUpdate.addStatement("this." + updateMethodName + "(" + callParams + ");"); } if (update != null && update.getThrows() != null && update.getThrows().getExceptions().contains("JsonProcessingException")) { srcUpdate.addThrow("JsonProcessingException"); } } } } for (MethodDeclaration srcInput: getInputMethods(srcComponent, src, model)) { List<Map.Entry<Type, Map.Entry<String, String>>> params = new ArrayList<>(); ResourcePath dstRes = out.getResource(); // Values of channel parameters. for (Selector selector: ch.getAllSelectors()) { if (selector.getExpression() instanceof Variable) { Variable selVar = (Variable) selector.getExpression(); params.add(new AbstractMap.SimpleEntry<>(selVar.getType(), new AbstractMap.SimpleEntry<>(selVar.getName(), selVar.getName()))); } } // Value of the source side (input side) resource. String srcFieldName = "this.value"; if (!JerseyCodeGenerator.generatesComponent(src.getResourceHierarchy())) { srcFieldName = JerseyCodeGenerator.toVariableName(srcResourceName); } params.add(new AbstractMap.SimpleEntry<>(src.getResourceStateType(), new AbstractMap.SimpleEntry<>(JerseyCodeGenerator.toVariableName(srcResourceName), srcFieldName))); Set<ResourcePath> referredSet = referredResources.get(srcInput); for (ChannelMember rc: ch.getReferenceChannelMembers()) { // For each reference channel member, get the current state of the reference side resource by pull data transfer. ResourcePath ref = rc.getResource(); if (referredSet == null) { referredSet = new HashSet<>(); referredResources.put(srcInput, referredSet); } if (!dst.getInSideResources().contains(ref)) { String refResourceName = ref.getLeafResourceName(); Type refResourceType = ref.getResourceStateType(); if (!referredSet.contains(ref)) { referredSet.add(ref); String[] sideEffects = new String[] {""}; if (rc.isOutside()) { List<String> pathParams = new ArrayList<>(); for (Expression pathExp: ref.getPathParams()) { pathParams.add("\" + " + pathExp.toImplementation(sideEffects) + " + \""); } generatePullDataTransfer(srcInput, refResourceName, ref.getResourceHierarchy().toResourcePath(pathParams), refResourceType); } else { ResourcePath srcRes = in.getResource(); if (!JerseyCodeGenerator.generatesComponent(srcRes.getResourceHierarchy())) { srcRes = srcRes.getParent(); } Expression refGetter = JerseyCodeGenerator.pullAccessor.getDirectStateAccessorFor(ref, srcRes); String refExp = refGetter.toImplementation(sideEffects); String refTypeName = ref.getResourceStateType().getInterfaceTypeName(); srcInput.addFirstStatement(sideEffects[0] + refTypeName + " " + ref.getLeafResourceName() + " = " + refExp + ";"); } } // Value of a reference side resource. params.add(new AbstractMap.SimpleEntry<>(refResourceType, new AbstractMap.SimpleEntry<>(refResourceName, refResourceName))); } } if (outsideOutputResource || (in.getResource().getCommonPrefix(dstRes) == null && JerseyCodeGenerator.differentTreesAsDifferentServices)) { // Inter-services String[] sideEffects = new String[] {""}; List<String> pathParams = new ArrayList<>(); for (Expression pathExp: dstRes.getPathParams()) { pathParams.add("\" + " + pathExp.toImplementation(sideEffects) + " + \""); } String srcResName = JerseyCodeGenerator.toVariableName(srcResourceName); if (inDegree <= 1) { srcResName = null; } if (!chainedCalls.contains(srcInput)) { // First call to an update method in this method srcInput.addStatement(getHttpMethodParamsStatement(srcComponent.getTypeName(), params, true)); srcInput.addStatement("String result = " + getHttpMethodCallStatement(baseURL, dstRes.getResourceHierarchy().toResourcePath(pathParams), srcResName, httpMethod)); chainedCalls.add(srcInput); } else { // After the second time of call to update methods in this method srcInput.addStatement(getHttpMethodParamsStatement(srcComponent.getTypeName(), params, false)); srcInput.addStatement("result = " + getHttpMethodCallStatement(baseURL, dstRes.getResourceHierarchy().toResourcePath(pathParams), srcResName, httpMethod)); } srcInput.addThrow("JsonProcessingException"); } else { // Inner-service String updateMethodName = null; if (JerseyCodeGenerator.generatesComponent(dst.getResourceHierarchy())) { updateMethodName = "updateFrom" + srcResourceName; } else { updateMethodName = "update" + dstResourceName + "From" + srcResourceName; } String callParams = ""; String delimiter = ""; // Values of path parameters. for (Expression pathParam: dstRes.getPathParams()) { if (pathParam instanceof Variable) { Variable pathVar = (Variable) pathParam; callParams += delimiter + pathVar.getName(); delimiter = ", "; } } // Values of other parameters. for (Map.Entry<Type, Map.Entry<String, String>> paramEnt: params) { callParams += delimiter + paramEnt.getValue().getValue(); delimiter = ", "; } if (srcComponent != dstComponent) { srcInput.addStatement("this." + JerseyCodeGenerator.toVariableName(dstResourceName) + "." + updateMethodName + "(" + callParams + ");"); } else { srcInput.addStatement("this." + updateMethodName + "(" + callParams + ");"); } if (update != null && update.getThrows() != null && update.getThrows().getExceptions().contains("JsonProcessingException")) { srcInput.addThrow("JsonProcessingException"); } } } } else if ((pushPull.getOptions().get(0) != PushPullValue.PUSH && !outsideOutputResource) || outsideInputResource) { // for pull (or push/pull) data transfer if (dstComponent == null) { String dstParentResourceName = JerseyCodeGenerator.getComponentName(dst.getResourceHierarchy().getParent()); dstComponent = componentMap.get(dstParentResourceName); } MethodDeclaration getter = null; if (JerseyCodeGenerator.generatesComponent(dst.getResourceHierarchy())) { getter = getMethod(dstComponent, "getValue"); } else { getter = getGetterMethod(dstComponent, dstResourceName); } if (getter.getBody() == null || getter.getBody().getStatements().size() == 0) { // generate a return statement. Expression curExp = ch.deriveUpdateExpressionOf(out, JerseyCodeGenerator.pullAccessor).getKey(); // no pull data transfer is included. Map<ChannelMember, Entry<ResourcePath, Set<ChannelMember>>> resourcePaths = ch.fillOutsideResourcePaths(out, JerseyCodeGenerator.pullAccessor); String[] sideEffects = new String[] {""}; String curState = curExp.toImplementation(sideEffects); getter.addStatement(sideEffects[0] + "return " + curState + ";"); // For each reference channel member, get the current state of the reference side resource by pull data transfer. for (ChannelMember rc: ch.getReferenceChannelMembers()) { ResourcePath refRes = rc.getResource(); String refResourceName = refRes.getLeafResourceName(); Type refResourceType = refRes.getResourceStateType(); if (rc.isOutside()) { List<String> pathParams = new ArrayList<>(); for (Expression pathExp: refRes.getPathParams()) { pathParams.add("\" + " + pathExp.toImplementation(sideEffects) + " + \""); } generatePullDataTransfer(getter, refResourceName, refRes.getResourceHierarchy().toResourcePath(pathParams), refResourceType); } else { ResourcePath dstRes = out.getResource(); if (!JerseyCodeGenerator.generatesComponent(dstRes.getResourceHierarchy())) { dstRes = dstRes.getParent(); } Expression refGetter = JerseyCodeGenerator.pullAccessor.getDirectStateAccessorFor(refRes, dstRes); String refExp = refGetter.toImplementation(sideEffects); String refTypeName = refResourceType.getInterfaceTypeName(); getter.addFirstStatement(sideEffects[0] + refTypeName + " " + refResourceName + " = " + refExp + ";"); } } for (Entry<ChannelMember, Entry<ResourcePath, Set<ChannelMember>>> pathEnt: resourcePaths.entrySet()) { ChannelMember cm = pathEnt.getKey(); ResourcePath src2 = pathEnt.getValue().getKey(); // get outside src resource state by pull data transfer. if (cm.isOutside() || src2.getCommonPrefix(dst.getInSideResource(ch)) == null) { Type srcResourceType = src2.getResourceStateType(); List<String> pathParams = new ArrayList<>(); for (Expression pathExp: src2.getPathParams()) { pathParams.add("\" + " + pathExp.toImplementation(sideEffects) + " + \""); } generatePullDataTransfer(getter, src2.getLeafResourceName(), src2.getResourceHierarchy().toResourcePath(pathParams), srcResourceType); } } } // get src resource state by pull data transfer. if (src.getNumberOfParameters() == 0 && src.getOutSideResource(ch).getCommonPrefix(dst.getInSideResource(ch)) == null) { Type srcResourceType = src.getResourceStateType(); List<String> pathParams = new ArrayList<>(); generatePullDataTransfer(getter, src.getResourceName(), src.getResourceHierarchy().toResourcePath(pathParams), srcResourceType); } } } } } } } // for source nodes for (ResourceHierarchy resource: model.getResourceHierarchies()) { String resourceName = JerseyCodeGenerator.getComponentName(resource); TypeDeclaration component = componentMap.get(resourceName); if (JavaCodeGenerator.generatesComponent(resource)) { if (component != null) { // state getter method Type resourceType = JerseyCodeGenerator.getImplStateType(resource); MethodDeclaration stateGetter = getMethod(component, "getValue"); if (stateGetter.getBody() == null || stateGetter.getBody().getStatements().size() == 0) { if (model.isPrimitiveType(resourceType)) { // primitive type stateGetter.addStatement("return value;"); } else { if (resource.getChildren() != null && resource.getChildren().size() == 1 && resource.getChildren().iterator().next().getNumParameters() > 0) { // list or map String implTypeName = resourceType.getImplementationTypeName(); // copy the current state to be returned as a 'value' stateGetter.addStatement("return new " + implTypeName + "(value);"); } else { if (resource.getChildren() == null || resource.getChildren().size() == 0) { // a leaf resource String implTypeName = resourceType.getImplementationTypeName(); stateGetter.addStatement("return new " + implTypeName + "(value);"); } else { Term composer = null; Term composerSub = new Constant(DataConstraintModel.nil); composerSub.setType(DataConstraintModel.typeMap); for (ResourceHierarchy child: resource.getChildren()) { String childTypeName = JerseyCodeGenerator.getComponentName(child); String fieldName = JerseyCodeGenerator.toVariableName(childTypeName); Term childGetter = null; if ((child.getChildren() == null || child.getChildren().size() == 0) && child.getNumParameters() == 0) { // the child is not a class childGetter = new Term(new Symbol("get" + childTypeName, 1, Symbol.Type.METHOD)); childGetter.addChild(new Constant("this")); } else { // the child is a class childGetter = new Term(new Symbol("getValue", 1, Symbol.Type.METHOD)); childGetter.addChild(new Field(fieldName, JerseyCodeGenerator.getImplStateType(child))); } composer = new Term(DataConstraintModel.insert); composer.addChild(composerSub); composer.addChild(new Constant(fieldName, DataConstraintModel.typeString)); // key composer.addChild(childGetter); // value composer.setType(DataConstraintModel.typeMap); composerSub = composer; } composer.setType(stateGetter.getReturnType()); String[] sideEffects = new String[] {null}; String returnValue = composer.toImplementation(sideEffects); if (sideEffects[0] != null) { stateGetter.addStatement(sideEffects[0] + "return " + returnValue+ ";"); } else { stateGetter.addStatement("return " + returnValue+ ";"); } } } } } // descendant getter method if (resource.getChildren().size() > 0) { for (ResourceHierarchy child: resource.getChildren()) { ResourceHierarchy parent = resource; ResourceHierarchy descendant = child; Set<ResourceHierarchy> children; Expression selector; int params = 0; if (DataConstraintModel.typeList.isAncestorOf(parent.getResourceStateType())) { selector = new Field("value"); params++; } else if (DataConstraintModel.typeMap.isAncestorOf(parent.getResourceStateType())) { selector = new Field("value"); params++; } else { String fieldName = JerseyCodeGenerator.getComponentName(descendant); selector = new Field(JerseyCodeGenerator.toVariableName(fieldName)); } do { String methodName = JerseyCodeGenerator.getComponentName(descendant); MethodDeclaration descendantGetter = null; for (MethodDeclaration getter: getGetterMethods(component, methodName)) { if ((getter.getParameters() == null && params == 0) || (getter.getParameters() != null && getter.getParameters().size() == params)) { descendantGetter = getter; } } if (descendantGetter != null) { if (DataConstraintModel.typeList.isAncestorOf(parent.getResourceStateType())) { Term newSelector = new Term(DataConstraintModel.get); newSelector.addChild(selector); newSelector.addChild(new Variable(descendantGetter.getParameters().get(descendantGetter.getParameters().size() - 1).getName())); newSelector.setType(descendantGetter.getReturnType()); selector = newSelector; params++; } else if (DataConstraintModel.typeMap.isAncestorOf(parent.getResourceStateType())) { Term newSelector = new Term(DataConstraintModel.lookup); newSelector.addChild(selector); newSelector.addChild(new Variable(descendantGetter.getParameters().get(descendantGetter.getParameters().size() - 1).getName())); newSelector.setType(descendantGetter.getReturnType()); selector = newSelector; params++; } if (descendantGetter != null && (descendantGetter.getBody() == null || descendantGetter.getBody().getStatements().size() == 0)) { String[] sideEffects = new String[] {""}; String returnValue = selector.toImplementation(sideEffects); if (sideEffects[0] != null) descendantGetter.addStatement(sideEffects[0]); descendantGetter.addStatement("return " + returnValue + ";"); } } if (JerseyCodeGenerator.generatesComponent(descendant)) { // If the descendant generates a component. break; } parent = descendant; children = descendant.getChildren(); } while (children != null && children.size() == 1 && (descendant = children.iterator().next()) != null); } } } } // methods for input events Map<DataTransferChannel, Set<ChannelMember>> ioChannelsAndMembers = getIOChannelsAndMembers(resource, model); for (Map.Entry<DataTransferChannel, Set<ChannelMember>> entry: ioChannelsAndMembers.entrySet()) { DataTransferChannel ch = entry.getKey(); Set<ChannelMember> outs = entry.getValue(); for (ChannelMember out: outs) { MethodDeclaration input = null; if (JerseyCodeGenerator.generatesComponent(resource)) { // A component is generated for this resource. input = getInputMethod(component, out, ch.getOutputChannelMembers().size()); } else { // No component is generated for this resource. ResourceHierarchy parent = resource.getParent(); if (parent != null) { TypeDeclaration parentType = componentMap.get(JerseyCodeGenerator.getComponentName(parent)); input = getInputMethod(parentType, out, ch.getOutputChannelMembers().size()); } } if (input != null) { // In each resource Set<ResourcePath> referredSet = referredResources.get(input); for (ChannelMember rc: ch.getReferenceChannelMembers()) { // For each reference channel member, get the current state of the reference side resource by pull data transfer. ResourcePath ref = rc.getResource(); if (referredSet == null) { referredSet = new HashSet<>(); referredResources.put(input, referredSet); } if (!out.getResource().equals(ref)) { String refResourceName = ref.getLeafResourceName(); Type refResourceType = ref.getResourceStateType(); if (!referredSet.contains(ref)) { referredSet.add(ref); String[] sideEffects = new String[] {""}; if (rc.isOutside()) { List<String> pathParams = new ArrayList<>(); for (Expression pathExp: ref.getPathParams()) { pathParams.add("\" + " + pathExp.toImplementation(sideEffects) + " + \""); } generatePullDataTransfer(input, refResourceName, ref.getResourceHierarchy().toResourcePath(pathParams), refResourceType); } else { ResourcePath dstRes = out.getResource(); if (!JerseyCodeGenerator.generatesComponent(dstRes.getResourceHierarchy())) { dstRes = dstRes.getParent(); } Expression refGetter = JerseyCodeGenerator.pullAccessor.getDirectStateAccessorFor(ref, dstRes); String refExp = refGetter.toImplementation(sideEffects); String refTypeName = refResourceType.getInterfaceTypeName(); input.addFirstStatement(sideEffects[0] + refTypeName + " " + refResourceName + " = " + refExp + ";"); } } } } Expression updateExp = ch.deriveUpdateExpressionOf(out, JerseyCodeGenerator.refAccessor).getKey(); // Replace Json constructor with a constructor of a descendant resource. ResourceHierarchy outRes = out.getResource().getResourceHierarchy(); if (outRes.getChildren().size() == 1 && outRes.getChildren().iterator().next().getNumParameters() > 0) { ResourceHierarchy descendantRes = outRes.getChildren().iterator().next(); Set<ResourceHierarchy> children; do { if (JerseyCodeGenerator.generatesComponent(descendantRes)) break; children = descendantRes.getChildren(); } while (children != null && children.size() == 1 && (descendantRes = children.iterator().next()) != null); Type descendantStateType = descendantRes.getResourceStateType(); String descendantComponentName = JerseyCodeGenerator.getComponentName(descendantRes); TypeDeclaration descendantComponent = componentMap.get(descendantComponentName); if (DataConstraintModel.typeJson.isAncestorOf(descendantStateType)) { replaceJsonTermWithConstructorInvocation(updateExp, descendantStateType, descendantComponentName, descendantComponent); } } // Replace the type of the state field. Type fieldType = JerseyCodeGenerator.getImplStateType(outRes); if (updateExp instanceof Term) { ((Term) updateExp).setType(fieldType); for (Map.Entry<Position, Variable> varEnt: ((Term) updateExp).getVariables().entrySet()) { if (varEnt.getValue().getName().equals("value")) { varEnt.getValue().setType(fieldType); } } } else if (updateExp instanceof Variable) { ((Variable) updateExp).setType(fieldType); } // Add statements to the input method. String[] sideEffects = new String[] {""}; String newState = updateExp.toImplementation(sideEffects); if (JerseyCodeGenerator.generatesComponent(resource)) { String updateStatement; if (updateExp instanceof Term && ((Term) updateExp).getSymbol().isImplWithSideEffect()) { updateStatement = sideEffects[0]; } else { updateStatement = sideEffects[0] + "this.value = " + newState + ";"; } if (input.getBody() == null || !input.getBody().getStatements().contains(updateStatement)) { input.addStatement(updateStatement); } } else { String updateStatement = ""; if (sideEffects[0] != null) { updateStatement = sideEffects[0]; updateStatement = updateStatement.replace(".value", "." + JerseyCodeGenerator.toVariableName(JerseyCodeGenerator.getComponentName(resource))); } if (DataConstraintModel.typeList.isAncestorOf(resource.getParent().getResourceStateType())) { Term selector = new Term(DataConstraintModel.set); selector.addChild(new Field("value")); selector.addChild(new Variable(input.getParameters().get(input.getParameters().size() - 2).getName())); selector.addChild(new Constant(newState)); String[] sideEffects2 = new String[] {""}; String newList = selector.toImplementation(sideEffects2); updateStatement += sideEffects2[0]; } else if (DataConstraintModel.typeMap.isAncestorOf(resource.getParent().getResourceStateType())) { Term selector = new Term(DataConstraintModel.insert); selector.addChild(new Field("value")); selector.addChild(new Variable(input.getParameters().get(input.getParameters().size() - 2).getName())); selector.addChild(new Constant(newState)); String[] sideEffects2 = new String[] {""}; String newMap = selector.toImplementation(sideEffects2); updateStatement += sideEffects2[0]; } else if (!(updateExp instanceof Term && ((Term) updateExp).getSymbol().isImplWithSideEffect())) { updateStatement += "this." + JerseyCodeGenerator.toVariableName(JerseyCodeGenerator.getComponentName(resource)) + " = " + newState + ";"; } if (updateStatement != null && (input.getBody() == null || !input.getBody().getStatements().contains(updateStatement))) { input.addStatement(updateStatement); } } if (out.getResource().getParent() != null && out.getResource().getParent().getParent() != null) { // In the root resource Expression message = out.getStateTransition().getMessageExpression(); String inputAccessorName = input.getName(); if (message instanceof Term) { inputAccessorName = ((Term) message).getSymbol().getImplName(); } else if (message instanceof Variable) { inputAccessorName = ((Variable) message).getName(); } MethodDeclaration inputAccessor = getMethod(componentMap.get(JerseyCodeGenerator.getComponentName(resource.getRoot())), inputAccessorName); if (inputAccessor != null) { // The expression of the receiver (resource) of the input method. Expression resExp = JerseyCodeGenerator.pullAccessor.getDirectStateAccessorFor(out.getResource(), out.getResource().getRoot()); String args = ""; String delimiter = ""; if (resExp instanceof Term) { // to access the parent if (((Term) resExp).getChildren().size() > 1 && ((Term) resExp).getChild(1) instanceof Variable) { args += delimiter + ((Variable)((Term) resExp).getChild(1)).getName(); delimiter = ", "; } resExp = ((Term) resExp).getChild(0); } String resourceAccess = resExp.toImplementation(new String[] {""}); // Values of channel parameters. for (Selector selector: ch.getAllSelectors()) { if (selector.getExpression() instanceof Variable) { Variable selVar = (Variable) selector.getExpression(); args += delimiter + selVar.getName(); delimiter = ", "; } } // Values of message parameters. if (message instanceof Term) { for (Variable mesVar: message.getVariables().values()) { args += delimiter + mesVar.getName(); delimiter = ", "; } } inputAccessor.addStatement(resourceAccess + "." + input.getName() + "(" + args + ");"); if (input != null && input.getThrows() != null && input.getThrows().getExceptions().contains("JsonProcessingException")) { inputAccessor.addThrow("JsonProcessingException"); } } } } } } } } catch (ParameterizedIdentifierIsFutureWork | ResolvingMultipleDefinitionIsFutureWork | InvalidMessage | UnificationFailed | ValueUndefined e1) { e1.printStackTrace(); } return codes; } private static void replaceJsonTermWithConstructorInvocation(Expression exp, Type replacedJsonType, String replacingClassName, TypeDeclaration descendantComponent) { Type descendantType = new Type(replacingClassName, replacingClassName); Map<Position, Term> subTerms = ((Term) exp).getSubTerms(Term.class); Iterator<Entry<Position, Term>> termEntItr = subTerms.entrySet().iterator(); while (termEntItr.hasNext()) { Entry<Position, Term> termEnt = termEntItr.next(); Term jsonTerm = termEnt.getValue(); if (jsonTerm.getType() != null) { if (jsonTerm.getType().equals(replacedJsonType)) { if (jsonTerm instanceof JsonTerm || jsonTerm.getSymbol().equals(DataConstraintModel.addMember)) { String constructorInvocation = "new " + replacingClassName + "("; MethodDeclaration descendantConstructor = getConstructor(descendantComponent); String delimiter = ""; for (VariableDeclaration var: descendantConstructor.getParameters()) { JsonAccessor jsonMember = new JsonAccessor(DataConstraintModel.dot); jsonMember.addChild(jsonTerm); jsonMember.addChild(new Constant(var.getName(), DataConstraintModel.typeString)); Expression param = jsonMember.reduce(); if (param != null) { if (param instanceof Term) { if (((Term) param).getType() == null) { ((Term) param).setType(var.getType()); } } else if (param instanceof Variable) { if (((Variable) param).getType() == null) { ((Variable) param).setType(var.getType()); } } constructorInvocation = constructorInvocation + delimiter + param.toImplementation(new String[] {""}); } else { constructorInvocation = constructorInvocation + delimiter + var.getName(); } delimiter = ", "; } constructorInvocation += ")"; ((Term) exp).replaceSubTerm(termEnt.getKey(), new Constant(constructorInvocation)); subTerms = ((Term) exp).getSubTerms(Term.class); termEntItr = subTerms.entrySet().iterator(); } else { jsonTerm.setType(descendantType); } } else { Type oldType = jsonTerm.getType(); Type newType = new Type(oldType.getTypeName(), oldType.getImplementationTypeName().replace(replacedJsonType.getInterfaceTypeName(), replacingClassName), oldType.getInterfaceTypeName().replace(replacedJsonType.getInterfaceTypeName(), replacingClassName)); for (Type parent: oldType.getParentTypes()) { newType.addParentType(parent); } jsonTerm.setType(newType); } } } } private static void generatePullDataTransfer(MethodDeclaration methodBody, String fromResourceName, String fromResourcePath, Type fromResourceType) { String varName = new String(fromResourceName); String respTypeName = fromResourceType.getInterfaceTypeName(); String respImplTypeName = fromResourceType.getImplementationTypeName(); String respConverter = ""; if (DataConstraintModel.typeList.isAncestorOf(fromResourceType) && fromResourceType != DataConstraintModel.typeList) { Type compType = TypeInference.getListComponentType(fromResourceType); if (DataConstraintModel.typeTuple.isAncestorOf(compType)) { varName += "_json"; String mapTypeName = convertFromEntryToMapType(compType); respTypeName = "List<" + mapTypeName + ">"; respConverter += fromResourceType.getInterfaceTypeName() + " " + fromResourceName + " = new " + fromResourceType.getImplementationTypeName() + "();\n"; respConverter += "for (" + mapTypeName + " i: " + varName + ") {\n"; respConverter += "\t" + fromResourceName + ".add(" + getCodeForConversionFromMapToTuple(compType, "i") + ");\n"; respConverter += "}"; methodBody.addThrow("JsonProcessingException"); } else if (DataConstraintModel.typeMap.isAncestorOf(compType)) { // To do. } } else if (DataConstraintModel.typeTuple.isAncestorOf(fromResourceType)) { varName += "_json"; respTypeName = convertFromEntryToMapType(fromResourceType); respConverter += fromResourceType.getInterfaceTypeName() + " " + fromResourceName + " = " + getCodeForConversionFromMapToTuple(fromResourceType, varName) + ";"; respImplTypeName = "HashMap"; } else if (DataConstraintModel.typePair.isAncestorOf(fromResourceType)) { varName += "_json"; respTypeName = convertFromEntryToMapType(fromResourceType); respConverter += fromResourceType.getInterfaceTypeName() + " " + fromResourceName + " = " + getCodeForConversionFromMapToPair(fromResourceType, varName) + ";"; respImplTypeName = "HashMap"; } else if (DataConstraintModel.typeMap.isAncestorOf(fromResourceType)) { varName += "_json"; respTypeName = convertFromEntryToMapType(fromResourceType); respConverter += fromResourceType.getInterfaceTypeName() + " " + fromResourceName + " = new " + fromResourceType.getImplementationTypeName() + "();\n"; respConverter += getCodeForConversionFromMapToMap(fromResourceType, varName, fromResourceName); respImplTypeName = "HashMap"; } if (respConverter.length() > 0) { methodBody.addFirstStatement(respConverter); } methodBody.addFirstStatement(respTypeName + " " + varName + " = " + getHttpMethodCallStatementWithResponse(baseURL, fromResourcePath, "get", respImplTypeName)); } private static String convertFromEntryToMapType(Type type) { String mapTypeName = null; if (DataConstraintModel.typePair.isAncestorOf(type)) { Type compType = TypeInference.getPairComponentType(type); String wrapperType = DataConstraintModel.getWrapperType(compType); if (wrapperType != null) { mapTypeName = "Map<String, " + wrapperType + ">"; } else { mapTypeName = "Map<String, " + compType.getInterfaceTypeName() + ">"; } } else if (DataConstraintModel.typeMap.isAncestorOf(type)) { List<Type> compTypes = TypeInference.getMapComponentTypes(type); String wrapperType = DataConstraintModel.getWrapperType(compTypes.get(1)); if (wrapperType != null) { mapTypeName = "Map<String, " + wrapperType + ">"; } else { mapTypeName = "Map<String, " + compTypes.get(1).getInterfaceTypeName() + ">"; } } else { mapTypeName = type.getInterfaceTypeName(); mapTypeName = mapTypeName.replace("Map.Entry", "Map"); for (int idx = mapTypeName.indexOf("<", 0); idx >= 0; idx = mapTypeName.indexOf("<", idx + 1)) { int to = mapTypeName.indexOf(",", idx); if (to > idx) { mapTypeName = mapTypeName.substring(0, idx + 1) + "String" + mapTypeName.substring(to); // All elements except for the last one have the string type. } } } return mapTypeName; } private static String getCodeForConversionFromMapToTuple(Type tupleType, String mapVar) { String decoded = "$x"; List<Type> elementsTypes = TypeInference.getTupleComponentTypes(tupleType); String elementBase = mapVar; for (Type elmType: elementsTypes.subList(0, elementsTypes.size() - 1)) { elementBase += ".entrySet().iterator().next()"; if (elmType == DataConstraintModel.typeBoolean || elmType == DataConstraintModel.typeInt || elmType == DataConstraintModel.typeLong || elmType == DataConstraintModel.typeFloat || elmType == DataConstraintModel.typeDouble) { String elmVal = CodeUtil.getToValueExp(elmType.getImplementationTypeName(), elementBase + ".getKey()"); decoded = decoded.replace("$x", "new AbstractMap.SimpleEntry<>(" + elmVal + ", $x)"); } else if (elmType == DataConstraintModel.typeString) { decoded = decoded.replace("$x", "new AbstractMap.SimpleEntry<>(" + elementBase + ".getKey(), $x)"); } else { // To do. } elementBase += ".getValue()"; } decoded = decoded.replace("$x", elementBase); return decoded; } private static String getCodeForConversionFromMapToPair(Type pairType, String mapVar) { String decoded = "$x"; decoded = decoded.replace("$x", "new Pair<>(" + mapVar + ".get(\"left\"), $x)"); decoded = decoded.replace("$x", mapVar + ".get(\"right\")"); return decoded; } private static String getCodeForConversionFromMapToMap(Type mapType, String mapVal, String mapVar) { List<Type> elementsTypes = TypeInference.getMapComponentTypes(mapType); Type keyType = elementsTypes.get(0); Type valType = elementsTypes.get(1); String keyVal = null; String decoded = ""; if (keyType == DataConstraintModel.typeBoolean || keyType == DataConstraintModel.typeInt || keyType == DataConstraintModel.typeLong || keyType == DataConstraintModel.typeFloat || keyType == DataConstraintModel.typeDouble) { decoded += "for (String k: " + mapVal + ".keySet()) {\n"; decoded += "\t" + mapVar + ".put("; keyVal = CodeUtil.getToValueExp(keyType.getImplementationTypeName(), "k"); decoded += keyVal + ", " + mapVal + ".get(" + keyVal + ")" + ");\n"; decoded += "}"; } else if (keyType == DataConstraintModel.typeString) { decoded += mapVar + " = " + mapVal + ";"; } return decoded; } private static String getHttpMethodParamsStatement(String callerResourceName, List<Map.Entry<Type, Map.Entry<String, String>>> params, boolean isFirstCall) { String statements = ""; if (isFirstCall) { statements += "Form "; } statements += "form = new Form();\n"; for (Map.Entry<Type, Map.Entry<String, String>> param: params) { Type paramType = param.getKey(); String paramName = param.getValue().getKey(); String value = param.getValue().getValue(); if (DataConstraintModel.typeList.isAncestorOf(paramType)) { Type compType = TypeInference.getListComponentType(paramType); String wrapperType = DataConstraintModel.getWrapperType(compType); if (wrapperType == null) { statements += "for (" + compType.getInterfaceTypeName() + " i: " + value + ") {\n"; } else { statements += "for (" + wrapperType + " i: " + value + ") {\n"; } if (DataConstraintModel.typeTuple.isAncestorOf(compType) || DataConstraintModel.typePair.isAncestorOf(paramType) || DataConstraintModel.typeList.isAncestorOf(compType) || DataConstraintModel.typeMap.isAncestorOf(paramType)) { statements += "\tform.param(\"" + paramName + "\", new ObjectMapper().writeValueAsString(i));\n"; // typeTuple: {"1.0":2.0}, typePair: {"left": 1.0, "right":2.0} } else { statements += "\tform.param(\"" + paramName + "\", i.toString());\n"; } statements += "}\n"; // return "Entity<String> entity = Entity.entity(" + paramName + ".toString(), MediaType.APPLICATION_JSON);"; } else if (DataConstraintModel.typeTuple.isAncestorOf(paramType) || DataConstraintModel.typePair.isAncestorOf(paramType) || DataConstraintModel.typeMap.isAncestorOf(paramType)) { // typeTuple: {"1.0":2.0}, typePair: {"left": 1.0, "right":2.0} statements += "form.param(\"" + paramName + "\", new ObjectMapper().writeValueAsString(" + value + "));\n"; } else { statements += "form.param(\"" + paramName + "\", " + CodeUtil.getToStringExp(paramType.getImplementationTypeName(), value) + ");\n"; } } if (isFirstCall) { statements += "Entity<Form> "; } statements += "entity = Entity.entity(form, MediaType.APPLICATION_FORM_URLENCODED);"; return statements; } private static String getHttpMethodCallStatement(String baseURL, String resourceName, String srcResName, String httpMethod) { if (srcResName == null) { return "client.target(\"" + baseURL + "\").path(\"/" + resourceName + "\").request()." + httpMethod + "(entity, String.class);"; } else { // For each source resource, a child resource is defined in the destination resource so that its state can be updated separately. return "client.target(\"" + baseURL + "\").path(\"/" + resourceName + "/" + srcResName + "\").request()." + httpMethod + "(entity, String.class);"; } } private static String getHttpMethodCallStatementWithResponse(String baseURL, String resourceName, String httpMethod, String respImplName) { String responseShortTypeName = respImplName; if (respImplName.contains("<")) { responseShortTypeName = respImplName.substring(0, respImplName.indexOf("<")); } return "client.target(\"" + baseURL + "\").path(\"/" + resourceName + "\").request()." + httpMethod + "(" + responseShortTypeName + ".class);"; } private static MethodDeclaration getConstructor(TypeDeclaration component) { for (MethodDeclaration m: component.getMethods()) { if (m.isConstructor()) return m; } return null; } private static MethodDeclaration getUpdateMethod(TypeDeclaration component, String dstResName, String srcResName) { for (MethodDeclaration m: component.getMethods()) { if (dstResName == null) { if (m.getName().equals("updateFrom" + srcResName)) return m; } else { if (m.getName().equals("update" + dstResName + "From" + srcResName)) return m; } } return null; } private static List<MethodDeclaration> getUpdateMethods(TypeDeclaration component, String resName) { List<MethodDeclaration> updates = new ArrayList<>(); for (MethodDeclaration m: component.getMethods()) { if (resName == null) { if (m.getName().startsWith("updateFrom")) { updates.add(m); } } else { if (m.getName().startsWith("update" + resName + "From")) { updates.add(m); } } } return updates; } private static MethodDeclaration getGetterMethod(TypeDeclaration component, String resourceName) { for (MethodDeclaration m: component.getMethods()) { if (m.getName().startsWith("get" + resourceName)) return m; } return null; } private static List<MethodDeclaration> getGetterMethods(TypeDeclaration component, String resourceName) { List<MethodDeclaration> getters = new ArrayList<>(); for (MethodDeclaration m: component.getMethods()) { if (m.getName().equals("get" + resourceName)) { getters.add(m); } } return getters; } private static Map<DataTransferChannel, Set<ChannelMember>> getIOChannelsAndMembers(ResourceHierarchy resource, DataTransferModel model) { Map<DataTransferChannel, Set<ChannelMember>> ioChannelsAndMembers = new HashMap<>(); for (Channel c: model.getInputChannels()) { DataTransferChannel ch = (DataTransferChannel) c; // I/O channel for (ChannelMember out: ch.getOutputChannelMembers()) { if (resource.equals(out.getResource().getResourceHierarchy())) { if (out.getStateTransition().getMessageExpression() instanceof Term || out.getStateTransition().getMessageExpression() instanceof Variable) { Set<ChannelMember> channelMembers = ioChannelsAndMembers.get(ch); if (channelMembers == null) { channelMembers = new HashSet<>(); ioChannelsAndMembers.put(ch, channelMembers); } channelMembers.add(out); } } } } return ioChannelsAndMembers; } private static List<MethodDeclaration> getInputMethods(TypeDeclaration component, ResourceNode resource, DataTransferModel model) { List<MethodDeclaration> inputs = new ArrayList<>(); for (Channel c: model.getInputChannels()) { DataTransferChannel channel = (DataTransferChannel) c; // I/O channel for (ChannelMember out: channel.getOutputChannelMembers()) { if (resource.getInSideResources().contains(out.getResource())) { MethodDeclaration input = getInputMethod(component, out, channel.getOutputChannelMembers().size()); inputs.add(input); } } } return inputs; } private static MethodDeclaration getInputMethod(TypeDeclaration component, ChannelMember cm, int outNumber) { String inputMethodName = null; if (cm.getStateTransition().getMessageExpression() instanceof Term) { Term message = (Term) cm.getStateTransition().getMessageExpression(); inputMethodName = message.getSymbol().getImplName(); } else if (cm.getStateTransition().getMessageExpression() instanceof Variable) { Variable message = (Variable) cm.getStateTransition().getMessageExpression(); inputMethodName = message.getName(); } if (outNumber > 1) { inputMethodName += "For" + JerseyCodeGenerator.getComponentName(cm.getResource().getResourceHierarchy()); } MethodDeclaration input = getMethod(component, inputMethodName); return input; } private static MethodDeclaration getMethod(TypeDeclaration component, String methodName) { for (MethodDeclaration m: component.getMethods()) { if (m.getName().equals(methodName)) return m; } return null; } }