| /******************************************************************************* |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you 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. |
| *******************************************************************************/ |
| package org.apache.ofbiz.minilang.method.callops; |
| |
| import java.net.MalformedURLException; |
| import java.net.URL; |
| import java.util.ArrayList; |
| import java.util.HashMap; |
| import java.util.List; |
| import java.util.Map; |
| |
| import org.apache.ofbiz.base.location.FlexibleLocation; |
| import org.apache.ofbiz.base.util.Debug; |
| import org.apache.ofbiz.base.util.UtilValidate; |
| import org.apache.ofbiz.base.util.UtilXml; |
| import org.apache.ofbiz.base.util.collections.FlexibleMapAccessor; |
| import org.apache.ofbiz.minilang.MiniLangException; |
| import org.apache.ofbiz.minilang.MiniLangRuntimeException; |
| import org.apache.ofbiz.minilang.MiniLangValidate; |
| import org.apache.ofbiz.minilang.SimpleMethod; |
| import org.apache.ofbiz.minilang.ValidationException; |
| import org.apache.ofbiz.minilang.artifact.ArtifactInfoContext; |
| import org.apache.ofbiz.minilang.method.MethodContext; |
| import org.apache.ofbiz.minilang.method.MethodOperation; |
| import org.w3c.dom.Element; |
| |
| /** |
| * Implements the <call-simple-method> element. |
| * |
| * @see <a href="https://cwiki.apache.org/confluence/display/OFBADMIN/Mini+Language+-+minilang+-+simple-method+-+Reference">Mini-language Reference</a> |
| */ |
| public final class CallSimpleMethod extends MethodOperation { |
| |
| public static final String module = CallSimpleMethod.class.getName(); |
| |
| private final String methodName; |
| private final String xmlResource; |
| private final URL xmlURL; |
| private final String scope; |
| private final List<ResultToField> resultToFieldList; |
| |
| public CallSimpleMethod(Element element, SimpleMethod simpleMethod) throws MiniLangException { |
| super(element, simpleMethod); |
| if (MiniLangValidate.validationOn()) { |
| MiniLangValidate.attributeNames(simpleMethod, element, "method-name", "xml-resource", "scope"); |
| MiniLangValidate.requiredAttributes(simpleMethod, element, "method-name"); |
| MiniLangValidate.constantAttributes(simpleMethod, element, "method-name", "xml-resource", "scope"); |
| MiniLangValidate.childElements(simpleMethod, element, "result-to-field"); |
| } |
| this.methodName = element.getAttribute("method-name"); |
| String xmlResourceAttribute = element.getAttribute("xml-resource"); |
| if (xmlResourceAttribute.isEmpty()) { |
| xmlResourceAttribute = simpleMethod.getFromLocation(); |
| } |
| this.xmlResource = xmlResourceAttribute; |
| URL xmlURL = null; |
| try { |
| xmlURL = FlexibleLocation.resolveLocation(this.xmlResource); |
| } catch (MalformedURLException e) { |
| MiniLangValidate.handleError("Could not find SimpleMethod XML document in resource: " + this.xmlResource + "; error was: " + e.toString(), simpleMethod, element); |
| } |
| this.xmlURL = xmlURL; |
| this.scope = element.getAttribute("scope"); |
| List<? extends Element> resultToFieldElements = UtilXml.childElementList(element, "result-to-field"); |
| if (UtilValidate.isNotEmpty(resultToFieldElements)) { |
| if (!"function".equals(this.scope)) { |
| MiniLangValidate.handleError("Inline scope cannot include <result-to-field> elements.", simpleMethod, element); |
| } |
| List<ResultToField> resultToFieldList = new ArrayList<ResultToField>(resultToFieldElements.size()); |
| for (Element resultToFieldElement : resultToFieldElements) { |
| resultToFieldList.add(new ResultToField(resultToFieldElement, simpleMethod)); |
| } |
| this.resultToFieldList = resultToFieldList; |
| } else { |
| this.resultToFieldList = null; |
| } |
| } |
| |
| @Override |
| public boolean exec(MethodContext methodContext) throws MiniLangException { |
| if (UtilValidate.isEmpty(this.methodName)) { |
| throw new MiniLangRuntimeException("method-name attribute is empty", this); |
| } |
| SimpleMethod simpleMethodToCall = SimpleMethod.getSimpleMethod(this.xmlURL, this.methodName); |
| if (simpleMethodToCall == null) { |
| throw new MiniLangRuntimeException("Could not find <simple-method name=\"" + this.methodName + "\"> in XML document " + this.xmlResource, this); |
| } |
| MethodContext localContext = methodContext; |
| if ("function".equals(this.scope)) { |
| Map<String, Object> localEnv = new HashMap<String, Object>(); |
| localEnv.putAll(methodContext.getEnvMap()); |
| localEnv.remove(this.simpleMethod.getEventResponseCodeName()); |
| localEnv.remove(this.simpleMethod.getServiceResponseMessageName()); |
| localContext = new MethodContext(localEnv, methodContext.getLoader(), methodContext.getMethodType()); |
| } |
| String returnVal = simpleMethodToCall.exec(localContext); |
| if (Debug.verboseOn()) |
| Debug.logVerbose("Called simple-method named [" + this.methodName + "] in resource [" + this.xmlResource + "], returnVal is [" + returnVal + "]", module); |
| if (simpleMethodToCall.getDefaultErrorCode().equals(returnVal)) { |
| if (methodContext.getMethodType() == MethodContext.EVENT) { |
| methodContext.putEnv(simpleMethod.getEventResponseCodeName(), simpleMethod.getDefaultErrorCode()); |
| } else if (methodContext.getMethodType() == MethodContext.SERVICE) { |
| methodContext.putEnv(simpleMethod.getServiceResponseMessageName(), simpleMethod.getDefaultErrorCode()); |
| } |
| return false; |
| } |
| if (methodContext.getMethodType() == MethodContext.EVENT) { |
| // FIXME: This doesn't make sense. We are comparing the called method's response code with this method's |
| // response code. Since response codes are configurable per method, this code will fail. |
| String responseCode = (String) localContext.getEnv(this.simpleMethod.getEventResponseCodeName()); |
| if (this.simpleMethod.getDefaultErrorCode().equals(responseCode)) { |
| Debug.logWarning("Got error [" + responseCode + "] calling inline simple-method named [" + this.methodName + "] in resource [" + this.xmlResource + "], message is " + methodContext.getEnv(this.simpleMethod.getEventErrorMessageName()), module); |
| return false; |
| } |
| } else if (methodContext.getMethodType() == MethodContext.SERVICE) { |
| // FIXME: This doesn't make sense. We are comparing the called method's response message with this method's |
| // response message. Since response messages are configurable per method, this code will fail. |
| String responseMessage = (String) localContext.getEnv(this.simpleMethod.getServiceResponseMessageName()); |
| if (this.simpleMethod.getDefaultErrorCode().equals(responseMessage)) { |
| Debug.logWarning("Got error [" + responseMessage + "] calling inline simple-method named [" + this.methodName + "] in resource [" + this.xmlResource + "], message is " + methodContext.getEnv(this.simpleMethod.getServiceErrorMessageName()) + ", and the error message list is: " |
| + methodContext.getEnv(this.simpleMethod.getServiceErrorMessageListName()), module); |
| return false; |
| } |
| } |
| if ("function".equals(this.scope) && this.resultToFieldList != null) { |
| Map<String, Object> results = localContext.getResults(); |
| if (results != null) { |
| for (ResultToField resultToField : this.resultToFieldList) { |
| resultToField.exec(methodContext.getEnvMap(), results); |
| } |
| } |
| } |
| return true; |
| } |
| |
| @Override |
| public void gatherArtifactInfo(ArtifactInfoContext aic) { |
| SimpleMethod simpleMethodToCall; |
| try { |
| simpleMethodToCall = SimpleMethod.getSimpleMethod(this.xmlURL, this.methodName); |
| if (simpleMethodToCall != null) { |
| if (!aic.hasVisited(simpleMethodToCall)) { |
| aic.addSimpleMethod(simpleMethodToCall); |
| simpleMethodToCall.gatherArtifactInfo(aic); |
| } |
| } |
| } catch (MiniLangException e) { |
| Debug.logWarning("Could not find <simple-method name=\"" + this.methodName + "\"> in XML document " + this.xmlResource + ": " + e.toString(), module); |
| } |
| } |
| |
| public String getMethodName() { |
| return this.methodName; |
| } |
| |
| public String getXmlResource() { |
| return this.xmlResource; |
| } |
| |
| @Override |
| public String toString() { |
| StringBuilder sb = new StringBuilder("<call-simple-method "); |
| if (this.methodName.length() > 0) { |
| sb.append("method-name=\"").append(this.methodName).append("\" "); |
| } |
| if (this.xmlResource.length() > 0) { |
| sb.append("xml-resource=\"").append(this.xmlResource).append("\" "); |
| } |
| if (this.scope.length() > 0) { |
| sb.append("scope=\"").append(this.scope).append("\" "); |
| } |
| sb.append("/>"); |
| return sb.toString(); |
| } |
| |
| /** |
| * A factory for the <call-simple-method> element. |
| */ |
| public static final class CallSimpleMethodFactory implements Factory<CallSimpleMethod> { |
| @Override |
| public CallSimpleMethod createMethodOperation(Element element, SimpleMethod simpleMethod) throws MiniLangException { |
| return new CallSimpleMethod(element, simpleMethod); |
| } |
| |
| @Override |
| public String getName() { |
| return "call-simple-method"; |
| } |
| } |
| |
| private final class ResultToField { |
| |
| private final FlexibleMapAccessor<Object> fieldFma; |
| private final FlexibleMapAccessor<Object> resultNameFma; |
| |
| private ResultToField(Element element, SimpleMethod simpleMethod) throws ValidationException { |
| if (MiniLangValidate.validationOn()) { |
| MiniLangValidate.attributeNames(simpleMethod, element, "result-name", "field"); |
| MiniLangValidate.requiredAttributes(simpleMethod, element, "result-name"); |
| MiniLangValidate.expressionAttributes(simpleMethod, element, "result-name", "field"); |
| MiniLangValidate.noChildElements(simpleMethod, element); |
| } |
| this.resultNameFma = FlexibleMapAccessor.getInstance(element.getAttribute("result-name")); |
| String fieldAttribute = element.getAttribute("field"); |
| if (fieldAttribute.length() == 0) { |
| this.fieldFma = this.resultNameFma; |
| } else { |
| this.fieldFma = FlexibleMapAccessor.getInstance(fieldAttribute); |
| } |
| } |
| |
| private void exec(Map<String, Object> context, Map<String, Object> results) throws MiniLangException { |
| Object value = this.resultNameFma.get(results); |
| if (value != null) { |
| this.fieldFma.put(context, value); |
| } |
| } |
| } |
| } |