Maven: добавить зависимость к банке по относительному пути

232

У меня есть запатентованная фляга, которую я хочу добавить к своему пом как зависимость.

Но я не хочу добавлять его в хранилище. Причина в том, что я хочу, чтобы мои обычные команды maven, такие как mvn compileи т. Д., Работали из коробки. (Не требуя от разработчиков добавления его в какой-либо репозиторий самостоятельно).

Я хочу, чтобы файл jar находился в третьей части библиотеки управления исходными кодами, и связывался с ней по относительному пути из файла pom.xml.

Можно ли это сделать? Как?

ЭЛЕКТРОДИСТАНЦИОННАЯ СИСТЕМА УПРАВЛЕНИЯ
источник

Ответы:

343

Я хочу, чтобы файл jar находился в третьей части библиотеки управления исходными кодами, и связывался с ней по относительному пути из файла pom.xml.

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

Поэтому вместо этого объявите локальный репозиторий проекта:

<repositories>
  <repository>
    <id>my-local-repo</id>
    <url>file://${project.basedir}/my-repo</url>
  </repository>
</repositories>

Установите свою библиотеку третьей стороны в нём , используя install:install-fileс localRepositoryPathпараметром:

mvn install:install-file -Dfile=<path-to-file> -DgroupId=<myGroup> \ 
                         -DartifactId=<myArtifactId> -Dversion=<myVersion> \
                         -Dpackaging=<myPackaging> -DlocalRepositoryPath=<path>

Обновление: похоже, что install:install-fileигнорирует localRepositoryPathпри использовании версии 2.2 плагина. Тем не менее, он работает с версией 2.3 и более поздней версии плагина. Поэтому используйте полное имя плагина для указания версии:

mvn org.apache.maven.plugins:maven-install-plugin:2.3.1:install-file \
                         -Dfile=<path-to-file> -DgroupId=<myGroup> \ 
                         -DartifactId=<myArtifactId> -Dversion=<myVersion> \
                         -Dpackaging=<myPackaging> -DlocalRepositoryPath=<path>

maven-install-plugin документация

Наконец, объявите это как любую другую зависимость (но без systemобласти):

<dependency>
  <groupId>your.group.id</groupId>
  <artifactId>3rdparty</artifactId>
  <version>X.Y.Z</version>
</dependency>

Это ИМХО лучшее решение, чем использование systemобласти видимости, поскольку ваша зависимость будет рассматриваться как хороший гражданин (например, она будет включена в сборку и т. Д.).

Теперь я должен упомянуть, что «правильный способ» справиться с этой ситуацией в корпоративной среде (может быть, и не здесь) - это использовать корпоративный репозиторий.

Паскаль Тивент
источник
2
Это отличная идея, но в Maven 2.2.1 плагин установки, кажется, игнорирует localRepositoryPath...
Джейк
1
Зачем объявлять локальный репо? Почему бы просто не отпустить это в ~ / .m2 / с остальными.
Лейф Грюнвольдт
6
@ leif81 Потому что тогда репозиторий и библиотеки проверяются в репозитории SCM -> Каждый, кто выполняет проверку исходного кода, имеет все необходимое для создания копии библиотеки / приложения.
Дарт Андроид
6
У меня была та же проблема, что и у @lemon, которую я исправил, выполнив вместо этого basedir/./my-local-repoодин ..
Брайан
2
Упаковка должна быть в банке, поэтому - Упаковка = баночка
Данила Пятов
127

Используя systemобъем. ${basedir}это каталог вашего пом.

<dependency>
    <artifactId>..</artifactId>
    <groupId>..</groupId>
    <scope>system</scope>
    <systemPath>${basedir}/lib/dependency.jar</systemPath>
</dependency>

Тем не менее, желательно, чтобы вы установили свой jar в репозиторий, а не передавали его в SCM - в конце концов это то, что maven пытается устранить.

Bozho
источник
15
Прицельной системы следует избегать везде, где это возможно. Установить JAR в хранилище - лучшее решение ...
Гэндальф StormCrow
14
да, если возможно он прямо сказал, что не хочет помещать это в хранилище. Я добавил комментарий, чтобы указать, что это не очень хорошая практика. Но это работает.
Божо
groovy, ваше решение является наиболее приемлемым на данный момент, я думаю .. Я совершенно неправильно понял вопрос
муравей
Да - сам вопрос исключает лучший ответ. Помещение всего в ваш единственный сервер управления исходным кодом имеет мало общего с «сборкой из коробки»; скорее, все просто должно быть «под контролем». Проверьте pom's & settings.xml (указывая на внутреннее хранилище) и используйте два сервера для своего проекта: (1) контроль версий, (2) сгенерированный контроль артефактов. Проверка в jar-файлах имеет столько же смысла, сколько и в dll-проверке (моя старая корпорация действительно проверяла jars & lib.a / .so / .dll. Наш сервер p4 впоследствии работал очень медленно, некоторые тайно использовали hg для дня работа. Проблема решена?
Майкл
любой способ указать каталог, который содержит вместо этого jar-файлы, поэтому нам не нужно добавлять каждый, как gradle может это сделать?
Дин Хиллер
29

Это еще один метод в дополнение к моему предыдущему ответу в разделе Могу ли я добавить jar-файлы в maven 2 buildpath classpath без их установки?

Это позволит преодолеть ограничение при использовании многомодульных сборок, особенно если на загруженный JAR ссылаются в дочерних проектах вне родительского. Это также уменьшает объем работ по настройке, создавая файлы POM и SHA1 как часть сборки. Это также позволяет файлу находиться в любом месте проекта, не фиксируя имен и не следуя структуре репозитория maven.

Это использует maven-install-plugin. Чтобы это работало, вам нужно настроить многомодульный проект и иметь новый проект, представляющий сборку, чтобы установить файлы в локальный репозиторий и убедиться, что он первый.

Ваш многомодульный проект pom.xml будет выглядеть так:

<packaging>pom</packaging>
<modules>
<!-- The repository module must be first in order to ensure
     that the local repository is populated -->
    <module>repository</module>
    <module>... other modules ...</module>
</modules>

Файл repository / pom.xml будет содержать определения для загрузки JAR-файлов, которые являются частью вашего проекта. Ниже приведены некоторые фрагменты файла pom.xml.

<artifactId>repository</artifactId>
<packaging>pom</packaging>

Упаковка pom не позволяет выполнять какие-либо тесты, компилировать или генерировать jar-файл. Мясо pom.xml находится в разделе сборки, где используется maven-install-plugin.

<build>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-install-plugin</artifactId>
            <executions>
                <execution>
                        <id>com.ibm.db2:db2jcc</id>
                        <phase>verify</phase>
                        <goals>
                            <goal>install-file</goal>
                        </goals>
                        <configuration>
                            <groupId>com.ibm.db2</groupId>
                            <artifactId>db2jcc</artifactId>
                            <version>9.0.0</version>
                            <packaging>jar</packaging>
                            <file>${basedir}/src/jars/db2jcc.jar</file>
                            <createChecksum>true</createChecksum>
                            <generatePom>true</generatePom>
                        </configuration>
                </execution>
                <execution>...</execution>
            </executions>
        </plugin>
    </plugins>
</build>

Чтобы установить более одного файла, просто добавьте больше исполнений.

Архимед Траяно
источник
Это единственное, что сработало для моего многомодульного проекта. Локальный подход <repository> по неизвестной причине не сработал. Так что спасибо!
Лонзак
10

Это работает для меня: допустим, у меня есть эта зависимость

<dependency>
    <groupId>com.company.app</groupId>
    <artifactId>my-library</artifactId>
    <version>1.0</version>
    <scope>system</scope>
    <systemPath>${project.basedir}/lib/my-library.jar</systemPath>
</dependency>

Затем добавьте путь к классу для вашей системной зависимости вручную, как это

<Class-Path>libs/my-library-1.0.jar</Class-Path>

Полная конфигурация:

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-jar-plugin</artifactId>
    <version>2.4</version>
    <configuration>
        <archive>
            <manifestEntries>
                <Build-Jdk>${jdk.version}</Build-Jdk>
                <Implementation-Title>${project.name}</Implementation-Title>
                <Implementation-Version>${project.version}</Implementation-Version>
                <Specification-Title>${project.name} Library</Specification-Title>
                <Specification-Version>${project.version}</Specification-Version>
                <Class-Path>libs/my-library-1.0.jar</Class-Path>
            </manifestEntries>
            <manifest>
                <addClasspath>true</addClasspath>
                <mainClass>com.company.app.MainClass</mainClass>
                <classpathPrefix>libs/</classpathPrefix>
            </manifest>
        </archive>
    </configuration>
</plugin>
<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-dependency-plugin</artifactId>
    <version>2.5.1</version>
    <executions>
        <execution>
            <id>copy-dependencies</id>
            <phase>package</phase>
            <goals>
                <goal>copy-dependencies</goal>
            </goals>
            <configuration>
                <outputDirectory>${project.build.directory}/libs/</outputDirectory>
            </configuration>
        </execution>
    </executions>
</plugin>
RufusSC2
источник
9

Я ранее писал о шаблоне для этого.

Это очень похоже на решение, предложенное Паскалем, хотя оно перемещает все такие зависимости в выделенный модуль репозитория, чтобы вам не приходилось повторять его везде, где используется зависимость, если это многомодульная сборка.

Бретт Портер
источник
6

По сути, добавьте это в pom.xml:

...

<repositories>
   <repository>
       <id>lib_id</id>
       <url>file://${project.basedir}/lib</url>
   </repository>
</repositories>

...

<dependencies>
  ...
  <dependency>
      <groupId>com.mylibrary</groupId>
      <artifactId>mylibraryname</artifactId>
      <version>1.0.0</version>
  </dependency>
  ...
</dependencies>
Фульхенсио Хара
источник
4

мы перешли на gradle, и в gradle это работает намного лучше;). мы просто указываем папку, в которую мы можем бросить банки для таких временных ситуаций. У нас все еще есть большинство наших jar-файлов, определенных в разделе управления зависимостями (т.е. так же, как в maven). Это еще одна зависимость, которую мы определяем.

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

Дин Хиллер
источник
1
Не могли бы вы привести пример, как вы это сделали?
Томас
2

Небольшое дополнение к решению, опубликованному Паскалем

Когда я пошел по этому маршруту, я получил ошибку в maven при установке ojdbc jar.

[INFO] --- maven-install-plugin:2.5.1:install-file (default-cli) @ validator ---
[INFO] pom.xml not found in ojdbc14.jar

После добавления -DpomFile проблема была решена.

$ mvn install:install-file -Dfile=./lib/ojdbc14.jar -DgroupId=ojdbc \
   -DartifactId=ojdbc -Dversion=14 -Dpackaging=jar -DlocalRepositoryPath=./repo \
   -DpomFile=~/.m2/repository/ojdbc/ojdbc/14/ojdbc-14.pom
veeseekay
источник
0

Вы можете использовать eclipse для создания исполняемого файла Jar: Export / Runable Jar

user1942990
источник
Не уверен, что это ответ на вопрос. У него уже есть файл в виде банки.
Йоханнес Яндер
Eclipse поддерживает Uberjar или Shaded Jar, так что это решение, но не для Maven
Алекс Леманн