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.filter.config.tools.merge;
24 
25 //_______________________________________________
26 
27 import java.lang.*;
28 import java.util.*;
29 import java.io.*;
30 import com.sun.star.filter.config.tools.utils.*;
31 
32 //_______________________________________________
33 
34 /** can merge different xml fragments together.
35  *
36  *
37  */
38 public class Merger
39 {
40     //___________________________________________
41     // const
42 
43     private static final java.lang.String PROP_XMLVERSION      = "xmlversion"               ; // <= global cfg file
44     private static final java.lang.String PROP_XMLENCODING     = "xmlencoding"              ; // <= global cfg file
45     private static final java.lang.String PROP_XMLPATH         = "xmlpath"                  ; // <= global cfg file
46     private static final java.lang.String PROP_XMLPACKAGE      = "xmlpackage"               ; // <= global cfg file
47 
48     private static final java.lang.String PROP_SETNAME_TYPES    = "setname_types"           ; // <= global cfg file
49     private static final java.lang.String PROP_SETNAME_FILTERS  = "setname_filters"         ; // <= global cfg file
50     private static final java.lang.String PROP_SETNAME_LOADERS  = "setname_frameloaders"    ; // <= global cfg file
51     private static final java.lang.String PROP_SETNAME_HANDLERS = "setname_contenthandlers" ; // <= global cfg file
52 
53     private static final java.lang.String PROP_SUBDIR_TYPES    = "subdir_types"             ; // <= global cfg file
54     private static final java.lang.String PROP_SUBDIR_FILTERS  = "subdir_filters"           ; // <= global cfg file
55     private static final java.lang.String PROP_SUBDIR_LOADERS  = "subdir_frameloaders"      ; // <= global cfg file
56     private static final java.lang.String PROP_SUBDIR_HANDLERS = "subdir_contenthandlers"   ; // <= global cfg file
57 
58     private static final java.lang.String PROP_EXTENSION_XCU   = "extension_xcu"            ; // <= global cfg file
59     private static final java.lang.String PROP_EXTENSION_PKG   = "extension_pkg"            ; // <= global cfg file
60 
61     private static final java.lang.String PROP_DELIMITER       = "delimiter"                ; // <= global cfg file
62     private static final java.lang.String PROP_TRIM            = "trim"                     ; // <= global cfg file
63     private static final java.lang.String PROP_DECODE          = "decode"                   ; // <= global cfg file
64 
65     private static final java.lang.String PROP_FRAGMENTSDIR    = "fragmentsdir"             ; // <= cmdline
66     private static final java.lang.String PROP_TEMPDIR         = "tempdir"                  ; // <= cmdline
67     private static final java.lang.String PROP_OUTDIR          = "outdir"                   ; // <= cmdline
68     private static final java.lang.String PROP_PKG             = "pkg"                      ; // <= cmdline
69     private static final java.lang.String PROP_DEBUG           = "debug"                    ; // <= cmdline
70 
71     private static final java.lang.String PROP_TCFG            = "tcfg"                     ; // <= cmdline
72     private static final java.lang.String PROP_FCFG            = "fcfg"                     ; // <= cmdline
73     private static final java.lang.String PROP_LCFG            = "lcfg"                     ; // <= cmdline
74     private static final java.lang.String PROP_CCFG            = "ccfg"                     ; // <= cmdline
75     private static final java.lang.String PROP_LANGUAGEPACK    = "languagepack"             ; // <= cmdline
76 
77     private static final java.lang.String PROP_ITEMS           = "items"                    ; // <= pkg cfg files!
78 
79     //___________________________________________
80     // member
81 
82     //-------------------------------------------
83     /** TODO */
84     private ConfigHelper m_aCfg;
85 
86     //-------------------------------------------
87     /** TODO */
88     private Logger m_aLog;
89 
90     //-------------------------------------------
91     /** TODO */
92     private java.io.File m_aFragmentsDir;
93 
94     //-------------------------------------------
95     /** TODO */
96     private java.io.File m_aTempDir;
97 
98     //-------------------------------------------
99     /** TODO */
100     private java.io.File m_aOutDir;
101 
102     //-------------------------------------------
103     /** TODO */
104     private java.util.Vector m_lTypes;
105     private java.util.Vector m_lFilters;
106     private java.util.Vector m_lLoaders;
107     private java.util.Vector m_lHandlers;
108 
109     //___________________________________________
110     // interface
111 
112     //-------------------------------------------
113     /** initialize a new instance of this class and
114      *  try to get all needed resources from the config module.
115      *
116      *  @param  aCfg
117      *          provides access to all values of the global
118      *          config file and to the command line.
119      *
120      *  @param  aLog
121      *          can be used to print out log informations.
122      */
Merger(ConfigHelper aCfg, Logger aLog)123     public Merger(ConfigHelper aCfg,
124                   Logger       aLog)
125         throws java.lang.Exception
126     {
127         m_aCfg = aCfg;
128         m_aLog = aLog;
129 
130         m_aFragmentsDir = new java.io.File(m_aCfg.getString(PROP_FRAGMENTSDIR));
131         m_aTempDir      = new java.io.File(m_aCfg.getString(PROP_TEMPDIR     ));
132 //        m_aOutDir       = new java.io.File(m_aCfg.getString(PROP_OUTDIR      ));
133 
134         java.lang.String sDelimiter = m_aCfg.getString(PROP_DELIMITER);
135         boolean          bTrim      = m_aCfg.getBoolean(PROP_TRIM);
136         boolean          bDecode    = m_aCfg.getBoolean(PROP_DECODE);
137 
138         try
139         {
140             ConfigHelper aFcfg = new ConfigHelper(m_aCfg.getString(PROP_TCFG), null);
141             m_lTypes = aFcfg.getStringList(PROP_ITEMS, sDelimiter, bTrim, bDecode);
142         }
143         catch(java.util.NoSuchElementException ex1)
144         {
145             m_lTypes = new java.util.Vector();
146             //m_aLog.setWarning("Fragment list of types is missing. Parameter \"items\" seems to be invalid.");
147         }
148 
149         try
150         {
151             ConfigHelper aFcfg = new ConfigHelper(m_aCfg.getString(PROP_FCFG), null);
152             m_lFilters = aFcfg.getStringList(PROP_ITEMS, sDelimiter, bTrim, bDecode);
153         }
154         catch(java.util.NoSuchElementException ex1)
155         {
156             m_lFilters = new java.util.Vector();
157             //m_aLog.setWarning("Fragment list of filters is missing. Parameter \"items\" seems to be invalid.");
158         }
159 
160         try
161         {
162             ConfigHelper aFcfg = new ConfigHelper(m_aCfg.getString(PROP_LCFG), null);
163             m_lLoaders = aFcfg.getStringList(PROP_ITEMS, sDelimiter, bTrim, bDecode);
164         }
165         catch(java.util.NoSuchElementException ex1)
166         {
167             m_lLoaders = new java.util.Vector();
168             //m_aLog.setWarning("Fragment list of frame loader objects is missing. Parameter \"items\" seems to be invalid.");
169         }
170 
171         try
172         {
173             ConfigHelper aFcfg = new ConfigHelper(m_aCfg.getString(PROP_CCFG), null);
174             m_lHandlers = aFcfg.getStringList(PROP_ITEMS, sDelimiter, bTrim, bDecode);
175         }
176         catch(java.util.NoSuchElementException ex1)
177         {
178             m_lHandlers = new java.util.Vector();
179             //m_aLog.setWarning("Fragment list of content handler objects is missing. Parameter \"items\" seems to be invalid.");
180         }
181     }
182 
183     //-------------------------------------------
184     /** TODO */
merge()185     public synchronized void merge()
186         throws java.lang.Exception
187     {
188         java.lang.StringBuffer sBuffer  = new java.lang.StringBuffer(1000000);
189         java.lang.String       sPackage = m_aCfg.getString(PROP_PKG);
190 
191         m_aLog.setGlobalInfo("create package \""+sPackage+"\" ...");
192         m_aLog.setDetailedInfo("generate package header ... ");
193 
194         sBuffer.append(
195             XMLHelper.generateHeader(
196                 m_aCfg.getString (PROP_XMLVERSION         ),
197                 m_aCfg.getString (PROP_XMLENCODING        ),
198                 m_aCfg.getString (PROP_XMLPATH            ),
199                 m_aCfg.getString (PROP_XMLPACKAGE         ),
200                 m_aCfg.getBoolean(PROP_LANGUAGEPACK, false)));
201 
202         // counts all transferred fragments
203         // Can be used later to decide, if a generated package file
204         // contains "nothing"!
205         int nItemCount = 0;
206 
207         for (int i=0; i<4; ++i)
208         {
209             java.lang.String sSetName   = null;
210             java.lang.String sSubDir    = null;
211             java.util.Vector lFragments = null;
212 
213             try
214             {
215                 switch(i)
216                 {
217                     case 0: // types
218                     {
219                         m_aLog.setDetailedInfo("generate set for types ... ");
220                         sSetName   = m_aCfg.getString(PROP_SETNAME_TYPES);
221                         sSubDir    = m_aCfg.getString(PROP_SUBDIR_TYPES );
222                         lFragments = m_lTypes;
223                     }
224                     break;
225 
226                     case 1: // filters
227                     {
228                         m_aLog.setDetailedInfo("generate set for filter ... ");
229                         sSetName   = m_aCfg.getString(PROP_SETNAME_FILTERS);
230                         sSubDir    = m_aCfg.getString(PROP_SUBDIR_FILTERS );
231                         lFragments = m_lFilters;
232                     }
233                     break;
234 
235                     case 2: // loaders
236                     {
237                         m_aLog.setDetailedInfo("generate set for frame loader ... ");
238                         sSetName   = m_aCfg.getString(PROP_SETNAME_LOADERS);
239                         sSubDir    = m_aCfg.getString(PROP_SUBDIR_LOADERS );
240                         lFragments = m_lLoaders;
241                     }
242                     break;
243 
244                     case 3: // handlers
245                     {
246                         m_aLog.setDetailedInfo("generate set for content handler ... ");
247                         sSetName   = m_aCfg.getString(PROP_SETNAME_HANDLERS);
248                         sSubDir    = m_aCfg.getString(PROP_SUBDIR_HANDLERS );
249                         lFragments = m_lHandlers;
250                     }
251                     break;
252                 }
253 
254                 nItemCount += lFragments.size();
255 
256                 getFragments(
257                     new java.io.File(m_aFragmentsDir, sSubDir),
258                     sSetName,
259                     lFragments,
260                     1,
261                     sBuffer);
262             }
263             catch(java.util.NoSuchElementException exIgnore)
264                 { continue; }
265         }
266 
267         m_aLog.setDetailedInfo("generate package footer ... ");
268         sBuffer.append(XMLHelper.generateFooter());
269 
270         // Attention!
271         // If the package seem to be empty, it make no sense to generate a corresponding
272         // xml file. We should suppress writing of this file on disk completely ...
273         if (nItemCount < 1)
274         {
275             m_aLog.setWarning("Package is empty and will not result into a xml file on disk!? Please check configuration file.");
276             return;
277         }
278         m_aLog.setGlobalInfo("package contains "+nItemCount+" items");
279 
280         java.io.File aPackage = new File(sPackage);
281         m_aLog.setGlobalInfo("write temp package \""+aPackage.getPath()); // TODO encoding must be read from the configuration
282         FileHelper.writeEncodedBufferToFile(aPackage, "UTF-8", false, sBuffer); // check for success is done inside this method!
283     }
284 
285     //-------------------------------------------
286     /** TODO */
getFragments(java.io.File aDir , java.lang.String sSetName , java.util.Vector lFragments , int nPrettyTabs, java.lang.StringBuffer sBuffer )287     private void getFragments(java.io.File           aDir       ,
288                               java.lang.String       sSetName   ,
289                               java.util.Vector       lFragments ,
290                               int                    nPrettyTabs,
291                               java.lang.StringBuffer sBuffer    )
292         throws java.lang.Exception
293     {
294         if (lFragments.size()<1)
295         {
296             m_aLog.setWarning("List of fragments is empty!? Will be ignored ...");
297             return;
298         }
299 
300         java.util.Enumeration  pFragments = lFragments.elements();
301         java.lang.String       sExtXcu    = m_aCfg.getString(PROP_EXTENSION_XCU);
302 
303         for (int tabs=0; tabs<nPrettyTabs; ++tabs)
304             sBuffer.append("\t");
305         sBuffer.append("<node oor:name=\""+sSetName+"\">\n");
306         ++nPrettyTabs;
307 
308         // special mode for generating language packs.
309         // In such case we must live with some missing fragment files.
310         // Reason behind; Not all filters are really localized.
311         // But we don't use a different fragment list. We try to locate
312         // any fragment file in its language-pack version ...
313         boolean bHandleLanguagePacks = m_aCfg.getBoolean(PROP_LANGUAGEPACK, false);
314         boolean bDebug               = m_aCfg.getBoolean(PROP_DEBUG       , false);
315         java.lang.String sEncoding   = "UTF-8";
316         if (bDebug)
317             sEncoding = "UTF-8Special";
318 
319         while(pFragments.hasMoreElements())
320         {
321             java.lang.String sFragment = (java.lang.String)pFragments.nextElement();
322             java.io.File     aFragment = new java.io.File(aDir, sFragment+"."+sExtXcu);
323 
324             // handle simple files only and check for existence!
325             if (!aFragment.exists())
326             {
327                 if (bHandleLanguagePacks)
328                 {
329                     m_aLog.setWarning("language fragment \""+aFragment.getPath()+"\" does not exist. Will be ignored.");
330                     continue;
331                 }
332                 else
333                     throw new java.io.IOException("fragment \""+aFragment.getPath()+"\" does not exists.");
334             }
335 
336             if (!aFragment.isFile())
337             {
338                 m_aLog.setWarning("fragment \""+aFragment.getPath()+"\" seem to be not a valid file.");
339                 continue;
340             }
341 
342             // copy file content of original fragment
343             // Note: A FileNotFoundException will be thrown automatically by the
344             // used reader objects. Let it break this method too. Our calli is interested
345             // on such errors :-)
346             m_aLog.setDetailedInfo("merge fragment \""+aFragment.getPath()+"\" ...");
347             FileHelper.readEncodedBufferFromFile(aFragment, sEncoding, sBuffer);
348 
349             sBuffer.append("\n");
350         }
351 
352         --nPrettyTabs;
353         for (int tabs=0; tabs<nPrettyTabs; ++tabs)
354             sBuffer.append("\t");
355         sBuffer.append("</node>\n");
356     }
357 }
358