Несколько развертываний в Maven - PullRequest
9 голосов
/ 11 марта 2009

У нас есть внутреннее артефактное хранилище. На данный момент все снимки будут размещены там. Мы также хотим иметь другой сервер с веб-интерфейсом и скопировать на него созданные артефакты.

Для наших сборок мы используем Hudson, но действие пост-сборки "Развертывание артефактов в репозитории Maven" вместе с scp не работает Поэтому возникает вопрос, как сделать это каким-то другим элегантным способом. Почему maven не может иметь несколько репозиториев? Есть идеи?

Самое приятное было бы, если бы артефакт поддерживал (автоматический!) Добавочный экспорт в стандартный репозиторий maven после каждого нового развертывания.

Ответы [ 4 ]

12 голосов
/ 13 марта 2009

Я не думаю, что maven поддерживает развертывание в нескольких репозиториях для одного профиля, но, возможно, профили могут изменить идентификатор и URL-адрес репозитория.

  <distributionManagement>
    <repository>
      <id>${repo-id}</id>
      <name>${repo-name}</name>
      <url>${repo-url}</url>
    </repository>
  </distributionManagement>

Maven Deployment

Затем используйте профили, чтобы выбрать репо для развертывания:

<profiles>
  <profile>
    <id>repo1</id>
    <activation>
      <activeByDefault>true</activeByDefault>
    </activation>
    <properties>
      <repo-id>repo1</repo-id>
      <repo-name>Repo1 Name </repo-name>
      <repo-url>http://url.com/maven2</repo-url>
    </properties>
  </profile>
  <profile>
    <id>repo2</id>
    <properties>
      <repo-id>repo2</repo-id>
      <repo-name>Repo2 Name </repo-name>
      <repo-url>http://url2.com/maven2</repo-url>
    </properties>
  </profile>
</profiles>

Профили Maven

3 голосов
/ 11 августа 2009

Если вы хотите использовать пользовательский плагин, вы можете настроить Maven для развертывания в список «зеркальных» расположений одновременно со стандартным развертыванием. Я бы порекомендовал определить это в профиле, чтобы вы могли контролировать, какие развертывания отражаются (это может быть нецелесообразно делать в каждой сборке).

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

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
  <modelVersion>4.0.0</modelVersion>
  <groupId>name.seller.rich</groupId>
  <artifactId>maven-mirror-plugin</artifactId>
  <packaging>maven-plugin</packaging>
  <version>0.0.1</version>
  <dependencies>
    <dependency>
      <groupId>org.apache.maven</groupId>
      <artifactId>maven-plugin-api</artifactId>
      <version>2.2.0</version>
    </dependency>
    <dependency>
      <groupId>org.apache.maven.plugins</groupId>
      <artifactId>maven-deploy-plugin</artifactId>
      <version>2.4</version>
    </dependency>
  </dependencies>
</project>

В src / main / java определите Mojo. Приведенный ниже код объявляет «зеркальную» цель: для зеркалирования развертывания артефакта требуется список элементов mirrorRepository (содержащий repositoryId и url). Плагин использует тот же подход к развертыванию, что и maven-deploy-plugin, и принимает большинство тех же параметров.

Обратите внимание, что вам все равно нужно определить сервер в вашем файле settings.xml для каждого репозитория с соответствующими разрешениями на развертывание, иначе сборка не удастся!

package name.seller.rich;

import java.io.File;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import org.apache.maven.artifact.Artifact;
import org.apache.maven.artifact.deployer.ArtifactDeployer;
import org.apache.maven.artifact.deployer.ArtifactDeploymentException;
import org.apache.maven.artifact.metadata.ArtifactMetadata;
import org.apache.maven.artifact.repository.ArtifactRepository;
import org.apache.maven.artifact.repository.ArtifactRepositoryFactory;
import org.apache.maven.artifact.repository.layout.ArtifactRepositoryLayout;
import org.apache.maven.plugin.AbstractMojo;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.project.MavenProject;
import org.apache.maven.project.artifact.ProjectArtifactMetadata;

/**
 * @goal mirror
 * @phase deploy
 */
public class MirrorMojo extends AbstractMojo {
    /**
     * @parameter expression=
     *            "${component.org.apache.maven.artifact.deployer.ArtifactDeployer}"
     * @required
     * @readonly
     */
    private ArtifactDeployer deployer;

    /**
     * Map that contains the layouts
     * 
     * @component role=
     *            "org.apache.maven.artifact.repository.layout.ArtifactRepositoryLayout"
     */
    private Map repositoryLayouts;

    /**
     * Component used to create a repository
     * 
     * @component
     */
    private ArtifactRepositoryFactory repositoryFactory;

    /**
     * The type of remote repository layout to deploy to. Try <i>legacy</i> for
     * a Maven 1.x-style repository layout.
     * 
     * @parameter expression="${repositoryLayout}" default-value="default"
     * @required
     */
    private String repositoryLayout;

    /**
     * Parameter used to update the metadata to make the artifact as release.
     * 
     * @parameter expression="${updateReleaseInfo}" default-value="false"
     */
    private boolean updateReleaseInfo;

    /**
     * Whether to deploy snapshots with a unique version or not.
     * 
     * @parameter expression="${uniqueVersion}" default-value="true"
     */
    private boolean uniqueVersion;

    /**
     * @parameter expression="${mirrorRepositories}"
     * @required
     */
    private MirrorRepository[] mirrorRepositories;

    /**
     * @parameter expression="${localRepository}"
     * @required
     * @readonly
     */
    private ArtifactRepository localRepository;

    /**
     * @parameter expression="${project}"
     * @required
     * @readonly
     */
    private MavenProject project;

    /**
     * Deploy all artifacts for the project to each mirror repository.
     */
    public void execute() throws MojoExecutionException, MojoFailureException {
        ArtifactRepositoryLayout layout;

        layout = (ArtifactRepositoryLayout) repositoryLayouts
                .get(repositoryLayout);

        for (int i = 0; i < mirrorRepositories.length; i++) {
            MirrorRepository mirrorRepository = mirrorRepositories[i];

            ArtifactRepository deploymentRepository = repositoryFactory
                    .createDeploymentArtifactRepository(mirrorRepository
                            .getRepositoryId(), mirrorRepository.getUrl(),
                            layout, uniqueVersion);

            String protocol = deploymentRepository.getProtocol();

            if ("".equals(protocol) || protocol == null) {
                throw new MojoExecutionException("No transfer protocol found.");
            }

            deployToRepository(deploymentRepository);
        }

    }

    /**
     * Deploy all artifacts to the passed repository.
     */
    private void deployToRepository(ArtifactRepository repo)
            throws MojoExecutionException {
        String protocol = repo.getProtocol();

        if (protocol.equalsIgnoreCase("scp")) {
            File sshFile = new File(System.getProperty("user.home"), ".ssh");

            if (!sshFile.exists()) {
                sshFile.mkdirs();
            }
        }

        File pomFile = project.getFile();
        Artifact artifact = project.getArtifact();
        // Deploy the POM
        boolean isPomArtifact = "pom".equals(project.getPackaging());
        if (!isPomArtifact) {
            ArtifactMetadata metadata = new ProjectArtifactMetadata(artifact,
                    pomFile);
            artifact.addMetadata(metadata);
        }

        if (updateReleaseInfo) {
            artifact.setRelease(true);
        }

        try {
            List attachedArtifacts = project.getAttachedArtifacts();

            if (isPomArtifact) {
                deployer.deploy(pomFile, artifact, repo, localRepository);
            } else {
                File file = artifact.getFile();

                if (file != null && !file.isDirectory()) {
                    deployer.deploy(file, artifact, repo, localRepository);
                } else if (!attachedArtifacts.isEmpty()) {
                    getLog()
                            .info(
                                    "No primary artifact to deploy, deploy attached artifacts instead.");
                } else {
                    String message = "The packaging for this project did not assign a file to the build artifact";
                    throw new MojoExecutionException(message);
                }
            }

            for (Iterator i = attachedArtifacts.iterator(); i.hasNext();) {
                Artifact attached = (Artifact) i.next();

                deployer.deploy(attached.getFile(), attached, repo,
                        localRepository);
            }
        } catch (ArtifactDeploymentException e) {
            throw new MojoExecutionException(e.getMessage(), e);
        }
    }
}

mojo ссылается на тип MirrorRepository для инкапсуляции repositoryId и url, это простой bean-компонент:

package name.seller.rich;

public class MirrorRepository {
    private String repositoryId;
    private String url;

    public String getRepositoryId() {
        return repositoryId;
    }

    public void setRepositoryId(String repositoryId) {
        this.repositoryId = repositoryId;
    }

    public String getUrl() {
        return url;
    }

    public void setUrl(String url) {
        this.url = url;
    }
}

Вот пример конфигурации с использованием плагина. Обратите внимание, что поддерживаются все форматы развертывания (http, scp, ftp):

<plugin>
  <groupId>name.seller.rich</groupId>
  <artifactId>maven-mirror-plugin</artifactId>
  <executions>
    <execution>
      <id>mirror</id>
      <phase>deploy</phase>
      <goals>
        <goal>mirror</goal>
      </goals>
    </execution>
  </executions>
  <configuration>
    <mirrorRepositories>
      <mirrorRepository>
        <repositoryId>mirror</repositoryId>
        <url>http://path/to/mirror</url>
      </mirrorRepository>
    </mirrorRepositories>
    <!--any other deploy configuration needed-->
  </configuration>
</plugin>
0 голосов
/ 26 марта 2009

Я думаю, что в Artifactory по умолчанию он поддерживает разные логические репозитории для загрузки снимков и снимков без снимков. Используя разрешения, вы можете сделать репозиторий моментальных снимков видимым только для некоторых.

Если этого недостаточно, другое решение, которое работает с Artifactory 2.0, заключается в том, чтобы Artifactory использовал базу данных MySQL, которая выполняет асинхронную репликацию в другую базу данных MySQL, которая, в свою очередь, читается отдельной установкой Artifactory. Если это слишком в режиме реального времени, вы можете просто иметь две разные установки, которые выполняют обновления на основе бизнес-правил.

0 голосов
/ 18 марта 2009

Артефакт имеет функцию автоматического экспорта. С документация :

Вы можете автоматически и периодически создавать резервные копии всей системы Artifactory. Процесс резервного копирования создает каталог с меткой времени (или zip-файл) в целевом каталоге резервного копирования и в основном идентичен выполнению полного экспорта системы с метаданными. [...] Каждая резервная копия может иметь свое собственное расписание и исключать определенные репозитории [...]

Содержимое резервной копии (при извлечении) находится в стандартном формате Maven и может быть загружено в любой внешний репозиторий Maven. [...]

Artifactory поддерживает поэтапное резервное копирование в тот же целевой каталог (с именем «текущий») в целевом каталоге резервного копирования. Этот вид резервного копирования только записывает дельты в выходной каталог, что приводит к чрезвычайно быстрому резервному копированию.

Разве это не то, что вам нужно? Чтобы передать файлы, вы можете либо подключить общий каталог к ​​удаленному серверу и выполнить там резервное копирование, либо выполнить резервное копирование локально, а затем выполнить rsync.

...