001 /** 002 * Copyright 2008-2013 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 package org.codehaus.mojo.wagon.shared; 017 018 /* 019 * Licensed to the Apache Software Foundation (ASF) under one or more contributor license 020 * agreements. See the NOTICE file distributed with this work for additional information regarding 021 * copyright ownership. The ASF licenses this file to you under the Apache License, Version 2.0 (the 022 * "License"); you may not use this file except in compliance with the License. You may obtain a 023 * copy of the License at 024 * 025 * http://www.apache.org/licenses/LICENSE-2.0 026 * 027 * Unless required by applicable law or agreed to in writing, software distributed under the License 028 * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express 029 * or implied. See the License for the specific language governing permissions and limitations under 030 * the License. 031 */ 032 033 import java.io.File; 034 import java.io.FileInputStream; 035 import java.io.FileReader; 036 import java.io.FileWriter; 037 import java.io.IOException; 038 import java.io.InputStream; 039 import java.io.Reader; 040 import java.io.Writer; 041 import java.security.MessageDigest; 042 import java.security.NoSuchAlgorithmException; 043 044 import org.apache.maven.artifact.repository.metadata.Metadata; 045 import org.apache.maven.artifact.repository.metadata.io.xpp3.MetadataXpp3Reader; 046 import org.apache.maven.artifact.repository.metadata.io.xpp3.MetadataXpp3Writer; 047 import org.apache.maven.plugin.logging.Log; 048 import org.apache.maven.shared.model.fileset.FileSet; 049 import org.apache.maven.wagon.ResourceDoesNotExistException; 050 import org.apache.maven.wagon.Wagon; 051 import org.apache.maven.wagon.WagonException; 052 import org.codehaus.plexus.util.DirectoryScanner; 053 import org.codehaus.plexus.util.FileUtils; 054 import org.codehaus.plexus.util.IOUtil; 055 import org.codehaus.plexus.util.xml.pull.XmlPullParserException; 056 057 /** 058 * A copy of stage's plugin RepositoryCopier but use WagonUpload and WagonDownload instead 059 * 060 * @plexus.component role="org.codehaus.mojo.wagon.shared.MavenRepoMerger" role-hint="default" 061 */ 062 063 public class DefaultMavenRepoMerger implements MavenRepoMerger { 064 /** 065 * @plexus.requirement role="org.codehaus.mojo.wagon.shared.WagonDownload" 066 */ 067 private WagonDownload downloader; 068 069 /** 070 * @plexus.requirement role="org.codehaus.mojo.wagon.shared.WagonUpload" 071 */ 072 private WagonUpload uploader; 073 074 @Override 075 public void merge(Wagon src, Wagon target, boolean optimize, Log logger) throws WagonException, IOException { 076 077 // copy src to a local dir 078 File downloadSrcDir = createTempDirectory("wagon-maven-plugin"); 079 080 WagonFileSet srcFileSet = new WagonFileSet(); 081 srcFileSet.setDownloadDirectory(downloadSrcDir); 082 // ignore archiva/nexus .index at root dir 083 String[] excludes = { ".index/**", ".indexer/**, .meta/**, .nexus/**" }; 084 srcFileSet.setExcludes(excludes); 085 086 try { 087 downloader.download(src, srcFileSet, logger, false); 088 089 // merge metadata 090 DirectoryScanner scanner = new DirectoryScanner(); 091 scanner.setBasedir(downloadSrcDir); 092 String[] includes = { "**/" + MAVEN_METADATA }; 093 scanner.setIncludes(includes); 094 scanner.scan(); 095 String[] files = scanner.getIncludedFiles(); 096 097 for (int i = 0; i < files.length; ++i) { 098 File srcMetadaFile = new File(downloadSrcDir, files[i] + IN_PROCESS_MARKER); 099 100 try { 101 target.get(files[i].replace('\\', '/'), srcMetadaFile); 102 } catch (ResourceDoesNotExistException e) { 103 // We don't have an equivalent on the targetRepositoryUrl side because we have something 104 // new on the sourceRepositoryUrl side so just skip the metadata merging. 105 continue; 106 } 107 108 try { 109 mergeMetadata(srcMetadaFile, logger); 110 } catch (XmlPullParserException e) { 111 throw new IOException("Metadata file is corrupt " + files[i] + " Reason: " + e.getMessage()); 112 } 113 114 } 115 116 // upload to target 117 FileSet tobeUploadedFileSet = new FileSet(); 118 tobeUploadedFileSet.setDirectory(downloadSrcDir.getAbsolutePath()); 119 120 this.uploader.upload(target, tobeUploadedFileSet, optimize, logger); 121 122 } finally { 123 FileUtils.deleteDirectory(downloadSrcDir); 124 } 125 126 } 127 128 private void mergeMetadata(File existingMetadata, Log logger) throws IOException, XmlPullParserException { 129 130 Writer stagedMetadataWriter = null; 131 Reader existingMetadataReader = null; 132 Reader stagedMetadataReader = null; 133 File stagedMetadataFile = null; 134 135 try { 136 MetadataXpp3Reader xppReader = new MetadataXpp3Reader(); 137 MetadataXpp3Writer xppWriter = new MetadataXpp3Writer(); 138 139 // Existing Metadata in target stage 140 existingMetadataReader = new FileReader(existingMetadata); 141 Metadata existing = xppReader.read(existingMetadataReader); 142 143 // Staged Metadata 144 stagedMetadataFile = new File(existingMetadata.getParentFile(), MAVEN_METADATA); 145 stagedMetadataReader = new FileReader(stagedMetadataFile); 146 Metadata staged = xppReader.read(stagedMetadataReader); 147 148 // Merge and write back to staged metadata to replace the remote one 149 existing.merge(staged); 150 151 stagedMetadataWriter = new FileWriter(stagedMetadataFile); 152 xppWriter.write(stagedMetadataWriter, existing); 153 154 logger.info("Merging metadata file: " + stagedMetadataFile); 155 156 } finally { 157 IOUtil.close(stagedMetadataWriter); 158 IOUtil.close(stagedMetadataReader); 159 IOUtil.close(existingMetadataReader); 160 161 existingMetadata.delete(); 162 } 163 164 // Mark all metadata as in-process and regenerate the checksums as they will be different 165 // after the merger 166 167 try { 168 File newMd5 = new File(stagedMetadataFile.getParentFile(), MAVEN_METADATA + ".md5"); 169 FileUtils.fileWrite(newMd5.getAbsolutePath(), checksum(stagedMetadataFile, MD5)); 170 171 File newSha1 = new File(stagedMetadataFile.getParentFile(), MAVEN_METADATA + ".sha1"); 172 FileUtils.fileWrite(newSha1.getAbsolutePath(), checksum(stagedMetadataFile, SHA1)); 173 } catch (NoSuchAlgorithmException e) { 174 throw new RuntimeException(e); 175 } 176 177 // We have the new merged copy so we're good 178 179 } 180 181 private String checksum(File file, String type) throws IOException, NoSuchAlgorithmException { 182 MessageDigest md5 = MessageDigest.getInstance(type); 183 184 InputStream is = new FileInputStream(file); 185 186 byte[] buf = new byte[8192]; 187 188 int i; 189 190 while ((i = is.read(buf)) > 0) { 191 md5.update(buf, 0, i); 192 } 193 194 IOUtil.close(is); 195 196 return encode(md5.digest()); 197 } 198 199 private String encode(byte[] binaryData) { 200 if (binaryData.length != 16 && binaryData.length != 20) { 201 int bitLength = binaryData.length * 8; 202 throw new IllegalArgumentException("Unrecognised length for binary data: " + bitLength + " bits"); 203 } 204 205 String retValue = ""; 206 207 for (int i = 0; i < binaryData.length; i++) { 208 String t = Integer.toHexString(binaryData[i] & 0xff); 209 210 if (t.length() == 1) { 211 retValue += ("0" + t); 212 } else { 213 retValue += t; 214 } 215 } 216 217 return retValue.trim(); 218 } 219 220 public static File createTempDirectory(String prefix) throws IOException { 221 final File temp = File.createTempFile(prefix, Long.toString(System.nanoTime())); 222 223 if (!temp.delete()) { 224 throw new IOException("Could not delete temp file: " + temp.getAbsolutePath()); 225 } 226 227 if (!temp.mkdir()) { 228 throw new IOException("Could not create temp directory: " + temp.getAbsolutePath()); 229 } 230 231 return temp; 232 } 233 234 }