/********************************************************************************** * $URL: https://source.sakaiproject.org/svn/sam/trunk/samigo-cp/src/java/org/sakaiproject/tool/assessment/contentpackaging/ImportService.java $ * $Id: ImportService.java 106463 2012-04-02 12:20:09Z david.horwitz@uct.ac.za $ *********************************************************************************** * * Copyright (c) 2007, 2008, 2009 The Sakai Foundation * * Licensed under the Educational Community License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.opensource.org/licenses/ECL-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * **********************************************************************************/ package org.sakaiproject.tool.assessment.contentpackaging; import java.io.ByteArrayInputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.util.ArrayList; import java.util.Set; import java.util.TreeSet; import java.util.zip.ZipEntry; import java.util.zip.ZipInputStream; import javax.faces.context.ExternalContext; import javax.faces.context.FacesContext; import javax.servlet.ServletContext; import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.sakaiproject.tool.assessment.facade.AgentFacade; import org.w3c.dom.Document; import org.w3c.dom.NamedNodeMap; import org.w3c.dom.Node; import org.w3c.dom.NodeList; /** * <p>Copyright: Copyright (c) 2007 Sakai</p> * @version $Id: ImportService.java 106463 2012-04-02 12:20:09Z david.horwitz@uct.ac.za $ */ public class ImportService { private static Log log = LogFactory.getLog(ImportService.class); private String qtiFilename; public String unzipImportFile(String filename) { FileInputStream fileInputStream = null; FileOutputStream ofile = null; ZipInputStream zipStream = null; ZipEntry entry = null; ExternalContext external = FacesContext.getCurrentInstance().getExternalContext(); StringBuilder unzipLocation = new StringBuilder((String)((ServletContext)external.getContext()).getAttribute("FILEUPLOAD_REPOSITORY_PATH")); log.debug("****"+unzipLocation); unzipLocation.append("/jsf/upload_tmp/qti_imports/"); unzipLocation.append(AgentFacade.getAgentString()); unzipLocation.append("/unzip_files/"); unzipLocation.append(Long.toString(new java.util.Date().getTime())); try { fileInputStream = new FileInputStream(new File(filename)); byte[] data = new byte[fileInputStream.available()]; fileInputStream.read(data, 0, fileInputStream.available()); File dir = new File(unzipLocation.toString()); // directory where file would be saved if (!dir.exists()) { if (!dir.mkdirs()) { log.error("unable to mkdir " + dir.getPath()); } } Set dirsMade = new TreeSet(); zipStream = new ZipInputStream(new ByteArrayInputStream(data)); entry = (ZipEntry) zipStream.getNextEntry(); // Get the name of the imported zip file name. The value of "filename" has timestamp append to it. String tmpName = filename.substring(filename.lastIndexOf("/") + 1); qtiFilename = "exportAssessment.xml"; ArrayList xmlFilenames = new ArrayList(); while (entry != null) { String zipName = entry.getName(); int ix = zipName.lastIndexOf('/'); if (ix > 0) { String dirName = zipName.substring(0, ix); if (!dirsMade.contains(dirName)) { File d = new File(dir.getPath() + "/" + dirName); // If it already exists as a dir, don't do anything if (!(d.exists() && d.isDirectory())) { // Try to create the directory, warn if it fails if (!d.mkdirs()) { log.error("unable to mkdir " + dir.getPath() + "/" + dirName); } dirsMade.add(dirName); } } } File zipEntryFile = new File(dir.getPath() + "/" + entry.getName()); if (!zipEntryFile.isDirectory()) { ofile = new FileOutputStream(zipEntryFile); byte[] buffer = new byte[1024 * 10]; int bytesRead; while ((bytesRead = zipStream.read(buffer)) != -1) { ofile.write(buffer, 0, bytesRead); } } // Now try to get the QTI xml file name from the imsmanifest.xml if ("imsmanifest.xml".equals(entry.getName())) { DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance(); try { DocumentBuilder db = dbf.newDocumentBuilder(); Document doc = db.parse(zipEntryFile); doc.getDocumentElement().normalize(); NodeList nodeLst = doc.getElementsByTagName("resource"); Node fstNode = nodeLst.item(0); NamedNodeMap namedNodeMap= fstNode.getAttributes(); qtiFilename = namedNodeMap.getNamedItem("href").getNodeValue(); } catch (Exception e) { log.error("error parsing imsmanifest.xml"); } } else if (entry.getName() != null && entry.getName().trim().endsWith(".xml")) { xmlFilenames.add(entry.getName().trim()); } zipStream.closeEntry(); entry = zipStream.getNextEntry(); } // If the QTI file doesn't exist in the zip, // we guess the name might be either exportAssessment.xml or the same as the zip file name if (!xmlFilenames.contains(qtiFilename.trim())) { if (xmlFilenames.contains("exportAssessment.xml")) { qtiFilename = "exportAssessment.xml"; } else { qtiFilename = tmpName.substring(0, tmpName.lastIndexOf("_")) + ".xml"; } } } catch (FileNotFoundException e) { log.error(e.getMessage()); } catch (IOException e) { log.error(e.getMessage()); e.printStackTrace(); } finally { if (ofile != null) { try { ofile.close(); } catch (IOException e) { log.error(e.getMessage()); } } if (fileInputStream != null) { try { fileInputStream.close(); } catch (IOException e) { log.error(e.getMessage()); } } if (zipStream != null) { try { zipStream.close(); } catch (IOException e) { log.error(e.getMessage()); } } } return unzipLocation.toString(); } public String setQTIFilename() { return qtiFilename; } public String getQTIFilename() { return qtiFilename; } }