UpToDateChecker.java revision b72c5c2e5482cf10117b2b25f642f7616b2326c3
1b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato/*
2b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * ProGuard -- shrinking, optimization, obfuscation, and preverification
3b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato *             of Java bytecode.
4b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato *
5b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * Copyright (c) 2002-2009 Eric Lafortune (eric@graphics.cornell.edu)
6b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato *
7b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * This program is free software; you can redistribute it and/or modify it
8b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * under the terms of the GNU General Public License as published by the Free
9b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * Software Foundation; either version 2 of the License, or (at your option)
10b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * any later version.
11b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato *
12b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * This program is distributed in the hope that it will be useful, but WITHOUT
13b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
14b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
15b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * more details.
16b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato *
17b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * You should have received a copy of the GNU General Public License along
18b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * with this program; if not, write to the Free Software Foundation, Inc.,
19b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato */
21b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onoratopackage proguard;
22b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
23b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onoratoimport java.io.File;
24b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
25b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato/**
26b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * This class checks whether the output is up to date.
27b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato *
28b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato * @author Eric Lafortune
29b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato */
30b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onoratopublic class UpToDateChecker
31b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato{
32b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    private final Configuration configuration;
33b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
34b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
35b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    /**
36b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato     * Creates a new UpToDateChecker with the given configuration.
37b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato     */
38b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    public UpToDateChecker(Configuration configuration)
39b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    {
40b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        this.configuration = configuration;
41b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    }
42b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
43b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
44b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    /**
45b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato     * Returns whether the output is up to date, based on the modification times
46b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato     * of the input jars, output jars, and library jars (or directories).
47b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato     */
48b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    public boolean check()
49b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    {
50b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        long inputLastModified  = configuration.lastModified;
51b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        long outputLastModified = Long.MAX_VALUE;
52b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
53b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        ClassPath programJars = configuration.programJars;
54b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        ClassPath libraryJars = configuration.libraryJars;
55b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
56b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        // Check the dates of the program jars, if any.
57b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        if (programJars != null)
58b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        {
59b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            for (int index = 0; index < programJars.size(); index++)
60b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            {
61b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                // Break early, if possible.
62b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                if (inputLastModified >= outputLastModified)
63b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                {
64b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    break;
65b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                }
66b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
67b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                // Update the input and output modification times.
68b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                ClassPathEntry classPathEntry = programJars.get(index);
69b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                if (classPathEntry.isOutput())
70b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                {
71b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    long lastModified = lastModified(classPathEntry.getFile(), true);
72b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    if (outputLastModified > lastModified)
73b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    {
74b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                        outputLastModified = lastModified;
75b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    }
76b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                }
77b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                else
78b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                {
79b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    long lastModified = lastModified(classPathEntry.getFile(), false);
80b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    if (inputLastModified < lastModified)
81b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    {
82b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                        inputLastModified = lastModified;
83b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    }
84b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                }
85b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            }
86b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        }
87b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
88b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        // Check the dates of the library jars, if any.
89b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        if (libraryJars != null)
90b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        {
91b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            for (int index = 0; index < libraryJars.size(); index++)
92b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            {
93b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                // Break early, if possible.
94b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                if (inputLastModified >= outputLastModified)
95b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                {
96b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    break;
97b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                }
98b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
99b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                // Update the input modification time.
100b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                ClassPathEntry classPathEntry = libraryJars.get(index);
101b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                long lastModified = lastModified(classPathEntry.getFile(), false);
102b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                if (inputLastModified < lastModified)
103b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                {
104b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    inputLastModified = lastModified;
105b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                }
106b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            }
107b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        }
108b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
109b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        boolean outputUpToDate = inputLastModified < outputLastModified;
110b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        if (outputUpToDate)
111b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        {
112b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            System.out.println("The output is up to date");
113b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        }
114b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
115b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        return outputUpToDate;
116b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    }
117b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
118b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
119b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    /**
120b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato     * Returns the minimum or maximum modification time of the given file or
121b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato     * of the files in the given directory (recursively).
122b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato     */
123b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    private long lastModified(File file, boolean minimum)
124b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    {
125b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        // Is it a directory?
126b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        if (file.isDirectory())
127b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        {
128b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            // Ignore the directory's modification time; just recurse on its files.
129b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            File[] files = file.listFiles();
130b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
131b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            // Still, an empty output directory is probably a sign that it is
132b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            // not up to date.
133b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            long lastModified = files.length != 0 && minimum ?
134b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                Long.MAX_VALUE : 0L;
135b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
136b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            for (int index = 0; index < files.length; index++)
137b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            {
138b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                long fileLastModified = lastModified(files[index], minimum);
139b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                if ((lastModified < fileLastModified) ^ minimum)
140b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                {
141b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                    lastModified = fileLastModified;
142b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato                }
143b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            }
144b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato
145b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            return lastModified;
146b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        }
147b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        else
148b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        {
149b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            // Return the file's modification time.
150b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato            return file.lastModified();
151b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato        }
152b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato    }
153b72c5c2e5482cf10117b2b25f642f7616b2326c3Joe Onorato}
154