Coverage Report - org.kuali.rice.krad.service.impl.AttachmentServiceImpl
 
Classes in this File Line Coverage Branch Coverage Complexity
AttachmentServiceImpl
0%
0/110
0%
0/50
3.6
 
 1  
 /*
 2  
  * Copyright 2006-2011 The Kuali Foundation
 3  
  *
 4  
  * Licensed under the Educational Community License, Version 2.0 (the "License");
 5  
  * you may not use this file except in compliance with the License.
 6  
  * You may obtain a copy of the License at
 7  
  *
 8  
  * http://www.opensource.org/licenses/ecl2.php
 9  
  *
 10  
  * Unless required by applicable law or agreed to in writing, software
 11  
  * distributed under the License is distributed on an "AS IS" BASIS,
 12  
  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 13  
  * See the License for the specific language governing permissions and
 14  
  * limitations under the License.
 15  
  */
 16  
 
 17  
 package org.kuali.rice.krad.service.impl;
 18  
 
 19  
 import org.apache.commons.lang.StringUtils;
 20  
 import org.apache.log4j.Logger;
 21  
 import org.kuali.rice.core.api.config.property.ConfigurationService;
 22  
 import org.kuali.rice.krad.bo.Attachment;
 23  
 import org.kuali.rice.krad.bo.Note;
 24  
 import org.kuali.rice.krad.bo.PersistableBusinessObject;
 25  
 import org.kuali.rice.krad.dao.AttachmentDao;
 26  
 import org.kuali.rice.krad.document.Document;
 27  
 import org.kuali.rice.krad.service.AttachmentService;
 28  
 import org.kuali.rice.krad.util.KRADConstants;
 29  
 import org.springframework.transaction.annotation.Transactional;
 30  
 
 31  
 import java.io.BufferedInputStream;
 32  
 import java.io.BufferedOutputStream;
 33  
 import java.io.File;
 34  
 import java.io.FileInputStream;
 35  
 import java.io.FileOutputStream;
 36  
 import java.io.IOException;
 37  
 import java.io.InputStream;
 38  
 import java.util.UUID;
 39  
 
 40  
 /**
 41  
  * Attachment service implementation
 42  
  */
 43  
 @Transactional
 44  0
 public class AttachmentServiceImpl implements AttachmentService {
 45  
         private static final int MAX_DIR_LEVELS = 6;
 46  0
     private static Logger LOG = Logger.getLogger(AttachmentServiceImpl.class);
 47  
 
 48  
     private ConfigurationService kualiConfigurationService;
 49  
     private AttachmentDao attachmentDao;
 50  
     /**
 51  
      * Retrieves an Attachment by note identifier.
 52  
      * 
 53  
      * @see org.kuali.rice.krad.service.AttachmentService#getAttachmentByNoteId(java.lang.Long)
 54  
      */
 55  
     public Attachment getAttachmentByNoteId(Long noteId) {
 56  0
                 return attachmentDao.getAttachmentByNoteId(noteId);
 57  
         }
 58  
 
 59  
     /**
 60  
      * @see org.kuali.rice.krad.service.DocumentAttachmentService#createAttachment(java.lang.String, java.lang.String, int,
 61  
      *      java.io.InputStream, Document)
 62  
      */
 63  
     public Attachment createAttachment(PersistableBusinessObject parent, String uploadedFileName, String mimeType, int fileSize, InputStream fileContents, String attachmentTypeCode) throws IOException {
 64  0
         if ( LOG.isDebugEnabled() ) {
 65  0
             LOG.debug("starting to create attachment for document: " + parent.getObjectId());
 66  
         }
 67  0
         if (parent == null) {
 68  0
             throw new IllegalArgumentException("invalid (null or uninitialized) document");
 69  
         }
 70  0
         if (StringUtils.isBlank(uploadedFileName)) {
 71  0
             throw new IllegalArgumentException("invalid (blank) fileName");
 72  
         }
 73  0
         if (StringUtils.isBlank(mimeType)) {
 74  0
             throw new IllegalArgumentException("invalid (blank) mimeType");
 75  
         }
 76  0
         if (fileSize <= 0) {
 77  0
             throw new IllegalArgumentException("invalid (non-positive) fileSize");
 78  
         }
 79  0
         if (fileContents == null) {
 80  0
             throw new IllegalArgumentException("invalid (null) inputStream");
 81  
         }
 82  
 
 83  0
         String uniqueFileNameGuid = UUID.randomUUID().toString();
 84  0
         String fullPathUniqueFileName = getDocumentDirectory(parent.getObjectId()) + File.separator + uniqueFileNameGuid;
 85  
 
 86  0
         writeInputStreamToFileStorage(fileContents, fullPathUniqueFileName);
 87  
 
 88  
         // create DocumentAttachment
 89  0
         Attachment attachment = new Attachment();
 90  0
         attachment.setAttachmentIdentifier(uniqueFileNameGuid);
 91  0
         attachment.setAttachmentFileName(uploadedFileName);
 92  0
         attachment.setAttachmentFileSize(new Long(fileSize));
 93  0
         attachment.setAttachmentMimeTypeCode(mimeType);
 94  0
         attachment.setAttachmentTypeCode(attachmentTypeCode);
 95  
         
 96  0
         LOG.debug("finished creating attachment for document: " + parent.getObjectId());
 97  0
         return attachment;
 98  
     }
 99  
 
 100  
     private void writeInputStreamToFileStorage(InputStream fileContents, String fullPathUniqueFileName) throws IOException {
 101  0
         File fileOut = new File(fullPathUniqueFileName);
 102  0
         FileOutputStream streamOut = null;
 103  0
         BufferedOutputStream bufferedStreamOut = null;
 104  
         try {
 105  0
             streamOut = new FileOutputStream(fileOut);
 106  0
             bufferedStreamOut = new BufferedOutputStream(streamOut);
 107  
             int c;
 108  0
             while ((c = fileContents.read()) != -1) {
 109  0
                 bufferedStreamOut.write(c);
 110  
             }
 111  
         }
 112  
         finally {
 113  0
             bufferedStreamOut.close();
 114  0
             streamOut.close();
 115  0
         }
 116  0
     }
 117  
     
 118  
     public void moveAttachmentWherePending(Note note) {
 119  0
             if (note == null) {
 120  0
                     throw new IllegalArgumentException("Note must be non-null");
 121  
             }
 122  0
             if (StringUtils.isBlank(note.getObjectId())) {
 123  0
                     throw new IllegalArgumentException("Note does not have a valid object id, object id was null or empty");
 124  
             }
 125  0
             Attachment attachment = note.getAttachment();
 126  0
             if(attachment!=null){
 127  
                     try {
 128  0
                             moveAttachmentFromPending(attachment, note.getObjectId());
 129  
                     }
 130  0
                     catch (IOException e) {
 131  0
                             throw new RuntimeException("Problem moving pending attachment to final directory");    
 132  0
                     }
 133  
             }
 134  0
     }
 135  
     
 136  
     private void moveAttachmentFromPending(Attachment attachment, String objectId) throws IOException {
 137  
         //This method would probably be more efficient if attachments had a pending flag
 138  0
         String fullPendingFileName = getPendingDirectory() + File.separator + attachment.getAttachmentIdentifier();
 139  0
         File pendingFile = new File(fullPendingFileName);
 140  
         
 141  0
         if(pendingFile.exists()) {
 142  0
             BufferedInputStream bufferedStream = null;
 143  0
             FileInputStream oldFileStream = null;
 144  0
             String fullPathNewFile = getDocumentDirectory(objectId) + File.separator + attachment.getAttachmentIdentifier();
 145  
             try {
 146  0
                 oldFileStream = new FileInputStream(pendingFile);
 147  0
                 bufferedStream = new BufferedInputStream(oldFileStream);
 148  0
                 writeInputStreamToFileStorage(bufferedStream,fullPathNewFile);
 149  
             }
 150  
             finally {
 151  
 
 152  0
                 bufferedStream.close();
 153  0
                 oldFileStream.close();
 154  
                 //this has to come after the close
 155  0
                 pendingFile.delete();
 156  
                 
 157  0
             }
 158  
         }
 159  
         
 160  0
     }
 161  
 
 162  
     public void deleteAttachmentContents(Attachment attachment) {
 163  0
             if (attachment.getNote() == null) throw new RuntimeException("Attachment.note must be set in order to delete the attachment");
 164  0
         String fullPathUniqueFileName = getDocumentDirectory(attachment.getNote().getRemoteObjectIdentifier()) + File.separator + attachment.getAttachmentIdentifier();
 165  0
         File attachmentFile = new File(fullPathUniqueFileName);
 166  0
         attachmentFile.delete();
 167  0
     }
 168  
     private String getPendingDirectory() {
 169  0
         return this.getDocumentDirectory("");
 170  
     }
 171  
 
 172  
     private String getDocumentDirectory(String objectId) {
 173  
         // Create a directory; all ancestor directories must exist
 174  0
         File documentDirectory = new File(getDocumentFileStorageLocation(objectId));
 175  0
         if (!documentDirectory.exists()) {
 176  0
             boolean success = documentDirectory.mkdirs();
 177  0
             if (!success) {
 178  0
                 throw new RuntimeException("Could not generate directory for File at: " + documentDirectory.getAbsolutePath());
 179  
             }
 180  
         }
 181  0
         return documentDirectory.getAbsolutePath();
 182  
     }
 183  
 
 184  
     /**
 185  
      * /* (non-Javadoc)
 186  
      *
 187  
      * @see org.kuali.rice.krad.service.DocumentAttachmentService#retrieveAttachmentContents(org.kuali.rice.krad.document.DocumentAttachment)
 188  
      */
 189  
     public InputStream retrieveAttachmentContents(Attachment attachment) throws IOException {
 190  
         //refresh to get Note object in case it's not there
 191  0
         if(attachment.getNoteIdentifier()!=null) {
 192  0
             attachment.refreshNonUpdateableReferences();
 193  
         }
 194  
         
 195  0
         String parentDirectory = "";
 196  0
         if(attachment.getNote()!=null) {
 197  
             // TODO : Samuel check why had to change from getRemoteObjectIdentifier for KRAD
 198  
             // and what is impact on KNS
 199  0
             parentDirectory = attachment.getNote().getObjectId();
 200  
         }
 201  
          
 202  0
         return new BufferedInputStream(new FileInputStream(getDocumentDirectory(parentDirectory) + File.separator + attachment.getAttachmentIdentifier()));
 203  
     }
 204  
 
 205  
     private String getDocumentFileStorageLocation(String objectId) {
 206  0
         String location = null;
 207  0
         if(StringUtils.isEmpty(objectId)) {
 208  0
             location = kualiConfigurationService.getPropertyValueAsString(
 209  
                     KRADConstants.ATTACHMENTS_PENDING_DIRECTORY_KEY);
 210  
         } else {    
 211  
                 /* 
 212  
                  * We need to create a hierarchical directory structure to store
 213  
                  * attachment directories, as most file systems max out at 16k
 214  
                  * or 32k entries.  If we use 6 levels of hierarchy, it allows
 215  
                  * hundreds of billions of attachment directories.
 216  
                  */
 217  0
             char[] chars = objectId.toUpperCase().replace(" ", "").toCharArray();            
 218  0
             int count = chars.length < MAX_DIR_LEVELS ? chars.length : MAX_DIR_LEVELS;
 219  
 
 220  0
             StringBuffer prefix = new StringBuffer();            
 221  0
             for ( int i = 0; i < count; i++ )
 222  0
                 prefix.append(File.separator + chars[i]);
 223  
             
 224  0
             location = kualiConfigurationService.getPropertyValueAsString(KRADConstants.ATTACHMENTS_DIRECTORY_KEY) + prefix + File.separator + objectId;
 225  
         }
 226  0
         return  location;
 227  
     }
 228  
 
 229  
     /**
 230  
      * @see org.kuali.rice.krad.service.AttachmentService#deletePendingAttachmentsModifiedBefore(long)
 231  
      */
 232  
     public void deletePendingAttachmentsModifiedBefore(long modificationTime) {
 233  0
         String pendingAttachmentDirName = getPendingDirectory();
 234  0
         if (StringUtils.isBlank(pendingAttachmentDirName)) {
 235  0
             throw new RuntimeException("Blank pending attachment directory name");
 236  
         }
 237  0
         File pendingAttachmentDir = new File(pendingAttachmentDirName);
 238  0
         if (!pendingAttachmentDir.exists()) {
 239  0
             throw new RuntimeException("Pending attachment directory does not exist");
 240  
         }
 241  0
         if (!pendingAttachmentDir.isDirectory()) {
 242  0
             throw new RuntimeException("Pending attachment directory is not a directory! " + pendingAttachmentDir.getAbsolutePath());
 243  
         }
 244  
         
 245  0
         File[] files = pendingAttachmentDir.listFiles();
 246  0
         for (File file : files) {
 247  0
             if (!file.getName().equals("placeholder.txt")) {
 248  0
                 if (file.lastModified() < modificationTime) {
 249  0
                     file.delete();
 250  
                 }
 251  
             }
 252  
         }
 253  
         
 254  0
     }
 255  
     
 256  
     // needed for Spring injection
 257  
     /**
 258  
      * Sets the data access object
 259  
      * 
 260  
      * @param d
 261  
      */
 262  
     public void setAttachmentDao(AttachmentDao d) {
 263  0
         this.attachmentDao = d;
 264  0
     }
 265  
 
 266  
     /**
 267  
      * Retrieves a data access object
 268  
      */
 269  
     public AttachmentDao getAttachmentDao() {
 270  0
         return attachmentDao;
 271  
     }
 272  
 
 273  
     /**
 274  
      * Gets the configService attribute. 
 275  
      * @return Returns the configService.
 276  
      */
 277  
     public ConfigurationService getKualiConfigurationService() {
 278  0
         return kualiConfigurationService;
 279  
     }
 280  
 
 281  
     /**
 282  
      * Sets the configService attribute value.
 283  
      * @param configService The configService to set.
 284  
      */
 285  
     public void setKualiConfigurationService(ConfigurationService configService) {
 286  0
         this.kualiConfigurationService = configService;
 287  0
     }
 288  
 }