package com.google.gridworks.commands.project; import java.io.BufferedInputStream; import java.io.File; import java.io.FileInputStream; import java.io.FilterInputStream; import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.io.Reader; import java.io.Serializable; import java.io.UnsupportedEncodingException; import java.net.URL; import java.net.URLConnection; import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import java.util.HashMap; import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.Properties; import java.util.Map.Entry; import java.util.zip.GZIPInputStream; import java.util.zip.ZipEntry; import java.util.zip.ZipInputStream; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.apache.commons.fileupload.FileItemIterator; import org.apache.commons.fileupload.FileItemStream; import org.apache.commons.fileupload.servlet.ServletFileUpload; import org.apache.commons.fileupload.util.Streams; import org.apache.tools.bzip2.CBZip2InputStream; import org.apache.tools.tar.TarEntry; import org.apache.tools.tar.TarInputStream; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import com.google.gridworks.ProjectManager; import com.google.gridworks.ProjectMetadata; import com.google.gridworks.commands.Command; import com.google.gridworks.importers.Importer; import com.google.gridworks.importers.TsvCsvImporter; import com.google.gridworks.model.Project; import com.google.gridworks.util.IOUtils; import com.google.gridworks.util.ParsingUtilities; import com.ibm.icu.text.CharsetDetector; import com.ibm.icu.text.CharsetMatch; public class CreateProjectCommand extends Command { final static Logger logger = LoggerFactory.getLogger("create-project_command"); static final private Map importers = new HashMap(); private static final String[][] importerNames = { {"ExcelImporter", "com.google.gridworks.importers.ExcelImporter"}, {"XmlImporter", "com.google.gridworks.importers.XmlImporter"}, {"RdfTripleImporter", "com.google.gridworks.importers.RdfTripleImporter"}, {"MarcImporter", "com.google.gridworks.importers.MarcImporter"}, {"TsvCsvImporter", "com.google.gridworks.importers.TsvCsvImporter"}, }; static { registerImporters(importerNames); } static public boolean registerImporters(String[][] importers) { boolean status = true; for (String[] importer : importerNames) { String importerName = importer[0]; String className = importer[1]; logger.debug("Loading command " + importerName + " class: " + className); Importer cmd; try { // TODO: May need to use the servlet container's class loader here cmd = (Importer) Class.forName(className).newInstance(); } catch (InstantiationException e) { logger.error("Failed to load importer class " + className, e); status = false; continue; } catch (IllegalAccessException e) { logger.error("Failed to load importer class " + className, e); status = false; continue; } catch (ClassNotFoundException e) { logger.error("Failed to load importer class " + className, e); status = false; continue; } status |= registerImporter(importerName, cmd); } return status; } /** * Register a single importer. * * @param name * importer verb for importer * @param commandObject * object implementing the importer * @return true if importer was loaded and registered successfully */ static public boolean registerImporter(String name, Importer importerObject) { if (importers.containsKey(name)) { return false; } importers.put(name, importerObject); return true; } // Currently only for test purposes static protected boolean unregisterImporter(String verb) { return importers.remove(verb) != null; } @Override public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { ProjectManager.singleton.setBusy(true); try { /* * The uploaded file is in the POST body as a "file part". If * we call request.getParameter() then the POST body will get * read and we won't have a chance to parse the body ourselves. * This is why we have to parse the URL for parameters ourselves. * Don't call request.getParameter() before calling internalImport(). */ Properties options = ParsingUtilities.parseUrlParameters(request); Project project = new Project(); internalImport(request, project, options); /* * The import process above populates options with parameters * in the POST body. That's why we're constructing the project * metadata object after calling internalImport(). */ ProjectMetadata pm = new ProjectMetadata(); pm.setName(options.getProperty("project-name")); pm.setPassword(options.getProperty("project-password")); pm.setEncoding(options.getProperty("encoding")); pm.setEncodingConfidence(options.getProperty("encoding_confidence")); ProjectManager.singleton.registerProject(project, pm); project.update(); redirect(response, "/project?project=" + project.id); } catch (Exception e) { redirect(response, "/error.html?redirect=index.html&msg=" + ParsingUtilities.encode("Failed to import file: " + e.getLocalizedMessage()) ); e.printStackTrace(); } finally { ProjectManager.singleton.setBusy(false); } } protected void internalImport( HttpServletRequest request, Project project, Properties options ) throws Exception { ServletFileUpload upload = new ServletFileUpload(); String url = null; FileItemIterator iter = upload.getItemIterator(request); while (iter.hasNext()) { FileItemStream item = iter.next(); String name = item.getFieldName().toLowerCase(); InputStream stream = item.openStream(); if (item.isFormField()) { if (name.equals("raw-text")) { Reader reader = new InputStreamReader(stream,"UTF-8"); try { internalInvokeImporter(project, new TsvCsvImporter(), options, reader); } finally { reader.close(); } } else if (name.equals("url")) { url = Streams.asString(stream); } else { options.put(name, Streams.asString(stream)); } } else { String fileName = item.getName().toLowerCase(); try { internalImportFile(project, options, fileName, stream); } finally { stream.close(); } } } if (url != null && url.length() > 0) { internalImportURL(request, project, options, url); } } static class SafeInputStream extends FilterInputStream { public SafeInputStream(InputStream stream) { super(stream); } @Override public void close() { // some libraries attempt to close the input stream while they can't // read anymore from it... unfortunately this behavior prevents // the zip input stream from functioning correctly so we just have // to ignore those close() calls and just close it ourselves // forcefully later } public void reallyClose() throws IOException { super.close(); } } protected void internalImportFile( Project project, Properties options, String fileName, InputStream inputStream ) throws Exception { logger.info("Importing '{}'", fileName); if (fileName.endsWith(".zip") || fileName.endsWith(".tar.gz") || fileName.endsWith(".tgz") || fileName.endsWith(".tar.bz2")) { // first, save the file on disk, since we need two passes and we might // not have enough memory to keep it all in there File file = save(inputStream); // in the first pass, gather statistics about what files are in there // unfortunately, we have to rely on files extensions, which is horrible but // better than nothing HashMap ext_map = new HashMap(); FileInputStream fis = new FileInputStream(file); InputStream is = getStream(fileName, fis); // NOTE(SM): unfortunately, java.io does not provide any generalized class for // archive-like input streams so while both TarInputStream and ZipInputStream // behave precisely the same, there is no polymorphic behavior so we have // to treat each instance explicitly... one of those times you wish you had // closures try { if (is instanceof TarInputStream) { TarInputStream tis = (TarInputStream) is; TarEntry te; while ((te = tis.getNextEntry()) != null) { if (!te.isDirectory()) { mapExtension(te.getName(),ext_map); } } } else if (is instanceof ZipInputStream) { ZipInputStream zis = (ZipInputStream) is; ZipEntry ze; while ((ze = zis.getNextEntry()) != null) { if (!ze.isDirectory()) { mapExtension(ze.getName(),ext_map); } } } } finally { try { is.close(); fis.close(); } catch (IOException e) {} } // sort extensions by how often they appear List> values = new ArrayList>(ext_map.entrySet()); Collections.sort(values, new ValuesComparator()); if (values.size() == 0) { throw new RuntimeException("The archive contains no files."); } // this will contain the set of extensions we'll load from the archive HashSet exts = new HashSet(); // find the extension that is most frequent or those who share the highest frequency value if (values.size() == 1) { exts.add(values.get(0).getKey()); } else { Entry most_frequent = values.get(0); Entry second_most_frequent = values.get(1); if (most_frequent.getValue() > second_most_frequent.getValue()) { // we have a winner exts.add(most_frequent.getKey()); } else { // multiple extensions have the same frequency int winning_frequency = most_frequent.getValue(); for (Entry e : values) { if (e.getValue() == winning_frequency) { exts.add(e.getKey()); } } } } logger.info("Most frequent extensions: {}", exts.toString()); // second pass, load the data for real is = getStream(fileName, new FileInputStream(file)); SafeInputStream sis = new SafeInputStream(is); try { if (is instanceof TarInputStream) { TarInputStream tis = (TarInputStream) is; TarEntry te; while ((te = tis.getNextEntry()) != null) { if (!te.isDirectory()) { String name = te.getName(); String ext = getExtension(name)[1]; if (exts.contains(ext)) { internalImportFile(project, options, name, sis); } } } } else if (is instanceof ZipInputStream) { ZipInputStream zis = (ZipInputStream) is; ZipEntry ze; while ((ze = zis.getNextEntry()) != null) { if (!ze.isDirectory()) { String name = ze.getName(); String ext = getExtension(name)[1]; if (exts.contains(ext)) { internalImportFile(project, options, name, sis); } } } } } finally { try { sis.reallyClose(); } catch (IOException e) {} } } else if (fileName.endsWith(".gz")) { internalImportFile(project, options, getExtension(fileName)[0], new GZIPInputStream(inputStream)); } else if (fileName.endsWith(".bz2")) { internalImportFile(project, options, getExtension(fileName)[0], new CBZip2InputStream(inputStream)); } else { load(project, options, fileName, inputStream); } } public static class ValuesComparator implements Comparator>, Serializable { private static final long serialVersionUID = 8845863616149837657L; public int compare(Entry o1, Entry o2) { return o2.getValue() - o1.getValue(); } } private void load(Project project, Properties options, String fileName, InputStream inputStream) throws Exception { Importer importer = guessImporter(null, fileName); internalInvokeImporter(project, importer, options, inputStream, null); } private File save(InputStream is) throws IOException { File temp = this.servlet.getTempFile(Long.toString(System.currentTimeMillis())); temp.deleteOnExit(); IOUtils.copy(is,temp); is.close(); return temp; } private void mapExtension(String name, Map ext_map) { String ext = getExtension(name)[1]; if (ext_map.containsKey(ext)) { ext_map.put(ext, ext_map.get(ext) + 1); } else { ext_map.put(ext, 1); } } private InputStream getStream(String fileName, InputStream is) throws IOException { if (fileName.endsWith(".tar.gz") || fileName.endsWith(".tgz")) { return new TarInputStream(new GZIPInputStream(is)); } else if (fileName.endsWith(".tar.bz2")) { return new TarInputStream(new CBZip2InputStream(is)); } else { return new ZipInputStream(is); } } private String[] getExtension(String filename) { String[] result = new String[2]; int ext_index = filename.lastIndexOf('.'); result[0] = (ext_index == -1) ? filename : filename.substring(0,ext_index); result[1] = (ext_index == -1) ? "" : filename.substring(ext_index + 1); return result; } protected void internalImportURL( HttpServletRequest request, Project project, Properties options, String urlString ) throws Exception { URL url = new URL(urlString); URLConnection connection = null; try { connection = url.openConnection(); connection.setConnectTimeout(5000); connection.connect(); } catch (Exception e) { throw new Exception("Cannot connect to " + urlString, e); } InputStream inputStream = null; try { inputStream = connection.getInputStream(); } catch (Exception e) { throw new Exception("Cannot retrieve content from " + url, e); } try { Importer importer = guessImporter( connection.getContentType(), url.getPath() ); internalInvokeImporter(project, importer, options, inputStream, connection.getContentEncoding()); } finally { inputStream.close(); } } protected void internalInvokeImporter( Project project, Importer importer, Properties options, InputStream rawInputStream, String encoding ) throws Exception { if (importer.takesReader()) { BufferedInputStream inputStream = new BufferedInputStream(rawInputStream); // NOTE(SM): The ICU4J char detection code requires the input stream to support mark/reset. // Unfortunately, not all ServletInputStream implementations are marking, so we need do // this memory-expensive wrapping to make it work. It's far from ideal but I don't have // a more efficient solution. byte[] bytes = new byte[1024 * 4]; inputStream.mark(bytes.length); inputStream.read(bytes); inputStream.reset(); CharsetDetector detector = new CharsetDetector(); detector.setDeclaredEncoding("utf8"); // most of the content on the web is encoded in UTF-8 so start with that Reader reader = null; CharsetMatch[] charsetMatches = detector.setText(bytes).detectAll(); for (CharsetMatch charsetMatch : charsetMatches) { try { reader = new InputStreamReader(inputStream, charsetMatch.getName()); options.setProperty("encoding", charsetMatch.getName()); options.setProperty("encoding_confidence", Integer.toString(charsetMatch.getConfidence())); logger.info("Best encoding guess: {} [confidence: {}]", charsetMatch.getName(), charsetMatch.getConfidence()); break; } catch (UnsupportedEncodingException e) { // silent } } if (reader == null) { // when all else fails reader = encoding != null ? new InputStreamReader(inputStream, encoding) : new InputStreamReader(inputStream); } importer.read(reader, project, options); } else { importer.read(rawInputStream, project, options); } } protected void internalInvokeImporter( Project project, Importer importer, Properties options, Reader reader ) throws Exception { importer.read(reader, project, options); } protected Importer guessImporter(String contentType, String fileName) { for(Importer i : importers.values()){ if(i.canImportData(contentType, fileName)){ return i; } } return new TsvCsvImporter(); //default } }