View Javadoc
1   package org.codehaus.plexus.digest;
2   
3   /*
4    * Copyright 2001-2007 The Codehaus.
5    *
6    * Licensed under the Apache License, Version 2.0 (the "License");
7    * you may not use this file except in compliance with the License.
8    * You may obtain a copy of the License at
9    *
10   *      http://www.apache.org/licenses/LICENSE-2.0
11   *
12   * Unless required by applicable law or agreed to in writing, software
13   * distributed under the License is distributed on an "AS IS" BASIS,
14   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15   * See the License for the specific language governing permissions and
16   * limitations under the License.
17   */
18  
19  import org.codehaus.plexus.util.FileUtils;
20  import org.codehaus.plexus.util.StringUtils;
21  
22  import java.io.File;
23  import java.io.FileNotFoundException;
24  import java.io.IOException;
25  
26  /**
27   * ChecksumFile 
28   *
29   * @author <a href="mailto:joakim@erdfelt.com">Joakim Erdfelt</a>
30   * @version $Id$
31   * 
32   * @plexus.component role="org.codehaus.plexus.digest.ChecksumFile"
33   */
34  public class ChecksumFile
35  {
36      /**
37       * @plexus.requirement role-hint="sha1"
38       */
39      private Digester digestSha1;
40  
41      /**
42       * @plexus.requirement role-hint="md5";
43       */
44      private Digester digestMd5;
45  
46      /**
47       * <p>
48       * Given a checksum file, check to see if the file it represents is valid according to the checksum.
49       * </p>
50       * 
51       * <dl>
52       *   <lh>Terminology:</lh>
53       *   <dt>Checksum File</dt>
54       *   <dd>The file that contains the previously calculated checksum value for the reference file.
55       *       This is a text file with the extension ".sha1" or ".md5", and contains a single entry
56       *       consisting of an optional reference filename, and a checksum string.
57       *   </dd>
58       *   <dt>Reference File</dt>
59       *   <dd>The file that is being referenced in the checksum file.</dd>
60       * </dl>
61       * 
62       * <p>
63       * NOTE: Only supports single file checksums of type MD5 or SHA1.
64       * </p>
65       * 
66       * @param checksumFile the checksum file (must end in ".sha1" or ".md5")
67       * @return true if the checksum is valid for the file it represents.
68       * @throws DigesterException if there is a digester problem during the check of the reference file. 
69       * @throws FileNotFoundException if the checksumFile itself or the file it refers to is not found.
70       * @throws IOException if the reading of the checksumFile or the file it refers to fails.
71       */
72      public boolean isValidChecksum( File checksumFile ) throws DigesterException, FileNotFoundException, IOException
73      {
74          if ( !checksumFile.exists() )
75          {
76              throw new FileNotFoundException( "Unable to find checksum file " + checksumFile.getAbsolutePath() );
77          }
78  
79          if ( !checksumFile.isFile() )
80          {
81              throw new IOException( "Unable to load checksum from non-file " + checksumFile.getAbsolutePath() );
82          }
83  
84          String path = checksumFile.getAbsolutePath();
85          Digester digester = null;
86  
87          if ( path.endsWith( digestMd5.getFilenameExtension() ) )
88          {
89              digester = digestMd5;
90          }
91          else if ( path.endsWith( digestSha1.getFilenameExtension() ) )
92          {
93              digester = digestSha1;
94          }
95          // TODO: Add more digester implementations here.
96  
97          if ( digester == null )
98          {
99              throw new DigesterException( "Unable to determine digester type from filename " + path );
100         }
101 
102         File referenceFile = new File( path.substring( 0, path.length() - digester.getFilenameExtension().length() ) );
103 
104         String rawChecksum = FileUtils.fileRead( checksumFile, "UTF-8" );
105         String expectedChecksum = DigestUtils.cleanChecksum( rawChecksum, digester, referenceFile.getName() );
106 
107         String actualChecksum = digester.calc( referenceFile );
108 
109         return StringUtils.equalsIgnoreCase( expectedChecksum, actualChecksum );
110     }
111 
112     /**
113      * Creates a checksum file of the provided referenceFile.
114      * 
115      * @param referenceFile the file to checksum.
116      * @param digester the digester to use.
117      * @return the checksum File that was created.
118      * @throws DigesterException if there was a problem calculating the checksum of the referenceFile. 
119      * @throws IOException if there was a problem either reading the referenceFile, or writing the checksum file.
120      */
121     public File createChecksum( File referenceFile, Digester digester ) throws DigesterException, IOException
122     {
123         File checksumFile = new File( referenceFile.getAbsolutePath() + digester.getFilenameExtension() );
124         String checksum = digester.calc( referenceFile );
125         FileUtils.fileWrite( checksumFile.getAbsolutePath(), "UTF-8", checksum + "  " + referenceFile.getName() );
126         return checksumFile;
127     }
128 }