blob: 002d86deac5fa663ea6c4f2ba0d40995341b7920 [file] [log] [blame]
/*
* Copyright 2013 Google Inc.
*
* Licensed under the Apache 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.apache.org/licenses/LICENSE-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 com.google.gwt.dev.cfg;
import com.google.gwt.dev.javac.CompilationUnit;
import com.google.gwt.dev.javac.CompiledClass;
import com.google.gwt.dev.jjs.CompilerIoException;
import com.google.gwt.dev.jjs.PermutationResult;
import com.google.gwt.dev.resource.Resource;
import com.google.gwt.dev.util.ZipEntryBackedObject;
import com.google.gwt.thirdparty.guava.common.base.Joiner;
import com.google.gwt.thirdparty.guava.common.base.Preconditions;
import com.google.gwt.thirdparty.guava.common.collect.LinkedHashMultimap;
import com.google.gwt.thirdparty.guava.common.collect.Maps;
import com.google.gwt.thirdparty.guava.common.collect.Multimap;
import com.google.gwt.thirdparty.guava.common.collect.Multimaps;
import com.google.gwt.thirdparty.guava.common.collect.Sets;
import com.google.gwt.thirdparty.guava.common.io.ByteStreams;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;
/**
* A library builder that writes contents to a zip file.
*/
public class ZipLibraryBuilder implements LibraryBuilder {
private class ZipLibraryWriter {
private boolean fileReady;
private final File zipFile;
private ZipOutputStream zipOutputStream;
private ZipLibraryWriter(String zipFileName) {
zipFile = new File(zipFileName);
}
private void createFileIfMissing() {
if (!zipFile.exists()) {
try {
zipFile.createNewFile();
if (!zipFile.canWrite()) {
throw new CompilerIoException(
"Created new library file " + zipFile.getPath() + " but am unable to write to it.");
}
} catch (IOException e) {
throw new CompilerIoException(
"Failed to create new library file " + zipFile.getPath() + ".", e);
}
}
}
private void createZipEntry(String entryName) {
ZipEntry zipEntry = new ZipEntry(entryName);
try {
zipOutputStream.putNextEntry(zipEntry);
} catch (Exception e) {
throw new CompilerIoException("Failed to create zip entry " + entryName + ".", e);
}
}
private synchronized void ensureFileReady() {
if (fileReady) {
return;
}
fileReady = true;
ensureParentDirectoryExists();
createFileIfMissing();
try {
zipOutputStream = new ZipOutputStream(new FileOutputStream(zipFile));
} catch (FileNotFoundException e) {
throw new CompilerIoException(
"Failed to open new library file " + zipFile.getPath() + " as a stream.", e);
}
}
private void ensureParentDirectoryExists() {
zipFile.getParentFile().mkdirs();
}
private ZipEntryBackedObject<PermutationResult> getPermutationResultHandle() {
return new ZipEntryBackedObject<PermutationResult>(zipOutputStream, zipFile.getPath(),
Libraries.PERMUTATION_RESULT_ENTRY_NAME, PermutationResult.class);
}
private boolean isTriviallySerializable(String string) {
return !string.contains(Libraries.KEY_VALUE_SEPARATOR)
&& !string.contains(Libraries.VALUE_SEPARATOR)
&& !string.contains(Libraries.LINE_SEPARATOR);
}
private void startEntry(String entryName) {
ZipEntry zipEntry = new ZipEntry(entryName);
try {
zipOutputStream.putNextEntry(zipEntry);
} catch (IOException e) {
throw new CompilerIoException("Failed to create entry " + entryName
+ " in new library file " + zipFile.getPath() + ".", e);
}
}
private void write() {
ensureFileReady();
try {
// Header
writeLibraryName();
writeVersionNumber();
// Dependency tree structure
writeDependencyLibraryNames();
// Precompiled sources
writeClassFilePaths();
writeCompilationUnitTypeNames();
// Resources
writeBuildResources();
writeBuildResourcePaths();
writePublicResources();
writePublicResourcePaths();
// Generator related
writeNewBindingPropertyValuesByName();
writeNewConfigurationPropertyValuesByName();
writeReboundTypeNames();
writeRanGeneratorNames();
} finally {
try {
zipOutputStream.close();
} catch (IOException e) {
throw new CompilerIoException(
"Failed to close new library file " + zipFile.getPath() + ".", e);
}
}
}
private void writeBuildResourcePaths() {
writeStringSet(Libraries.BUILD_RESOURCE_PATHS_ENTRY_NAME, buildResourcesByPath.keySet());
}
private void writeBuildResources() {
writeResources("build", Libraries.DIRECTORY_BUILD_RESOURCES, buildResourcesByPath);
}
private void writeClassFile(String classFilePath, byte[] classBytes) {
try {
ensureFileReady();
startEntry(Libraries.computeClassFileEntryName(classFilePath));
zipOutputStream.write(classBytes);
} catch (IOException e) {
throw new CompilerIoException("Failed to write class file " + classFilePath
+ " to new library file " + zipFile.getPath() + ".", e);
}
}
private void writeClassFilePaths() {
writeStringSet(Libraries.CLASS_FILE_PATHS_ENTRY_NAME, classFilePaths);
writeStringSet(Libraries.SUPER_SOURCE_CLASS_FILE_PATHS_ENTRY_NAME, superSourceClassFilePaths);
}
private void writeCompilationUnitFile(CompilationUnit compilationUnit) {
ensureFileReady();
startEntry(Libraries.computeCompilationUnitEntryName(compilationUnit.getTypeName()));
try {
ObjectOutputStream out = new ObjectOutputStream(zipOutputStream);
out.writeObject(compilationUnit);
} catch (IOException e) {
throw new CompilerIoException("Failed to serialize compilation unit "
+ compilationUnit.getTypeName() + " in new library " + zipFile.getPath() + ".", e);
}
}
private void writeCompilationUnitTypeNames() {
writeStringSet(Libraries.COMPILATION_UNIT_TYPE_NAMES_ENTRY_NAME, compilationUnitTypeNames);
writeStringSet(Libraries.SUPER_SOURCE_COMPILATION_UNIT_TYPE_NAMES_ENTRY_NAME,
superSourceCompilationUnitTypeNames);
}
private void writeDependencyLibraryNames() {
writeStringSet(Libraries.DEPENDENCY_LIBRARY_NAMES_ENTRY_NAME, dependencyLibraryNames);
}
private void writeLibraryName() {
writeString(Libraries.LIBRARY_NAME_ENTRY_NAME, libraryName);
}
private void writeNewBindingPropertyValuesByName() {
writeStringMultimap(
Libraries.NEW_BINDING_PROPERTY_VALUES_BY_NAME_ENTRY_NAME, newBindingPropertyValuesByName);
}
private void writeNewConfigurationPropertyValuesByName() {
writeStringMultimap(Libraries.NEW_CONFIGURATION_PROPERTY_VALUES_BY_NAME_ENTRY_NAME,
newConfigurationPropertyValuesByName);
}
private void writePublicResourcePaths() {
writeStringSet(Libraries.PUBLIC_RESOURCE_PATHS_ENTRY_NAME, publicResourcesByPath.keySet());
}
private void writePublicResources() {
writeResources("public", Libraries.DIRECTORY_PUBLIC_RESOURCES, publicResourcesByPath);
}
private void writeRanGeneratorNames() {
writeStringSet(Libraries.RAN_GENERATOR_NAMES_ENTRY_NAME, ranGeneratorNames);
}
private void writeReboundTypeNames() {
writeStringSet(Libraries.REBOUND_TYPE_NAMES_ENTRY_NAME, reboundTypeNames);
}
private void writeResources(
String typeName, String directory, Map<String, Resource> resourcesByPath) {
for (Resource resource : resourcesByPath.values()) {
startEntry(directory + resource.getPath());
try {
ByteStreams.copy(resource.openContents(), zipOutputStream);
} catch (IOException e) {
throw new CompilerIoException("Failed to copy " + typeName + " resource "
+ resource.getPath() + " into new library file " + zipFile.getPath() + ".", e);
}
}
}
private void writeString(String entryName, String string) {
createZipEntry(entryName);
try {
zipOutputStream.write(string.getBytes());
} catch (IOException e) {
throw new CompilerIoException("Failed to write " + entryName + " as a String.", e);
}
}
private void writeStringMultimap(String entryName, Multimap<String, String> stringMultimap) {
Map<String, Collection<String>> stringListsByString = stringMultimap.asMap();
createZipEntry(entryName);
Iterator<Entry<String, Collection<String>>> entryIterator =
stringListsByString.entrySet().iterator();
try {
while (entryIterator.hasNext()) {
Entry<String, Collection<String>> entry = entryIterator.next();
String key = entry.getKey();
Preconditions.checkState(
isTriviallySerializable(key), "Nonserializable characters in key '%s'.", key);
zipOutputStream.write(key.getBytes());
boolean first = true;
for (String value : entry.getValue()) {
Preconditions.checkState(
isTriviallySerializable(value), "Nonserializable characters in value '%s'.", value);
if (first) {
first = false;
zipOutputStream.write(Libraries.KEY_VALUE_SEPARATOR.getBytes());
} else {
zipOutputStream.write(Libraries.VALUE_SEPARATOR.getBytes());
}
zipOutputStream.write(value.getBytes());
}
if (entryIterator.hasNext()) {
zipOutputStream.write(Libraries.LINE_SEPARATOR.getBytes());
}
}
} catch (IOException e) {
throw new CompilerIoException("Failed to write " + entryName + " as a String multimap.", e);
}
}
private void writeStringSet(String entryName, Set<String> stringSet) {
createZipEntry(entryName);
try {
for (String string : stringSet) {
Preconditions.checkState(isTriviallySerializable(string),
"Nonserializable characters in string '%s'.", string);
}
zipOutputStream.write(Joiner.on(Libraries.LINE_SEPARATOR).join(stringSet).getBytes());
} catch (IOException e) {
throw new CompilerIoException("Failed to write " + entryName + " as a String set.", e);
}
}
private void writeVersionNumber() {
writeString(
Libraries.VERSION_NUMBER_ENTRY_NAME, Integer.toString(ZipLibraries.versionNumber));
}
}
private Map<String, Resource> buildResourcesByPath = Maps.newHashMap();
private Set<String> classFilePaths = Sets.newHashSet();
private Map<String, CompilationUnit> compilationUnitsByTypeName = Maps.newHashMap();
private Set<String> compilationUnitTypeNames = Sets.newLinkedHashSet();
private Set<String> dependencyLibraryNames = Sets.newHashSet();
private String libraryName;
private Multimap<String, String> newBindingPropertyValuesByName = LinkedHashMultimap.create();
private Multimap<String, String> newConfigurationPropertyValuesByName =
LinkedHashMultimap.create();
private ZipEntryBackedObject<PermutationResult> permutationResultHandle;
private Map<String, Resource> publicResourcesByPath = Maps.newHashMap();
private Set<String> ranGeneratorNames = Sets.newHashSet();
private Set<String> reboundTypeNames = Sets.newHashSet();
private Set<String> superSourceClassFilePaths = Sets.newHashSet();
private Set<String> superSourceCompilationUnitTypeNames = Sets.newLinkedHashSet();
private ZipLibraryWriter zipLibraryWriter;
public ZipLibraryBuilder(String fileName) {
zipLibraryWriter = new ZipLibraryWriter(fileName);
}
@Override
public void addBuildResource(Resource buildResource) {
buildResourcesByPath.put(buildResource.getPath(), buildResource);
}
@Override
public void addCompilationUnit(CompilationUnit compilationUnit) {
// The ResourceOracle system should already have deduped input source with colliding names, but
// it's best to be sure.
Preconditions.checkState(
!compilationUnitsByTypeName.containsKey(compilationUnit.getTypeName()));
if (compilationUnit.isSuperSource()) {
superSourceCompilationUnitTypeNames.add(compilationUnit.getTypeName());
} else {
compilationUnitTypeNames.add(compilationUnit.getTypeName());
}
compilationUnitsByTypeName.put(compilationUnit.getTypeName(), compilationUnit);
for (CompiledClass compiledClass : compilationUnit.getCompiledClasses()) {
if (compilationUnit.isSuperSource()) {
String classFilePath = compiledClass.getInternalName();
superSourceClassFilePaths.add(Libraries.computeClassFileName(classFilePath));
zipLibraryWriter.writeClassFile(classFilePath, compiledClass.getBytes());
} else {
String classFilePath = compiledClass.getInternalName();
classFilePaths.add(Libraries.computeClassFileName(classFilePath));
zipLibraryWriter.writeClassFile(classFilePath, compiledClass.getBytes());
}
}
zipLibraryWriter.writeCompilationUnitFile(compilationUnit);
}
@Override
public void addDependencyLibraryName(String libraryName) {
dependencyLibraryNames.add(libraryName);
}
@Override
public void addDependencyLibraryNames(Set<String> dependencyLibraryNames) {
this.dependencyLibraryNames.addAll(dependencyLibraryNames);
}
@Override
public void addNewBindingPropertyValuesByName(
String propertyName, Iterable<String> propertyValues) {
newBindingPropertyValuesByName.putAll(propertyName, propertyValues);
}
@Override
public void addNewConfigurationPropertyValuesByName(
String propertyName, Iterable<String> propertyValues) {
newConfigurationPropertyValuesByName.putAll(propertyName, propertyValues);
}
@Override
public void addPublicResource(Resource publicResource) {
publicResourcesByPath.put(publicResource.getPath(), publicResource);
}
@Override
public void addRanGeneratorName(String generatorName) {
ranGeneratorNames.add(generatorName);
}
@Override
public Multimap<String, String> getNewBindingPropertyValuesByName() {
return Multimaps.unmodifiableMultimap(newBindingPropertyValuesByName);
}
@Override
public Multimap<String, String> getNewConfigurationPropertyValuesByName() {
return Multimaps.unmodifiableMultimap(newConfigurationPropertyValuesByName);
}
@Override
public ZipEntryBackedObject<PermutationResult> getPermutationResultHandle() {
if (permutationResultHandle == null) {
permutationResultHandle = zipLibraryWriter.getPermutationResultHandle();
}
return permutationResultHandle;
}
@Override
public Set<String> getReboundTypeNames() {
return Collections.unmodifiableSet(reboundTypeNames);
}
@Override
public void setLibraryName(String libraryName) {
this.libraryName = libraryName;
}
@Override
public void setReboundTypeNames(Set<String> reboundTypeNames) {
this.reboundTypeNames = reboundTypeNames;
}
@Override
public void write() {
zipLibraryWriter.write();
}
}