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 complex.memCheck;
24 
25 import com.sun.star.beans.PropertyValue;
26 import com.sun.star.frame.XStorable;
27 import com.sun.star.lang.XComponent;
28 import com.sun.star.lang.XMultiServiceFactory;
29 import com.sun.star.uno.UnoRuntime;
30 import com.sun.star.util.XCloseable;
31 // import complexlib.ComplexTestCase;
32 import helper.ProcessHandler;
33 import java.io.File;
34 // import java.io.FilePermission;
35 import java.io.FileWriter;
36 import java.io.FilenameFilter;
37 import java.io.PrintWriter;
38 import java.util.Enumeration;
39 import java.util.StringTokenizer;
40 import java.util.Vector;
41 import lib.*;
42 import util.DesktopTools;
43 // import util.WriterTools;
44 
45 import org.junit.After;
46 import org.junit.AfterClass;
47 import org.junit.Before;
48 import org.junit.BeforeClass;
49 import org.junit.Test;
50 import org.openoffice.test.OfficeConnection;
51 import static org.junit.Assert.*;
52 
53 /**
54  * Documents are opened and exported with StarOffice. The memory usage of
55  * StarOffice is monitored and if the usage exceeds the allowed kilobytes,
56  * the test is failed. Used for monitoring the StarOffice process is the
57  * command line tool 'pmap', available on Solaris or Linux. This test will not
58  * run on Windows.<br>Test procedure: every given document type is searched in
59  * the source directory
60  * Needed paramters:
61  * <ul>
62  *   <li>"TestDocumentPath" - the path where test documents are located.</li>
63  *   <li>"AllowMemoryIncrease" (optional) - the allowed memory increase measured in kByte per exported document. The default is 10 kByte.</li>
64  *   <li>"ExportDocCount" (optional) - the amount of exports for each document that is loaded. Is defaulted to 25.
65  *   <li>"FileExportFilter" (optional) - a relation between loaded document type and used export filter. Is defaulted to
66  *       writer, calc and impress. This parameter can be set with a number to give more than one relation. Example:<br>
67  *       "FileExportFilter1=sxw,writer_pdf_Export"<br>
68  *       "FileExportFilter2=sxc,calc_pdf_Export"<br>
69  *       "FileExportFilter3=sxi,impress_pdf_Export"<br></li>
70  *       All parameters are used for iteration over the test document path.
71  * </ul>
72  */
73 class TempDir
74 {
75 
76     private String m_sTempDir;
77 
78     public TempDir(String _sTempDir)
79     {
80         m_sTempDir = _sTempDir;
81     }
82 
83     public String getOfficeTempDir()
84     {
85         return m_sTempDir;
86     }
87 
88     public String getTempDir()
89     {
90         final String sTempDir = FileHelper.getJavaCompatibleFilename(m_sTempDir);
91         return sTempDir;
92     }
93 }
94 
95 public class CheckMemoryUsage /* extends ComplexTestCase */
96 
97 {
98 
99     private final String sWriterDoc = "sxw,writer_pdf_Export";
100     private final String sCalcDoc = "sxc,calc_pdf_Export";
101     private final String sImpressDoc = "sxi,impress_pdf_Export";
102     // private String sProcessIdCommand = null;
103     TempDir m_aTempDir;
104     // private String sFS = null;
105     // private String sMemoryMap1 = null;
106     // private String sMemoryMap2 = null;
107     // private String sDocumentPath = "";
108     private String[][] sDocTypeExportFilter;
109     private String[][] sDocuments;
110     private int iAllowMemoryIncrease = 10;
111     private int iExportDocCount = 25;
112     /**
113      * The test parameters
114      */
115     private static TestParameters param = null;
116 
117     /**
118      * Get all test methods
119      * @return The test methods.
120     //     */
121 //    public String[] getTestMethodNames() {
122 //        return new String[] {"loadAndSaveDocuments"};
123 //    }
124     /**
125      * Collect all documnets to load and all filters used for export.
126      */
127     @Before
128     public void before()
129     {
130 
131         final XMultiServiceFactory xMsf = getMSF();
132 
133         // some Tests need the qadevOOo TestParameters, it is like a Hashmap for Properties.
134         param = new TestParameters();
135         param.put("ServiceFactory", xMsf); // some qadevOOo functions need the ServiceFactory
136 
137         // test does definitely not run on Windows.
138         if (param.get("OperatingSystem").equals("wntmsci"))
139         {
140             System.out.println("Test can only reasonably be executed with a tool that "
141                     + "displays the memory usage of StarOffice.");
142             System.out.println("Test does not run on Windows, only on Solaris or Linux.");
143             // in an automatic environment it is better to say, there is no error here.
144             // it is a limitation, but no error.
145             System.exit(0);
146         }
147 
148 
149         // how many times is every document exported.
150         int count = param.getInt("ExportDocCount");
151         if (count != 0)
152         {
153             iExportDocCount = count;
154         }
155 
156         // get the temp dir for creating the command scripts.
157         // sTempDir = System.getProperty("java.io.tmpdir");
158         m_aTempDir = new TempDir(util.utils.getOfficeTemp/*Dir*/(xMsf));
159 
160         // get the file extension, export filter connection
161         Enumeration keys = param.keys();
162         Vector<String> v = new Vector<String>();
163         while (keys.hasMoreElements())
164         {
165             String key = (String) keys.nextElement();
166             if (key.startsWith("FileExportFilter"))
167             {
168                 v.add((String) param.get(key));
169             }
170         }
171         // if no param given, set defaults.
172         if (v.size() == 0)
173         {
174             v.add(sWriterDoc);
175             v.add(sCalcDoc);
176             v.add(sImpressDoc);
177         }
178         // store a file extension
179         sDocTypeExportFilter = new String[v.size()][2];
180         for (int i = 0; i < v.size(); i++)
181         {
182             // 2do: error routine for wrong given params
183             final String sVContent = v.get(i);
184             StringTokenizer t = new StringTokenizer(sVContent, ",");
185             final String sExt = t.nextToken();
186             final String sName = t.nextToken();
187             sDocTypeExportFilter[i][0] = sExt;
188             sDocTypeExportFilter[i][1] = sName;
189         }
190 
191         // get files to load and export
192 //        sDocumentPath = (String) param.get("TestDocumentPath");
193         String sDocumentPath = TestDocument.getUrl();
194         File f = new File(FileHelper.getJavaCompatibleFilename(sDocumentPath));
195         // sDocumentPath = f.getAbsolutePath();
196         // String sFS = System.getProperty("file.separator");
197         sDocuments = new String[sDocTypeExportFilter.length][];
198         for (int j = 0; j < sDocTypeExportFilter.length; j++)
199         {
200             FileFilter filter = new FileFilter(sDocTypeExportFilter[j][0]);
201             String[] doc = f.list(filter);
202             sDocuments[j] = new String[doc.length];
203             for (int i = 0; i < doc.length; i++)
204             {
205                 // final String sDocument = FileHelper.appendPath(sDocumentPath, doc[i]);
206                 // sDocuments[j][i] = utils.getFullURL(sDocuments[j][i]);
207                 sDocuments[j][i] = TestDocument.getUrl(doc[i]);
208             }
209         }
210     }
211 
212     /**
213      * delete all created files on disk
214      */
215     @After
216     public void after()
217     {
218         // delete the constructed files.
219 // we don't need to delete anything, all is stored in $USER_TREE
220 //        for (int i = 0; i < iExportDocCount; i++)
221 //        {
222 //            final String sDocumentName = "DocExport" + i + ".pdf";
223 //            final String sFilename = FileHelper.appendPath(m_sTempDir, sDocumentName);
224 //            File f = new File(FileHelper.getJavaCompatibleFilename(sFilename));
225 //            f.delete();
226 //        }
227         // File f = new File(sProcessIdCommand);
228         // f.delete();
229         // f = new File(sOfficeMemoryCommand);
230         // f.delete();
231     }
232 
233     /**
234      * The test function: load documents and save them using the given filters
235      * for each given document type.
236      */
237     @Test
238     public void loadAndSaveDocuments()
239     {
240         int nOk = 0;
241         int nRunThrough = 0;
242 
243         // At first:
244         // we load the document, there will be some post work in office like late initialisations
245         // we store exact one time the document
246         // so the memory footprint should be right
247 
248         // iterate over all document types
249         for (int k = 0; k < sDocTypeExportFilter.length; k++)
250         {
251             // iterate over all documents of this type
252             for (int i = 0; i < sDocuments[k].length; i++)
253             {
254 
255                 final String sDocument = sDocuments[k][i];
256                 final String sExtension = sDocTypeExportFilter[k][1];
257 
258 //                OfficeMemchecker aChecker = new OfficeMemchecker();
259 //                aChecker.setDocumentName(FileHelper.getBasename(sDocument));
260 //                aChecker.setExtension(sExtension);
261 //                aChecker.start();
262 
263                 loadAndSaveNTimesDocument(sDocument, 1, sExtension);
264 
265 //                nOk += checkMemory(aChecker);
266 //                nRunThrough ++;
267             }
268             System.out.println();
269             System.out.println();
270         }
271 
272         shortWait(10000);
273 
274         // Now the real test, load document and store 25 times
275 
276         // iterate over all document types
277         for (int k = 0; k < sDocTypeExportFilter.length; k++)
278         {
279             // iterate over all documents of this type
280             for (int i = 0; i < sDocuments[k].length; i++)
281             {
282 
283                 final String sDocument = sDocuments[k][i];
284                 final String sExtension = sDocTypeExportFilter[k][1];
285 
286                 OfficeMemchecker aChecker = new OfficeMemchecker();
287                 aChecker.setDocumentName(FileHelper.getBasename(sDocument));
288                 aChecker.setExtension(sExtension);
289                 aChecker.start();
290 
291                 loadAndSaveNTimesDocument(sDocument, iExportDocCount, sExtension);
292 
293                 aChecker.stop();
294                 final int nConsumMore = aChecker.getConsumMore();
295 
296                 nOk += checkMemory(nConsumMore);
297                 nRunThrough++;
298             }
299             System.out.println();
300             System.out.println();
301         }
302         System.out.println("Find the output of used 'pmap' here: " + m_aTempDir.getTempDir() + " if test failed.");
303         assertTrue("Office consumes too many memory.", nOk == nRunThrough);
304     }
305 
306     /**
307      * Checks how much memory should consum
308      * @param storageBefore
309      * @return 1 if consum is ok, else 0
310      */
311     private int checkMemory(int nConsumMore)
312     {
313         int nAllowed = iAllowMemoryIncrease * iExportDocCount;
314         System.out.println("The Office consumes now " + nConsumMore
315                 + "K more memory than at the start of the test; allowed were "
316                 + nAllowed + "K.");
317         if (nConsumMore > nAllowed)
318         {
319             System.out.println("ERROR: This is not allowed.");
320             return 0;
321         }
322         System.out.println("OK.");
323         return 1;
324     }
325 
326     /**
327      * load and save exact one document
328      */
329     private void loadAndSaveNTimesDocument(String _sDocument, int _nCount, String _sStoreExtension)
330     {
331         System.out.println("Document: " + _sDocument);
332         XComponent xComponent = DesktopTools.loadDoc(getMSF(), _sDocument, null);
333         XStorable xStorable = UnoRuntime.queryInterface(XStorable.class, xComponent);
334         if (xStorable != null)
335         {
336             // export each document iExportDocCount times
337             for (int j = 0; j < _nCount; j++)
338             {
339                 final String sDocumentName = FileHelper.getBasename(_sDocument) + "_" + j + ".pdf";
340                 final String sFilename = FileHelper.appendPath(m_aTempDir.getOfficeTempDir(), sDocumentName);
341                 // String url = utils.getFullURL(sFilename);
342                 String url = sFilename; // graphical.FileHelper.getFileURLFromSystemPath(sFilename);
343                 try
344                 {
345                     PropertyValue[] props = new PropertyValue[1];
346                     props[0] = new PropertyValue();
347                     props[0].Name = "FilterName";
348                     // use export filter for this doc type
349                     props[0].Value = _sStoreExtension;
350                     xStorable.storeToURL(url, props);
351                 }
352                 catch (com.sun.star.io.IOException e)
353                 {
354                     fail("Could not store to '" + url + "'");
355                 }
356             }
357             // close the doc
358             XCloseable xCloseable = UnoRuntime.queryInterface(XCloseable.class, xStorable);
359             try
360             {
361                 xCloseable.close(true);
362             }
363             catch (com.sun.star.util.CloseVetoException e)
364             {
365                 e.printStackTrace();
366                 fail("Cannot close document: test is futile, Office will surely use more space.");
367             }
368         }
369         else
370         {
371             System.out.println("Cannot query for XStorable interface on document '" + _sDocument + "'");
372             System.out.println(" -> Skipping storage.");
373         }
374 
375     }
376 
377 // -----------------------------------------------------------------------------
378     private class OfficeMemchecker
379     {
380 
381         /**
382          * After called start() it contains the memory need at startup
383          */
384         private int m_nMemoryStart;
385         /**
386          * After called stop() it contains the memory usage
387          */
388         private int m_nMemoryUsage;
389         private String m_sDocumentName;
390         private String m_sExtension;
391 
392         public OfficeMemchecker()
393         {
394             m_nMemoryStart = 0;
395         }
396 
397         public void setDocumentName(String _sDocName)
398         {
399             m_sDocumentName = _sDocName;
400         }
401 
402         public void setExtension(String _sExt)
403         {
404             m_sExtension = _sExt;
405         }
406 
407         public void start()
408         {
409             m_nMemoryStart = getOfficeMemoryUsage(createModeName("start", 0));
410         }
411 
412         private String createModeName(String _sSub, int _nCount)
413         {
414             StringBuffer aBuf = new StringBuffer();
415             aBuf.append(_sSub);
416             aBuf.append('_').append(m_sDocumentName).append('_').append(m_sExtension);
417             aBuf.append('_').append(_nCount);
418             return aBuf.toString();
419         }
420 
421         public void stop()
422         {
423             // short wait for the office to 'calm down' and free some memory
424             shortWait(20000);
425             // wait util memory is not freed anymore.
426             int storageAfter = getOfficeMemoryUsage(createModeName("stop", 0));
427             int mem = 0;
428             int count = 0;
429             while (storageAfter != mem && count < 10)
430             {
431                 count++;
432                 mem = storageAfter;
433                 storageAfter = getOfficeMemoryUsage(createModeName("stop", count));
434                 shortWait(1000);
435             }
436             m_nMemoryUsage = (storageAfter - m_nMemoryStart);
437         }
438 
439         public int getConsumMore()
440         {
441             return m_nMemoryUsage;
442         }
443 
444         /**
445          * Get the process ID from the Office
446          * @return the Id as String
447          */
448         private String getOfficeProcessID()
449         {
450             String sProcessIdCommand = FileHelper.appendPath(m_aTempDir.getTempDir(), "getPS");
451             final String sofficeArg = org.openoffice.test.Argument.get("soffice");
452             final String sPSGrep = "ps -ef | grep $USER | grep <soffice>.bin | grep -v grep";
453             final String sProcessId = sPSGrep.replaceAll("<soffice>", FileHelper.getJavaCompatibleFilename(sofficeArg));
454 
455             createExecutableFile(sProcessIdCommand, sProcessId);
456             ProcessHandler processID = new ProcessHandler(sProcessIdCommand);
457             processID.noOutput();
458             processID.executeSynchronously();
459             String text = processID.getOutputText();
460             if (text == null || text.equals("") || text.indexOf(' ') == -1)
461             {
462                 fail("Could not determine Office process ID. Check " + sProcessIdCommand);
463             }
464             StringTokenizer aToken = new StringTokenizer(text);
465             // this is not nice, but ps gives the same output on every machine
466             aToken.nextToken();
467             String id = aToken.nextToken();
468             return id;
469         }
470 
471         /**
472          * Get the memory usage of the Office in KByte.
473          * @return The memory used by the Office.
474          */
475         private int getOfficeMemoryUsage(String _sMode)
476         {
477             final String sMemoryMonitor = "pmap <processID> |tee <pmapoutputfile> | grep total";
478             String sOfficeMemoryCommand = null;
479             sOfficeMemoryCommand = FileHelper.appendPath(m_aTempDir.getTempDir(), "getPmap");
480             // sOfficeMemoryCommand = FileHelper.getJavaCompatibleFilename(sOfficeMemoryCommand);
481             String command = sMemoryMonitor.replaceAll("<processID>", getOfficeProcessID());
482             String sPmapOutputFile = FileHelper.appendPath(m_aTempDir.getTempDir(), "pmap_" + _sMode + ".txt");
483             command = command.replaceAll("<pmapoutputfile>", sPmapOutputFile);
484             createExecutableFile(sOfficeMemoryCommand, command);
485 
486             ProcessHandler processID = new ProcessHandler(sOfficeMemoryCommand);
487             processID.noOutput();
488             processID.executeSynchronously();
489             int nError = processID.getExitCode();
490             assertTrue("Execute of " + sOfficeMemoryCommand + " failed", nError == 0);
491             String text = processID.getOutputText();
492             if (text == null || text.equals("") || text.indexOf(' ') == -1)
493             {
494                 fail("Could not determine Office memory usage. Check " + sOfficeMemoryCommand);
495             }
496             StringTokenizer aToken = new StringTokenizer(text);
497             // this works, because the output of pmap is quite standardized.
498             aToken.nextToken();
499             String mem = aToken.nextToken();
500             mem = mem.substring(0, mem.indexOf('K'));
501             Integer memory = new Integer(mem);
502             return memory.intValue();
503         }
504 
505         /**
506          * Write a script file and set its rights to rwxrwxrwx.
507          * @param fileName The name of the created file
508          * @param line The commandline that has to be written inside of the file.
509          */
510         private void createExecutableFile(String fileName, String line)
511         {
512             final String sChmod = "chmod a+x ";
513             final String bash = "#!/bin/bash";
514 
515             try
516             {
517                 String sFilename = FileHelper.getJavaCompatibleFilename(fileName);
518                 PrintWriter fWriter = new PrintWriter(new FileWriter(sFilename));
519                 fWriter.println(bash);
520                 fWriter.println(line);
521                 fWriter.close();
522                 // change rights to rwxrwxrwx
523                 ProcessHandler processID = new ProcessHandler(sChmod + sFilename);
524                 processID.noOutput();
525                 processID.executeSynchronously();
526                 int nError = processID.getExitCode();
527                 assertTrue("chmod failed. ", nError == 0);
528             }
529             catch (java.io.IOException e)
530             {
531             }
532         }
533     }
534 
535     /**
536      * Let this thread sleep for some time
537      * @param milliSeconds time to wait in milliseconds.
538      */
539     public static void shortWait(int milliSeconds)
540     {
541         System.out.println("Wait for: " + milliSeconds + "ms");
542         try
543         {
544             Thread.sleep(milliSeconds);
545         }
546         catch (java.lang.InterruptedException e)
547         { // ignore
548         }
549     }
550 
551     /**
552      * Own file filter, will just return ok for all files that end with a given
553      * suffix
554      */
555     private class FileFilter implements FilenameFilter
556     {
557 
558         private String suffix = null;
559 
560         /**
561          * C'tor.
562          * @param suffix The suffix each filename should end with.
563          */
564         public FileFilter(String suffix)
565         {
566             this.suffix = suffix;
567         }
568 
569         /**
570          * Returns true, if the name of the file has the suffix given to the
571          * c'tor.
572          * @param name The filename that is tested.
573          * @param file Not used.
574          * @return True, if name ends with suffix.
575          */
576         public boolean accept(File file, String name)
577         {
578             return name.endsWith(suffix);
579         }
580     }
581 
582     private XMultiServiceFactory getMSF()
583     {
584         final XMultiServiceFactory xMSF1 = UnoRuntime.queryInterface(XMultiServiceFactory.class, connection.getComponentContext().getServiceManager());
585         return xMSF1;
586     }
587 
588     // setup and close connections
589     @BeforeClass
590     public static void setUpConnection() throws Exception
591     {
592         System.out.println("setUpConnection()");
593         connection.setUp();
594     }
595 
596     @AfterClass
597     public static void tearDownConnection()
598             throws InterruptedException, com.sun.star.uno.Exception
599     {
600         System.out.println("tearDownConnection()");
601         connection.tearDown();
602     }
603     private static final OfficeConnection connection = new OfficeConnection();
604 }
605