Создайте файл Version.java в Maven - PullRequest
54 голосов
/ 18 марта 2010

У меня есть проект Java, который я создаю, используя скрипт Ant. Я пытаюсь преобразовать проект в Maven.

Одна из задач генерирует исходный файл Java с именем Version.java, который содержит статическое строковое представление метки времени компиляции, как показано ниже:

package com.foo.bar;
public final class Version {
 public static String VERSION="100301.1046";
}

Задача Ant очень проста:

<target name="version" depends="init" description="Create Version.java">
    <echo file="src/${package.dir}/Version.java" message="package ${package.name};${line.separator}" />
    <echo file="src/${package.dir}/Version.java" append="true" message="public final class Version {${line.separator}" />
    <echo file="src/${package.dir}/Version.java"
          append="true"
          message=" public static String VERSION=&quot;${buildtime}&quot;;${line.separator}" />
    <echo file="src/${package.dir}/Version.java" append="true" message="}${line.separator}" />
    <echo message="BUILD ${buildtime}" />
</target>

Можно ли сделать что-то подобное в Maven, используя generate-sources или каким-нибудь другим простым методом?

Ответы [ 10 ]

78 голосов
/ 18 марта 2010

Не думаю, что это хороший способ решить эту проблему.

Лучший способ - поместить информацию о версии в файл properties, который будет прочитан вашей программой Java:

Ваш файл свойств будет содержать следующую строку:

myapp.version=${project.version}

Затем в вашем pom.xml укажите, что файл будет отфильтрован по Maven:

<resources>
    <resource>
        <directory>the/directory/that/contains/your/properties/file</directory>
        <filtering>true</filtering>
    </resource>
</resources>

Когда Maven создаст ваше приложение, оно заменит все ${...} на их значения. По умолчанию ${project.version} определяет версию pom.xml (т.е. значение тега <version>).

Затем в вашем Java-коде вам всего лишь нужно загрузить файл properties и получить значение свойства myApp.version.

Обратите внимание, что вы можете использовать плагин Build Number , чтобы установить что-то более "сложное", чем просто текущая версия (например, если вы хотите указать время сборки в своем свойстве).

27 голосов
/ 11 января 2012

Вы также можете использовать maven-replacer-plugin, если чувствуете, что муравей немного уродлив: Запись о помпе может быть:

<project>
  ...
  <properties>
    <version.template.file>src/main/java/com/stackoverflowVersion.java.template</version.template.file>
<version.file>src/main/java/com/stackoverflow/Version.java</version.file>
  </properties>
  ...
  <build>
    <plugins>
      <plugin>
        <groupId>com.google.code.maven-replacer-plugin</groupId>
            <artifactId>maven-replacer-plugin</artifactId>
            <version>1.4.0</version>
            <executions>                
                <execution>
                    <phase>process-sources</phase>
                    <goals>
                        <goal>replace</goal>
                    </goals>
                </execution>
            </executions>
            <configuration>
                <file>${version.template.file}</file>
                <outputFile>${version.file}</outputFile>
                <replacements>
                    <replacement>
                        <token>@buildnumber@</token>
                        <value>${svn.revision}</value>
                    </replacement>
                    <replacement>
                        <token>@buildtime@</token>
                        <value>${maven.build.timestamp}</value>
                    </replacement>
                    <replacement>
                        <token>@pomversion@</token>
                        <value>${project.version}</value>
                    </replacement>
                </replacements>                        
            </configuration>
      </plugin>
    </plugins>
  </build>
  ...
</project>

Версия.java.template может быть:

package com.stackoverflow;

public final class Version {

    public static final String build_number="@buildnumber@";

    public static final String build_time="@buildtime@";

    public static final String pomversion="@pomversion@";

}
15 голосов
/ 14 апреля 2016

Это старый вопрос, но есть другое решение, которое делает отличную работу это прекрасно (в смысле Maven): Шаблонный плагин Maven .

Использование этого плагина приводит к тому, что обработанный файл Java помещается в папку target/generated-sources, как и следовало ожидать. И это добавляет папку под generated-sources к пути сборки. Вы больше не будете регистрировать обработанный файл по ошибке.

Как пользоваться

Сначала поставьте следующее под src/main/java-templates/com/foo/bar/Version.java:

package com.foo.bar;
public final class Version {
    public static final String VERSION = "${project.version}";
}

Затем добавьте в свой POM следующее:

<build>
    <plugins>
    ...
        <plugin>
            <groupId>org.codehaus.mojo</groupId>
            <artifactId>templating-maven-plugin</artifactId>
            <version>1.0.0</version>
            <executions>
                <execution>
                    <id>filtering-java-templates</id>
                    <goals>
                        <goal>filter-sources</goal>
                    </goals>
                </execution>
            </executions>
        </plugin>
    ...
    </plugins>
</build>

Папка target/generated-sources/java-templates добавлена ​​в путь сборки Maven.

12 голосов
/ 23 августа 2011

Вот еще одно решение, которое приведет к тому же самому ответу Ральфа, используя фильтрацию свойств pom и файл шаблона:

Файл шаблона (VersionJava.template, помещенный в src / main / resources / version):

package ${ver.package.name};
public final class ${ver.class.name} {
    public static String VERSION="${ver.buildtime}";
}

Пом:

<properties>
    ...
    <ver.package.dir>com/foo/bar${project.artifactId}</ver.package.dir>
    <ver.package.name>com.foo.bar${project.artifactId}</ver.package.name>
    <ver.class.name>Version</ver.class.name>
    <ver.buildtime>${maven.build.timestamp}</ver.buildtime>
    <ver.template.dir>src/main/resources/version</ver.template.dir>
    <ver.template.file>VersionJava.template</ver.template.file>
</properties>
<build>
    <resources>
        <resource>
            <directory>src/main/resources</directory>
            <excludes>
                <exclude>version/*</exclude>
            </excludes>
        </resource>
        <resource>
            <directory>${ver.template.dir}</directory>
            <includes>
                <include>*.java</include>
            </includes>
            <filtering>true</filtering>
            <targetPath>${basedir}/src/main/java/${ver.package.dir}</targetPath>
        </resource>
    </resources>        
    <plugins>
        <plugin>
            <artifactId>maven-antrun-plugin</artifactId>
            <executions>
                <execution>
                    <phase>generate-sources</phase>
                    <configuration>
                        <tasks>
                            <copy file="${ver.template.dir}/${ver.template.file}" tofile="${ver.template.dir}/${ver.class.name}.java" />
                        </tasks>
                    </configuration>
                    <goals>
                        <goal>run</goal>
                    </goals>
                </execution>
                <execution>
                    <phase>compile</phase>
                    <configuration>
                        <tasks>
                            <delete file="${ver.template.dir}/${ver.class.name}.java" />
                        </tasks>
                    </configuration>
                    <goals>
                        <goal>run</goal>
                    </goals>
                </execution>
            </executions>
        </plugin>
    </plugins>
</build>

Теперь это может показаться чрезмерным, но это чрезвычайно универсально, и что мне больше всего нравится в этом является то, что у меня есть файл шаблона в читаемом формате (а не эхо-заявления в pom). Это также позволяет мне изменять класс версии, не меняя помп

8 голосов
/ 27 ноября 2013

На основании ответа @ superole . Это упрощенная версия без необходимости устанавливать дополнительные свойства. Просто версия проекта копируется в Version.java.

Поместите Version.java в src/main/templates:

package thepackage;

public final class Version {

 public static String VERSION="${project.version}";

}

Проинструктируйте maven о замене токенов в Version.java

<resources>
    <resource>
        <directory>src/main/templates</directory>
        <includes>
            <include>*.java</include>
        </includes>
        <filtering>true</filtering>
        <targetPath>${project.build.directory}/generated-sources/java/thepackage</targetPath>
    </resource>
</resources>

Поручите maven узнать generated-sources/java как путь сборки:

<plugin>
    <groupId>org.codehaus.mojo</groupId>
    <artifactId>build-helper-maven-plugin</artifactId>
    <version>1.8</version>
    <executions>
        <execution>
             <id>add-source</id>
            <phase>generate-sources</phase>
            <goals>
                <goal>add-source</goal>
            </goals>
            <configuration>
                <sources>
                    <source>${project.build.directory}/generated-sources/java/</source>
                </sources>
            </configuration>
        </execution>
    </executions>
</plugin>

Наконец, пусть Eclipse m2e

  • знать о новом пути сборки
  • и не попадать в бесконечный цикл сборки.

Второй момент достигается отключением с помощью maven-resources-plugin во время инкрементальной сборки eclipse.

<pluginManagement>
    <plugins>
        <plugin>
            <groupId>org.eclipse.m2e</groupId>
            <artifactId>lifecycle-mapping</artifactId>
            <version>1.0.0</version>
            <configuration>
                <lifecycleMappingMetadata>
                    <pluginExecutions>
                        <pluginExecution>
                          <pluginExecutionFilter>
                            <groupId>org.codehaus.mojo</groupId>
                            <artifactId>build-helper-maven-plugin</artifactId>
                            <versionRange>[1.0,)</versionRange>
                            <goals>
                              <goal>parse-version</goal>
                              <goal>add-source</goal>
                              <goal>maven-version</goal>
                              <goal>add-resource</goal>
                              <goal>add-test-resource</goal>
                              <goal>add-test-source</goal>
                            </goals>
                          </pluginExecutionFilter>
                          <action>
                            <execute>
                              <runOnConfiguration>true</runOnConfiguration>
                              <runOnIncremental>true</runOnIncremental>
                            </execute>
                          </action>
                        </pluginExecution>
                        <pluginExecution>
                            <pluginExecutionFilter>
                                <groupId>org.apache.maven.plugins</groupId>
                                <artifactId>maven-resources-plugin</artifactId>
                                <versionRange>[1.0.0,)</versionRange>
                                <goals>
                                    <goal>resources</goal>
                                </goals>
                            </pluginExecutionFilter>
                            <action>
                                <execute>
                                    <runOnConfiguration>true</runOnConfiguration>
                                    <runOnIncremental>false</runOnIncremental>
                                </execute>
                            </action>
                        </pluginExecution>
                    </pluginExecutions>
                </lifecycleMappingMetadata>
            </configuration>
        </plugin>
    </plugins>
</pluginManagement>

thepackage необходимо заменить вашим пакетом: также отрегулируйте targetPath соответственно. Я обнаружил, что проще установить путь в targetpath вместо множества подпапок в src/main/templates.

8 голосов
/ 18 марта 2010

После дополнительного поиска в Google, я придумал это (в pom.xml):

<plugins>
  ...
  <plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-antrun-plugin</artifactId>
    <version>1.3</version>
    <executions>
      <execution>
        <goals>
          <goal>run</goal>
        </goals>
        <phase>generate-sources</phase>
        <configuration>
          <tasks>
            <property name="src.dir" value="${project.build.sourceDirectory}" />
            <property name="package.dir" value="com/foo/bar" />
            <property name="package.name" value="com.foo.bar" />
            <property name="buildtime" value="${maven.build.timestamp}" />

            <echo file="${src.dir}/${package.dir}/Version.java" message="package ${package.name};${line.separator}" />
            <echo file="${src.dir}/${package.dir}/Version.java" append="true" message="public final class Version {${line.separator}" />
            <echo file="${src.dir}/${package.dir}/Version.java" append="true"
              message=" public static String VERSION=&quot;${buildtime}&quot;;${line.separator}" />
            <echo file="${src.dir}/${package.dir}/Version.java" append="true" message="}${line.separator}" />
            <echo message="BUILD ${buildtime}" />
          </tasks>
        </configuration>
      </execution>
    </executions>
  </plugin>
  ...
</plugins>

Кажется, что работает хорошо и создает этот файл Java:

package com.foo.bar;
public final class Version {
 public static String VERSION="100318.1211";
}
3 голосов
/ 18 марта 2010

В соответствии с предложением @Romain, вы можете прочитать версию из файла свойств (либо /META-INF/maven/groupId/artifactId/pom.properties, если вы можете подождать до упаковки или откатить свой собственный отфильтрованный файл, если не можете, либо если он не предоставляет все, что необходимо).

И хотите ли вы придерживаться своего фактического Version класса, а затем взгляните на этот поток в списке пользователей maven, который точно предлагает решение для этого (на основе плагина antrun, который вы будете привязаны к фазе generated-sources).

2 голосов
/ 30 декабря 2015

Я делаю это с помощью Maven WAR Plugin , добавляющего информацию в файл MANIFEST.MF и позже читающего этот файл MANIFEST.MF на Java:

     <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-war-plugin</artifactId>
        <version>2.6</version>
        <configuration>
           <archive>
              <manifest>
                 <addDefaultImplementationEntries>true</addDefaultImplementationEntries>
                 <addDefaultSpecificationEntries>true</addDefaultSpecificationEntries>
              </manifest>
              <manifestEntries>
                 <Build-Time>${maven.build.timestamp}</Build-Time>
              </manifestEntries>
           </archive>
        </configuration>
     </plugin>

Эта конфигурация создает следующий файл MANIFEST.MF:

Manifest-Version: 1.0
Implementation-Title: MyApp
Implementation-Version: 2.11.0-SNAPSHOT
Built-By: niestroj
Specification-Title: MyApp
Implementation-Vendor-Id: com.mycompany
Build-Time: 2017-01-09 15:30
Created-By: Apache Maven 3.0.5
Build-Jdk: 1.8.0_40
Specification-Version: 2.11

А позже я читаю это на Java следующим образом:

  try {
     Manifest manifest = new Manifest(getServletContext().getResourceAsStream("/META-INF/MANIFEST.MF"));
     Attributes attributes = manifest.getMainAttributes();
     attributes.getValue("Implementation-Version");
     attributes.getValue("Build-Time");
  } catch (IOException ex) {
     LOGGER.debug("Error reading manifest file information", ex);
  }
2 голосов
/ 30 апреля 2011
1 голос
/ 27 августа 2013

Стандартный способ сделать это с очень небольшим количеством строк XML-кода - использовать шаблон-плагин.

См. Мой ответ в Фильтрация исходного кода в Maven

Как правило, Maven описывает что вы хотите сделать. Тогда цифра как . Когда как требуется десятки или сотни строк XML, либо найдите подходящий плагин, который это делает, либо напишите его. Это было обоснование, которое создало плагин шаблон-maven: -).

...