All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.gxp.compiler..svn.text-base.CompilationSet.svn-base Maven / Gradle / Ivy

/*
 * Copyright (C) 2008 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.gxp.compiler;

import com.google.common.base.Charsets;
import com.google.common.base.Preconditions;
import com.google.common.base.Predicate;
import com.google.common.base.Predicates;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import com.google.gxp.compiler.alerts.AlertPolicy;
import com.google.gxp.compiler.alerts.AlertSink;
import com.google.gxp.compiler.alerts.SourcePosition;
import com.google.gxp.compiler.alerts.common.IOError;
import com.google.gxp.compiler.alerts.common.ProgressAlert;
import com.google.gxp.compiler.base.ExtractedMessage;
import com.google.gxp.compiler.base.OutputLanguage;
import com.google.gxp.compiler.base.TemplateName;
import com.google.gxp.compiler.base.Util;
import com.google.gxp.compiler.codegen.CodeGeneratorFactory;
import com.google.gxp.compiler.fs.FileRef;
import com.google.gxp.compiler.parser.Parser;
import com.google.gxp.compiler.servicedir.OnDemandServiceDirectory;
import com.google.gxp.compiler.servicedir.ServiceDirectory;
import com.google.transconsole.common.messages.MessageBundle;
import com.google.transconsole.common.messages.PropertiesBundleWriter;

import java.io.IOException;
import java.io.Writer;
import java.nio.charset.UnmappableCharacterException;
import java.util.*;

/**
 * A set of {@code CompilationUnit}s, and the logic for compiling (subsets of)
 * them.
 */
public class CompilationSet {
  private final Map compilationUnits;
  private final Parser parser;
  private final CodeGeneratorFactory codeGeneratorFactory;
  private final CompilationManager manager;
  private final ServiceDirectory serviceDirectory;
  private final long compilationVersion;
  private final FileRef propertiesFile;

  /**
   * Builds a {@code CompilationSet}.
   */
  public static final class Builder {
    // Required parameters to build CompilationSet.
    private final Parser parser;
    private final CodeGeneratorFactory codeGeneratorFactory;
    private final CompilationManager manager;

    // Optional parameters to build CompilationSet.
    private long compilationVersion = 0;
    private FileRef propertiesFile = null;

    /**
     * Creates a {@code CompilationSet}.
     *
     * @param parser the {@code Parser} that will be used for parsing source
     *     files
     * @param codeGeneratorFactory the {@code CodeGeneratorFactory} that will be
     *     used for constructing {@code CodeGenerator}s, which in turn will be
     *     used to generate output
     */
    public Builder(Parser parser, CodeGeneratorFactory codeGeneratorFactory,
                   CompilationManager manager) {
      this.parser = Preconditions.checkNotNull(parser);
      this.codeGeneratorFactory = Preconditions.checkNotNull(codeGeneratorFactory);
      this.manager = Preconditions.checkNotNull(manager);
    }

    /**
     * Builds the {@code CompilationSet}.
     *
     * @param inputFileRefs the {@code FileRef}s for the source files. A {@code
     *     CompilationUnit} will be created for each element.
     */
    public CompilationSet build(Iterable inputFileRefs) {
      return new CompilationSet(this, inputFileRefs);
    }

    /**
     * Builds the {@code CompilationSet}.
     *
     * @param inputFileRefs the {@code FileRef}s for the source files. A {@code
     *     CompilationUnit} will be created for each element.
     */
    public CompilationSet build(FileRef... inputFileRefs) {
      return build(Arrays.asList(inputFileRefs));
    }

    public Builder setCompilationVersion(long compilationVersion) {
      this.compilationVersion = compilationVersion;
      return this;
    }

    public Builder setPropertiesFile(FileRef propertiesFile) {
      this.propertiesFile = propertiesFile;
      return this;
    }
  }

  private CompilationSet(Builder builder, Iterable inputFileRefs) {
    this.parser = builder.parser;
    this.codeGeneratorFactory = builder.codeGeneratorFactory;
    this.manager = builder.manager;
    this.compilationVersion = builder.compilationVersion;
    this.propertiesFile = builder.propertiesFile;

    this.serviceDirectory = new OnDemandServiceDirectory(this);
    this.compilationUnits = createCompilationUnits(serviceDirectory,
                                                   inputFileRefs);
  }

  private Map
      createCompilationUnits(ServiceDirectory serviceDirectory,
                             Iterable inputFileRefs) {
    ImmutableMap.Builder
        mapBuilder = ImmutableMap.builder();
    for (FileRef inputFileRef : inputFileRefs) {
      CompilationUnit unit =
          new CompilationUnit(serviceDirectory, parser, inputFileRef,
                              compilationVersion);
      mapBuilder.put(unit.getTemplateName(), unit);
    }
    return mapBuilder.build();
  }

  /**
   * @return the {@code CompilationUnit}s that make up this {@code
   * CompilationSet}.
   */
  public final List getCompilationUnits() {
    return ImmutableList.copyOf(compilationUnits.values());
  }

  /**
   * @return the {@code CompilationUnit} with the specified TemplateName
   * CompilationSet}.
   */
  public final CompilationUnit getCompilationUnit(TemplateName.FullyQualified templateName) {
    return compilationUnits.get(templateName);
  }

  /**
   * Compiles {@code CompilationUnits} into the specified {@code
   * OutputLanguage}s. Only outputs that {@code allowedOutputPredicate} returns
   * {@code true} for will actually be generated.
   */
  public void compile(AlertSink alertSink, AlertPolicy alertPolicy,
                      Iterable outputLanguages,
                      Predicate allowedOutputPredicate) {
    Set extractMessagesFrom = Sets.newHashSet();
    List sourceNotChanged = Lists.newArrayList();
    for (CompilationUnit cUnit : getCompilationUnits()) {
      FileRef sourceFileRef = cUnit.getSourceFileRef();
      SourcePosition sourcePosition = new SourcePosition(sourceFileRef);

      for (OutputLanguage language : outputLanguages) {
        String suffix = language.getSuffix(compilationVersion);
        FileRef outputFileRef = sourceFileRef.removeExtension().addSuffix(suffix);

        if (allowedOutputPredicate.apply(outputFileRef)) {
          extractMessagesFrom.add(cUnit);
          CompilationTask task =
              new CompilationTask(cUnit, codeGeneratorFactory, language,
                                  outputFileRef);
          // if the output file does not exist (last modified = 0) or if the source has been
          // modified since the last time that the output has been generated, or if the source has
          // changed, then we need to recompile the target
          if (outputFileRef.getLastModified() < sourceFileRef.getLastModified()
              || manager.sourceChanged(task)) {
            task.execute(alertSink, alertPolicy);
          } else {
            alertSink.add(new ProgressAlert(sourcePosition, "Skipped (source unchanged)"));
            sourceNotChanged.add(task);
          }
        } else {
          alertSink.add(new ProgressAlert(sourcePosition, "Skipped (output supressed)"));
        }
      }
    }

    // For each task we didn't execute, check to see if any of the interfaces
    // it depends on have changed (which could happen as a result of
    // recompiling one of the things it depends on).
    // TODO(laurence): see whether it's possible to combine these two loops by
    // having usedInterfacesChanged not change its value.
    for (CompilationTask task : sourceNotChanged) {
      if (manager.usedInterfacesChanged(task)) {
        FileRef sourceFileRef = task.getCompilationUnit().getSourceFileRef();
        SourcePosition sourcePosition = new SourcePosition(sourceFileRef);
        alertSink.add(new ProgressAlert(sourcePosition, "Reconsidered; callees have changed"));
        task.execute(alertSink, alertPolicy);
      }
    }

    // Optionally write out a java properties file that contains
    // strings extracted from s
    if (propertiesFile != null && !extractMessagesFrom.isEmpty()) {
      SourcePosition outputPosition = new SourcePosition(propertiesFile);
      alertSink.add(new ProgressAlert(outputPosition, "Generating"));

      List messages = Lists.newArrayList();
      for (CompilationUnit cUnit : extractMessagesFrom) {
        messages.addAll(cUnit.getMessageExtractedTree().getMessages());
      }

      MessageBundle messageBundle = Util.bundleMessages(alertSink, messages);
      PropertiesBundleWriter pbw = new PropertiesBundleWriter(messageBundle);
      try {
        Writer writer = propertiesFile.openWriter(Charsets.US_ASCII);
        try {
          pbw.write(writer);
        } finally {
          writer.close();
        }
      } catch (UnmappableCharacterException uce) {
        // These are caused by coding errors, not user error.
        throw new AssertionError(uce);
      } catch (IOException iox) {
        alertSink.add(new IOError(propertiesFile, iox));
      }
      alertSink.add(new ProgressAlert(outputPosition, "Generate finished"));
    }
  }

  /**
   * Convenience method which compiles allowing all outputs.
   */
  public final void compile(AlertSink alertSink, AlertPolicy alertPolicy,
                            Iterable outputLanguages) {
    compile(alertSink, alertPolicy, outputLanguages, Predicates.alwaysTrue());
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy