1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 package com.sun.star.report.pentaho;
24 
25 
26 import com.sun.star.lang.XServiceInfo;
27 import com.sun.star.lib.uno.helper.ComponentBase;
28 import com.sun.star.lib.uno.helper.PropertySetMixin;
29 import com.sun.star.sheet.FormulaLanguage;
30 import com.sun.star.sheet.FormulaMapGroup;
31 import com.sun.star.sheet.FormulaMapGroupSpecialOffset;
32 import com.sun.star.sheet.FormulaOpCodeMapEntry;
33 import com.sun.star.sheet.FormulaToken;
34 import com.sun.star.sheet.XFormulaOpCodeMapper;
35 import com.sun.star.uno.Any;
36 import com.sun.star.uno.Exception;
37 import com.sun.star.uno.Type;
38 import com.sun.star.uno.UnoRuntime;
39 import com.sun.star.uno.XComponentContext;
40 
41 import java.io.StringReader;
42 
43 import java.util.ArrayList;
44 import java.util.HashMap;
45 import java.util.Iterator;
46 import java.util.List;
47 import java.util.Map;
48 
49 import org.pentaho.reporting.libraries.base.config.Configuration;
50 import org.pentaho.reporting.libraries.formula.DefaultFormulaContext;
51 import org.pentaho.reporting.libraries.formula.function.FunctionRegistry;
52 import org.pentaho.reporting.libraries.formula.parser.FormulaParser;
53 import org.pentaho.reporting.libraries.formula.parser.GeneratedFormulaParserConstants;
54 import org.pentaho.reporting.libraries.formula.parser.GeneratedFormulaParserTokenManager;
55 import org.pentaho.reporting.libraries.formula.parser.JavaCharStream;
56 import org.pentaho.reporting.libraries.formula.parser.ParseException;
57 import org.pentaho.reporting.libraries.formula.parser.Token;
58 import org.pentaho.reporting.libraries.formula.parser.TokenMgrError;
59 
60 
61 public final class SOFormulaParser extends ComponentBase
62         implements com.sun.star.report.meta.XFormulaParser, XServiceInfo
63 {
64 
65     public static final int SEPARATORS = 0;
66     public static final int ARRAY_SEPARATORS = 1;
67     public static final int UNARY_OPERATORS = 2;
68     public static final int BINARY_OPERATORS = 3;
69     public static final int FUNCTIONS = 4;
70     private final XComponentContext m_xContext;
71     private final PropertySetMixin m_prophlp;
72     private static final String __serviceName = "com.sun.star.report.meta.FormulaParser";
73     private static final String OPERATORS = "org.pentaho.reporting.libraries.formula.operators.";
74     // attributes
75     final private List m_OpCodeMap = new ArrayList();
76     private XFormulaOpCodeMapper formulaOpCodeMapper = null;
77     private final Map parserAllOpCodes = new HashMap();
78     private final Map parserNames = new HashMap();
79     private final Map[] groupOpCodes = new HashMap[5];
80     private final List specialOpCodes = new ArrayList();
81 
getSpecialOpCodes()82     public List getSpecialOpCodes()
83     {
84         return specialOpCodes;
85     }
86     private int ownTokenCounter = 1000;
87     private final FormulaOpCodeMapEntry opCodePush;
88     private final FormulaParser parser;
89 
SOFormulaParser(final XComponentContext context)90     public SOFormulaParser(final XComponentContext context)
91     {
92 
93         m_xContext = context;
94         final ClassLoader cl = java.lang.Thread.currentThread().getContextClassLoader();
95         Thread.currentThread().setContextClassLoader(this.getClass().getClassLoader());
96 
97         parser = new FormulaParser();
98         try
99         {
100             final XFormulaOpCodeMapper mapper = (XFormulaOpCodeMapper) UnoRuntime.queryInterface(XFormulaOpCodeMapper.class, m_xContext.getServiceManager().createInstanceWithContext("simple.formula.FormulaOpCodeMapperObj", m_xContext));
101             FormulaOpCodeMapEntry[] opCodes = mapper.getAvailableMappings(FormulaLanguage.ODFF, FormulaMapGroup.FUNCTIONS);
102             final DefaultFormulaContext defaultContext = new DefaultFormulaContext();
103             final FunctionRegistry functionRegistry = defaultContext.getFunctionRegistry();
104 
105             String[] names = functionRegistry.getFunctionNames();
106             addOpCodes(names, opCodes, FUNCTIONS);
107             names = getOperators(defaultContext, OPERATORS);
108             opCodes = mapper.getAvailableMappings(FormulaLanguage.ODFF, FormulaMapGroup.UNARY_OPERATORS);
109             addOpCodes(names, opCodes, UNARY_OPERATORS);
110             opCodes = mapper.getAvailableMappings(FormulaLanguage.ODFF, FormulaMapGroup.BINARY_OPERATORS);
111             addOpCodes(names, opCodes, BINARY_OPERATORS);
112 
113             names = GeneratedFormulaParserConstants.tokenImage.clone();
114             for (int i = 0; i < names.length; i++)
115             {
116                 final String token = names[i];
117                 if (token != null && token.length() > 0 && token.charAt(0) == '"')
118                 {
119                     names[i] = token.substring(1, token.length() - 1);
120                 }
121             }
122             opCodes = mapper.getAvailableMappings(FormulaLanguage.ODFF, FormulaMapGroup.SEPARATORS);
123             addOpCodes(names, opCodes, SEPARATORS, false);
124 
125             opCodes = mapper.getAvailableMappings(FormulaLanguage.ODFF, FormulaMapGroup.ARRAY_SEPARATORS);
126             addOpCodes(names, opCodes, ARRAY_SEPARATORS, false);
127 
128             opCodes = mapper.getAvailableMappings(FormulaLanguage.ODFF, FormulaMapGroup.SPECIAL);
129 
130             for (int i = 0; i < opCodes.length; i++)
131             {
132                 final FormulaOpCodeMapEntry opCode = opCodes[i];
133                 parserAllOpCodes.put(opCode.Token.OpCode, opCode);
134                 specialOpCodes.add(opCode);
135             }
136             // addOpCodes(names, opCodes,SPECIAL,false);
137         }
138         catch (Exception ex)
139         {
140             ex.printStackTrace();
141         }
142         opCodePush = (FormulaOpCodeMapEntry) specialOpCodes.get(FormulaMapGroupSpecialOffset.PUSH);
143         Thread.currentThread().setContextClassLoader(cl);
144         // use the last parameter of the PropertySetMixin constructor
145         // for your optional attributes if necessary. See the documentation
146         // of the PropertySetMixin helper for further information.
147         // Ensure that your attributes are initialized correctly!
148         m_prophlp = new PropertySetMixin(m_xContext, this,
149                 new Type(com.sun.star.report.meta.XFormulaParser.class), null);
150     }
151 
152     // com.sun.star.sheet.XFormulaParser:
parseFormula(String aFormula, com.sun.star.table.CellAddress aReferencePos)153     public com.sun.star.sheet.FormulaToken[] parseFormula(String aFormula, com.sun.star.table.CellAddress aReferencePos)
154     {
155         final ArrayList tokens = new ArrayList();
156         if (!"=".equals(aFormula))
157         {
158             String formula;
159             if (aFormula.charAt(0) == '=')
160             {
161                 formula = aFormula.substring(1);
162             }
163             else
164             {
165                 formula = aFormula;
166             }
167             final ArrayList images = new ArrayList();
168             try
169             {
170                 int brackets = 0;
171                 final GeneratedFormulaParserTokenManager tokenParser = new GeneratedFormulaParserTokenManager(new JavaCharStream(new StringReader(formula), 1, 1));
172                 Token token = tokenParser.getNextToken();
173                 while (token.kind != GeneratedFormulaParserConstants.EOF)
174                 {
175                     final FormulaToken formulaToken;
176                     images.add(token.image);
177                     final String upper = token.image.toUpperCase();
178                     if (parserNames.containsKey(upper))
179                     {
180                         if ("(".equals(token.image))
181                         {
182                             brackets++;
183                         }
184                         else if (")".equals(token.image))
185                         {
186                             --brackets;
187                         }
188                         final FormulaOpCodeMapEntry opCode = (FormulaOpCodeMapEntry) parserNames.get(upper);
189                         formulaToken = opCode.Token;
190                     }
191                     else if (token.kind == GeneratedFormulaParserConstants.WHITESPACE)
192                     {
193                         final FormulaOpCodeMapEntry opCode = (FormulaOpCodeMapEntry) specialOpCodes.get(FormulaMapGroupSpecialOffset.SPACES);
194                         formulaToken = opCode.Token;
195                     }
196                     else
197                     {
198                         formulaToken = new FormulaToken();
199                         formulaToken.OpCode = opCodePush.Token.OpCode;
200                         formulaToken.Data = new Any(Type.STRING, token.image);
201                     }
202 
203                     tokens.add(formulaToken);
204                     token = tokenParser.getNextToken();
205                 }
206                 if (brackets > 0)
207                 {
208                     final FormulaOpCodeMapEntry opCode = (FormulaOpCodeMapEntry) parserNames.get(")");
209                     while (brackets-- != 0)
210                     {
211                         formula = formula.concat(")");
212                         images.add(")");
213                         tokens.add(opCode.Token);
214                     }
215 
216                 }
217 
218                 parser.parse(formula);
219             }
220             catch (ParseException ex)
221             {
222                 boolean found = false;
223                 // error occurred so all token must be bad
224                 for (int i = 0; i < tokens.size(); i++)
225                 {
226                     if (!found && ex.currentToken != null && images.get(i).equals(ex.currentToken.image))
227                     {
228                         found = true;
229                     }
230                     if (found)
231                     {
232                         final FormulaToken dest = new FormulaToken();
233                         dest.OpCode = ((FormulaOpCodeMapEntry) specialOpCodes.get(FormulaMapGroupSpecialOffset.BAD)).Token.OpCode;
234                         dest.Data = new Any(Type.STRING, images.get(i));
235                         tokens.remove(i);
236                         tokens.add(i, dest);
237                     }
238                 }
239             }
240             catch (java.lang.Exception e)
241             {
242             }
243             catch (TokenMgrError e)
244             {
245             }
246         }
247         return (FormulaToken[]) tokens.toArray(new FormulaToken[tokens.size()]);
248     }
249 
printFormula(com.sun.star.sheet.FormulaToken[] aTokens, com.sun.star.table.CellAddress aReferencePos)250     public String printFormula(com.sun.star.sheet.FormulaToken[] aTokens, com.sun.star.table.CellAddress aReferencePos)
251     {
252         final StringBuffer ret = new StringBuffer();
253         for (int i = 0; i < aTokens.length; i++)
254         {
255             final FormulaToken formulaToken = aTokens[i];
256             if (formulaToken.OpCode == opCodePush.Token.OpCode && !formulaToken.Data.equals(Any.VOID))
257             {
258                 ret.append(formulaToken.Data);
259             }
260             else if (parserAllOpCodes.containsKey(formulaToken.OpCode))
261             {
262                 final FormulaOpCodeMapEntry opCode = (FormulaOpCodeMapEntry) parserAllOpCodes.get(formulaToken.OpCode);
263                 if (opCode.Name.length() > 0)
264                 {
265                     ret.append(opCode.Name);
266                 }
267                 else if (!formulaToken.Data.equals(Any.VOID))
268                 {
269                     ret.append(formulaToken.Data);
270                 }
271             }
272         }
273         return ret.toString();
274     }
275 
276     // com.sun.star.beans.XPropertySet:
getPropertySetInfo()277     public com.sun.star.beans.XPropertySetInfo getPropertySetInfo()
278     {
279         return m_prophlp.getPropertySetInfo();
280     }
281 
setPropertyValue(String aPropertyName, Object aValue)282     public void setPropertyValue(String aPropertyName, Object aValue) throws com.sun.star.beans.UnknownPropertyException, com.sun.star.beans.PropertyVetoException, com.sun.star.lang.IllegalArgumentException, com.sun.star.lang.WrappedTargetException
283     {
284         m_prophlp.setPropertyValue(aPropertyName, aValue);
285     }
286 
getPropertyValue(String aPropertyName)287     public Object getPropertyValue(String aPropertyName) throws com.sun.star.beans.UnknownPropertyException, com.sun.star.lang.WrappedTargetException
288     {
289         return m_prophlp.getPropertyValue(aPropertyName);
290     }
291 
addPropertyChangeListener(String aPropertyName, com.sun.star.beans.XPropertyChangeListener xListener)292     public void addPropertyChangeListener(String aPropertyName, com.sun.star.beans.XPropertyChangeListener xListener) throws com.sun.star.beans.UnknownPropertyException, com.sun.star.lang.WrappedTargetException
293     {
294         m_prophlp.addPropertyChangeListener(aPropertyName, xListener);
295     }
296 
removePropertyChangeListener(String aPropertyName, com.sun.star.beans.XPropertyChangeListener xListener)297     public void removePropertyChangeListener(String aPropertyName, com.sun.star.beans.XPropertyChangeListener xListener) throws com.sun.star.beans.UnknownPropertyException, com.sun.star.lang.WrappedTargetException
298     {
299         m_prophlp.removePropertyChangeListener(aPropertyName, xListener);
300     }
301 
addVetoableChangeListener(String aPropertyName, com.sun.star.beans.XVetoableChangeListener xListener)302     public void addVetoableChangeListener(String aPropertyName, com.sun.star.beans.XVetoableChangeListener xListener) throws com.sun.star.beans.UnknownPropertyException, com.sun.star.lang.WrappedTargetException
303     {
304         m_prophlp.addVetoableChangeListener(aPropertyName, xListener);
305     }
306 
removeVetoableChangeListener(String aPropertyName, com.sun.star.beans.XVetoableChangeListener xListener)307     public void removeVetoableChangeListener(String aPropertyName, com.sun.star.beans.XVetoableChangeListener xListener) throws com.sun.star.beans.UnknownPropertyException, com.sun.star.lang.WrappedTargetException
308     {
309         m_prophlp.removeVetoableChangeListener(aPropertyName, xListener);
310     }
311 
312     // com.sun.star.report.meta.XFormulaParser:
getOpCodeMap()313     public com.sun.star.sheet.FormulaOpCodeMapEntry[] getOpCodeMap()
314     {
315         return (com.sun.star.sheet.FormulaOpCodeMapEntry[]) m_OpCodeMap.toArray(new FormulaOpCodeMapEntry[m_OpCodeMap.size()]);
316     }
317 
setOpCodeMap(com.sun.star.sheet.FormulaOpCodeMapEntry[] the_value)318     public void setOpCodeMap(com.sun.star.sheet.FormulaOpCodeMapEntry[] the_value)
319     {
320 //        m_prophlp.prepareSet("OpCodeMap", null);
321 //        synchronized (this)
322 //        {
323 //            m_OpCodeMap.clear();
324 //        }
325     }
326 
getImplementationName()327     public String getImplementationName()
328     {
329         return SOFormulaParser.class.getName();
330     }
331 
supportsService(String sServiceName)332     public boolean supportsService(String sServiceName)
333     {
334         return sServiceName.equals(__serviceName);
335     }
336 
getSupportedServiceNames()337     public String[] getSupportedServiceNames()
338     {
339         return getServiceNames();
340     }
341 
342     /**
343      * This method is a simple helper function to used in the static component initialisation functions as well as
344      * in getSupportedServiceNames.
345      * @return
346      */
getServiceNames()347     public static String[] getServiceNames()
348     {
349         return new String[]
350                 {
351                     __serviceName
352                 };
353     }
354 
getFormulaOpCodeMapper()355     public XFormulaOpCodeMapper getFormulaOpCodeMapper()
356     {
357         if (formulaOpCodeMapper == null)
358         {
359             formulaOpCodeMapper = new SOFormulaOpCodeMapper(this);
360         }
361 
362         return formulaOpCodeMapper;
363     }
364 
addOpCodes(String[] names, FormulaOpCodeMapEntry[] opCodes, int group)365     private void addOpCodes(String[] names, FormulaOpCodeMapEntry[] opCodes, int group)
366     {
367         addOpCodes(names, opCodes, group, true);
368     }
369 
addOpCodes(String[] names, FormulaOpCodeMapEntry[] opCodes, int group, boolean add)370     private void addOpCodes(String[] names, FormulaOpCodeMapEntry[] opCodes, int group, boolean add)
371     {
372         groupOpCodes[group] = new HashMap();
373         for (int j = 0; j < names.length; j++)
374         {
375             FormulaOpCodeMapEntry opCode = null;
376             int i = 0;
377             for (; i < opCodes.length; i++)
378             {
379                 opCode = opCodes[i];
380                 if (names[j].equals(opCode.Name))
381                 {
382                     break;
383                 }
384             }
385             if (i >= opCodes.length)
386             {
387                 if (!add)
388                 {
389                     continue;
390                 }
391                 final FormulaToken token = new FormulaToken(ownTokenCounter++, Any.VOID);
392                 opCode = new FormulaOpCodeMapEntry(names[j], token);
393             }
394             parserNames.put(names[j], opCode);
395             parserAllOpCodes.put(opCode.Token.OpCode, opCode);
396             groupOpCodes[group].put(opCode.Token.OpCode, opCode);
397         }
398     }
399 
getNames()400     public Map getNames()
401     {
402         return parserNames;
403     }
404 
getGroup(int group)405     public Map getGroup(int group)
406     {
407         return groupOpCodes[group];
408     }
409 
getOperators(DefaultFormulaContext defaultContext, final String _kind)410     private String[] getOperators(DefaultFormulaContext defaultContext, final String _kind)
411     {
412         final ArrayList ops = new ArrayList();
413         final Configuration configuration = defaultContext.getConfiguration();
414         final Iterator iter = configuration.findPropertyKeys(_kind);
415         while (iter.hasNext())
416         {
417             final String configKey = (String) iter.next();
418             if (!configKey.endsWith(".class"))
419             {
420                 continue;
421             }
422             final String operatorClass = configuration.getConfigProperty(configKey);
423             if (operatorClass == null)
424             {
425                 continue;
426             }
427             if (operatorClass.length() == 0)
428             {
429                 continue;
430             }
431             final String tokenKey = configKey.substring(0, configKey.length() - ".class".length()) + ".token";
432             final String token = configuration.getConfigProperty(tokenKey);
433             if (token == null)
434             {
435                 continue;
436             }
437             ops.add(token.trim());
438         }
439         return (String[]) ops.toArray(new String[ops.size()]);
440     }
441 }
442 
443