You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

AbstractMavenRepositorySearch.java 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  1. package org.apache.archiva.indexer.maven.search;
  2. /*
  3. * Licensed to the Apache Software Foundation (ASF) under one
  4. * or more contributor license agreements. See the NOTICE file
  5. * distributed with this work for additional information
  6. * regarding copyright ownership. The ASF licenses this file
  7. * to you under the Apache License, Version 2.0 (the
  8. * "License"); you may not use this file except in compliance
  9. * with the License. You may obtain a copy of the License at
  10. *
  11. * http://www.apache.org/licenses/LICENSE-2.0
  12. *
  13. * Unless required by applicable law or agreed to in writing,
  14. * software distributed under the License is distributed on an
  15. * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
  16. * KIND, either express or implied. See the License for the
  17. * specific language governing permissions and limitations
  18. * under the License.
  19. */
  20. import junit.framework.TestCase;
  21. import org.apache.archiva.common.utils.FileUtils;
  22. import org.apache.archiva.configuration.ArchivaConfiguration;
  23. import org.apache.archiva.configuration.Configuration;
  24. import org.apache.archiva.configuration.ConfigurationListener;
  25. import org.apache.archiva.configuration.ManagedRepositoryConfiguration;
  26. import org.apache.archiva.indexer.ArchivaIndexingContext;
  27. import org.apache.archiva.indexer.search.SearchResultHit;
  28. import org.apache.archiva.indexer.search.SearchResults;
  29. import org.apache.archiva.proxy.ProxyRegistry;
  30. import org.apache.archiva.repository.Repository;
  31. import org.apache.archiva.repository.base.ArchivaRepositoryRegistry;
  32. import org.apache.archiva.repository.base.RepositoryHandlerDependencies;
  33. import org.apache.archiva.repository.features.IndexCreationFeature;
  34. import org.apache.archiva.test.utils.ArchivaSpringJUnit4ClassRunner;
  35. import org.apache.commons.lang3.SystemUtils;
  36. import org.apache.maven.index.ArtifactContext;
  37. import org.apache.maven.index.ArtifactContextProducer;
  38. import org.apache.maven.index.ArtifactScanningListener;
  39. import org.apache.maven.index.DefaultScannerListener;
  40. import org.apache.maven.index.Indexer;
  41. import org.apache.maven.index.IndexerEngine;
  42. import org.apache.maven.index.QueryCreator;
  43. import org.apache.maven.index.Scanner;
  44. import org.apache.maven.index.ScanningRequest;
  45. import org.apache.maven.index.ScanningResult;
  46. import org.apache.maven.index.context.IndexingContext;
  47. import org.apache.maven.index_shaded.lucene.index.IndexUpgrader;
  48. import org.easymock.EasyMock;
  49. import org.easymock.IMocksControl;
  50. import org.junit.After;
  51. import org.junit.Before;
  52. import org.junit.runner.RunWith;
  53. import org.slf4j.Logger;
  54. import org.slf4j.LoggerFactory;
  55. import org.springframework.test.context.ContextConfiguration;
  56. import javax.inject.Inject;
  57. import java.io.IOException;
  58. import java.nio.file.Files;
  59. import java.nio.file.Path;
  60. import java.nio.file.Paths;
  61. import java.util.List;
  62. import java.util.Locale;
  63. import static org.apache.archiva.indexer.ArchivaIndexManager.DEFAULT_INDEX_PATH;
  64. /**
  65. * @author Olivier Lamy
  66. */
  67. @RunWith( ArchivaSpringJUnit4ClassRunner.class )
  68. @ContextConfiguration( locations = { "classpath*:/META-INF/spring-context.xml", "classpath:/spring-context.xml" } )
  69. public abstract class AbstractMavenRepositorySearch
  70. extends TestCase
  71. {
  72. protected Logger log = LoggerFactory.getLogger( getClass() );
  73. public static String TEST_REPO_1 = "maven-search-test-repo";
  74. public static String TEST_REPO_2 = "maven-search-test-repo-2";
  75. public static String REPO_RELEASE = "repo-release";
  76. MavenRepositorySearch search;
  77. ArchivaConfiguration archivaConfig;
  78. @Inject
  79. ArtifactContextProducer artifactContextProducer;
  80. @Inject
  81. ArchivaRepositoryRegistry repositoryRegistry;
  82. @SuppressWarnings( "unused" )
  83. @Inject
  84. RepositoryHandlerDependencies repositoryHandlerDependencies;
  85. @Inject
  86. ProxyRegistry proxyRegistry;
  87. @Inject
  88. private IndexerEngine indexerEngine;
  89. IMocksControl archivaConfigControl;
  90. Configuration config;
  91. @Inject
  92. Indexer indexer;
  93. @Inject
  94. Scanner scanner;
  95. @Inject
  96. QueryCreator queryCreator;
  97. @Before
  98. @Override
  99. public void setUp()
  100. throws Exception
  101. {
  102. super.setUp();
  103. FileUtils.deleteDirectory( Paths.get( org.apache.archiva.common.utils.FileUtils.getBasedir(), "/target/repos/" + TEST_REPO_1 + "/.indexer" ) );
  104. assertFalse( Files.exists(Paths.get( org.apache.archiva.common.utils.FileUtils.getBasedir(), "/target/repos/" + TEST_REPO_1 + "/.indexer" )) );
  105. Files.createDirectories( Paths.get( org.apache.archiva.common.utils.FileUtils.getBasedir(), "/target/repos/" + TEST_REPO_1 + "/.indexer" ) );
  106. FileUtils.deleteDirectory( Paths.get( org.apache.archiva.common.utils.FileUtils.getBasedir(), "/target/repos/" + TEST_REPO_2 + "/.indexer" ) );
  107. assertFalse( Files.exists(Paths.get( org.apache.archiva.common.utils.FileUtils.getBasedir(), "/target/repos/" + TEST_REPO_2 + "/.indexer" )) );
  108. Files.createDirectories( Paths.get( org.apache.archiva.common.utils.FileUtils.getBasedir(), "/target/repos/" + TEST_REPO_2 + "/.indexer" ) );
  109. archivaConfigControl = EasyMock.createControl();
  110. archivaConfig = archivaConfigControl.createMock( ArchivaConfiguration.class );
  111. repositoryRegistry.setArchivaConfiguration( archivaConfig );
  112. search = new MavenRepositorySearch( indexer, repositoryRegistry, proxyRegistry,
  113. queryCreator );
  114. assertNotNull( repositoryRegistry );
  115. config = new Configuration();
  116. config.addManagedRepository( createRepositoryConfig( TEST_REPO_1 ) );
  117. config.addManagedRepository( createRepositoryConfig( TEST_REPO_2 ) );
  118. config.addManagedRepository( createRepositoryConfig( REPO_RELEASE ) );
  119. archivaConfig.addListener( EasyMock.anyObject( ConfigurationListener.class ) );
  120. EasyMock.expect( archivaConfig.getDefaultLocale() ).andReturn( Locale.getDefault( ) ).anyTimes();
  121. EasyMock.expect( archivaConfig.getConfiguration() ).andReturn(config).anyTimes();
  122. archivaConfig.save(EasyMock.anyObject(Configuration.class), EasyMock.anyString());
  123. EasyMock.expectLastCall().anyTimes();
  124. archivaConfigControl.replay();
  125. repositoryRegistry.reload();
  126. }
  127. @After
  128. @Override
  129. public void tearDown()
  130. throws Exception
  131. {
  132. archivaConfigControl.reset();
  133. EasyMock.expect( archivaConfig.getDefaultLocale() ).andReturn( Locale.getDefault( ) ).anyTimes();
  134. EasyMock.expect( archivaConfig.getConfiguration() ).andReturn(config).anyTimes();
  135. archivaConfig.save(EasyMock.anyObject(Configuration.class), EasyMock.anyString());
  136. EasyMock.expectLastCall().anyTimes();
  137. archivaConfigControl.replay();
  138. repositoryRegistry.removeRepository(TEST_REPO_1);
  139. repositoryRegistry.removeRepository(TEST_REPO_2);
  140. repositoryRegistry.removeRepository(REPO_RELEASE);
  141. repositoryRegistry.destroy();
  142. FileUtils.deleteDirectory( Paths.get( org.apache.archiva.common.utils.FileUtils.getBasedir(), "/target/repos/" + TEST_REPO_1 ) );
  143. assertFalse( Files.exists(Paths.get( org.apache.archiva.common.utils.FileUtils.getBasedir(), "/target/repos/" + TEST_REPO_1 )) );
  144. FileUtils.deleteDirectory( Paths.get( org.apache.archiva.common.utils.FileUtils.getBasedir(), "/target/repos/" + TEST_REPO_2 ) );
  145. assertFalse( Files.exists(Paths.get( org.apache.archiva.common.utils.FileUtils.getBasedir(), "/target/repos/" + TEST_REPO_2 )) );
  146. super.tearDown();
  147. }
  148. protected ManagedRepositoryConfiguration createRepositoryConfig( String repository )
  149. {
  150. ManagedRepositoryConfiguration repositoryConfig = new ManagedRepositoryConfiguration();
  151. repositoryConfig.setId( repository );
  152. repositoryConfig.setLocation( org.apache.archiva.common.utils.FileUtils.getBasedir() + "/target/repos/" + repository );
  153. Path f = Paths.get( repositoryConfig.getLocation() );
  154. if ( !Files.exists(f) )
  155. {
  156. try
  157. {
  158. Files.createDirectories( f );
  159. }
  160. catch ( IOException e )
  161. {
  162. log.error("Could not create directories for {}", f);
  163. }
  164. }
  165. repositoryConfig.setLayout( "default" );
  166. repositoryConfig.setName( repository );
  167. repositoryConfig.setScanned( true );
  168. repositoryConfig.setSnapshots( false );
  169. repositoryConfig.setReleases( true );
  170. repositoryConfig.setIndexDir(DEFAULT_INDEX_PATH);
  171. return repositoryConfig;
  172. }
  173. protected void createIndex( String repository, List<Path> filesToBeIndexed, boolean scan) throws Exception {
  174. createIndex(repository, filesToBeIndexed, scan, null, true);
  175. }
  176. protected void createIndex( String repository, List<Path> filesToBeIndexed, boolean scan, Path indexDir, boolean copyFiles)
  177. throws Exception
  178. {
  179. final Repository rRepo = repositoryRegistry.getRepository(repository);
  180. IndexCreationFeature icf = rRepo.getFeature(IndexCreationFeature.class).get();
  181. ArchivaIndexingContext archivaCtx = rRepo.getIndexingContext();
  182. IndexingContext context = archivaCtx.getBaseContext(IndexingContext.class);
  183. if ( archivaCtx != null )
  184. {
  185. archivaCtx.close(true);
  186. }
  187. Path repoDir = Paths.get(org.apache.archiva.common.utils.FileUtils.getBasedir()).resolve("target").resolve("repos").resolve(repository);
  188. Path indexerDirectory = repoDir.resolve(".indexer" );
  189. if ( indexDir == null && Files.exists(indexerDirectory) )
  190. {
  191. FileUtils.deleteDirectory( indexerDirectory );
  192. assertFalse( Files.exists(indexerDirectory) );
  193. }
  194. Path lockFile = repoDir.resolve(".indexer/write.lock" );
  195. if ( Files.exists(lockFile) )
  196. {
  197. Files.delete(lockFile);
  198. }
  199. assertFalse( Files.exists(lockFile) );
  200. if (indexDir==null) {
  201. Path indexDirectory =
  202. Paths.get(org.apache.archiva.common.utils.FileUtils.getBasedir(), "target/index/test-" + Long.toString(System.currentTimeMillis()));
  203. indexDirectory.toFile().deleteOnExit();
  204. FileUtils.deleteDirectory(indexDirectory);
  205. icf.setIndexPath(indexDirectory.toUri());
  206. config.getManagedRepositories( ).stream( ).filter( r -> r.getId( ).equals( rRepo.getId( ) ) ).findFirst( ).ifPresent( r ->
  207. r.setIndexDir( indexDirectory.toAbsolutePath( ).toString( ) )
  208. );
  209. // IndexUpgrader.main(new String[]{indexDirectory.toAbsolutePath().toString()});
  210. } else {
  211. icf.setIndexPath(indexDir.toUri());
  212. Files.createDirectories( indexDir );
  213. config.getManagedRepositories( ).stream( ).filter( r -> r.getId( ).equals( rRepo.getId( ) ) ).findFirst( ).ifPresent( r ->
  214. r.setIndexDir( indexDir.toAbsolutePath( ).toString( ) )
  215. );
  216. IndexUpgrader.main(new String[]{indexDir.toAbsolutePath().toString()});
  217. }
  218. if (copyFiles) {
  219. Path repo = Paths.get(org.apache.archiva.common.utils.FileUtils.getBasedir(), "src/test/" + repository);
  220. assertTrue(Files.exists(repo));
  221. org.apache.commons.io.FileUtils.copyDirectory(repo.toFile(), repoDir.toFile());
  222. }
  223. archivaConfigControl.reset();
  224. archivaConfig.addListener( EasyMock.anyObject( ConfigurationListener.class ) );
  225. EasyMock.expect( archivaConfig.getConfiguration() ).andReturn(config).anyTimes();
  226. archivaConfig.save(EasyMock.anyObject(Configuration.class), EasyMock.anyString());
  227. EasyMock.expectLastCall().anyTimes();
  228. archivaConfigControl.replay();
  229. repositoryRegistry.reload();
  230. archivaConfigControl.reset();
  231. Repository rRepo2 = repositoryRegistry.getRepository( repository );
  232. icf = rRepo2.getFeature(IndexCreationFeature.class).get();
  233. archivaCtx = rRepo2.getIndexingContext();
  234. context = archivaCtx.getBaseContext(IndexingContext.class);
  235. // minimize datas in memory
  236. // context.getIndexWriter().setMaxBufferedDocs( -1 );
  237. // context.getIndexWriter().setRAMBufferSizeMB( 1 );
  238. for ( Path artifactFile : filesToBeIndexed )
  239. {
  240. assertTrue( "file not exists " + artifactFile, Files.exists(artifactFile) );
  241. ArtifactContext ac = artifactContextProducer.getArtifactContext( context, artifactFile.toFile() );
  242. if ( artifactFile.toString().endsWith( ".pom" ) )
  243. {
  244. ac.getArtifactInfo().setFileExtension( "pom" );
  245. ac.getArtifactInfo().setPackaging( "pom" );
  246. ac.getArtifactInfo().setClassifier( "pom" );
  247. }
  248. indexer.addArtifactToIndex( ac, context );
  249. context.updateTimestamp( true );
  250. }
  251. if ( scan )
  252. {
  253. DefaultScannerListener listener = new DefaultScannerListener( context, indexerEngine, true, new ArtifactScanListener());
  254. ScanningRequest req = new ScanningRequest(context, listener );
  255. scanner.scan( req );
  256. context.commit();
  257. }
  258. // force flushing
  259. context.commit();
  260. // context.getIndexWriter().commit();
  261. context.setSearchable( true );
  262. }
  263. static class ArtifactScanListener
  264. implements ArtifactScanningListener
  265. {
  266. protected Logger log = LoggerFactory.getLogger( getClass() );
  267. @Override
  268. public void scanningStarted( IndexingContext ctx )
  269. {
  270. //
  271. }
  272. @Override
  273. public void scanningFinished( IndexingContext ctx, ScanningResult result )
  274. {
  275. // no op
  276. }
  277. @Override
  278. public void artifactError( ArtifactContext ac, Exception e )
  279. {
  280. log.debug( "artifactError {}", ac.getArtifact().getPath(), e );
  281. }
  282. @Override
  283. public void artifactDiscovered( ArtifactContext ac )
  284. {
  285. log.debug( "artifactDiscovered {}:{}", //
  286. ac.getArtifact() == null ? "" : ac.getArtifact().getPath(), //
  287. ac.getArtifact() == null ? "" : ac.getArtifactInfo() );
  288. }
  289. }
  290. public String niceDisplay( SearchResults searchResults )
  291. throws Exception
  292. {
  293. StringBuilder sb = new StringBuilder();
  294. for ( SearchResultHit hit : searchResults.getHits() )
  295. {
  296. sb.append( hit.toString() ).append( SystemUtils.LINE_SEPARATOR );
  297. }
  298. return sb.toString();
  299. }
  300. }