mirror of
				https://github.com/apache/cloudstack.git
				synced 2025-10-26 08:42:29 +01:00 
			
		
		
		
	CLOUDSTACK-10231: Asserted fixes for Direct Download on KVM (#2408)
Several fixes addressed: - Dettach ISO fails when trying to detach a direct download ISO - Fix for metalink support on SSVM agents (this closes CLOUDSTACK-10238) - Reinstall VM from bypassed registered template (this closes CLOUDSTACK-10250) - Fix upload certificate error message even though operation was successful - Fix metalink download, checksum retry logic and metalink SSVM downloader
This commit is contained in:
		
							parent
							
								
									1ad04cbc9b
								
							
						
					
					
						commit
						6a75423779
					
				| @ -22,6 +22,7 @@ import com.cloud.utils.exception.CloudRuntimeException; | |||||||
| import com.cloud.utils.script.Script; | import com.cloud.utils.script.Script; | ||||||
| import org.apache.cloudstack.utils.security.DigestHelper; | import org.apache.cloudstack.utils.security.DigestHelper; | ||||||
| import org.apache.commons.lang.StringUtils; | import org.apache.commons.lang.StringUtils; | ||||||
|  | import org.apache.log4j.Logger; | ||||||
| 
 | 
 | ||||||
| import java.io.File; | import java.io.File; | ||||||
| import java.io.FileInputStream; | import java.io.FileInputStream; | ||||||
| @ -37,6 +38,8 @@ public abstract class DirectTemplateDownloaderImpl implements DirectTemplateDown | |||||||
|     private String downloadedFilePath; |     private String downloadedFilePath; | ||||||
|     private String installPath; |     private String installPath; | ||||||
|     private String checksum; |     private String checksum; | ||||||
|  |     private boolean redownload = false; | ||||||
|  |     public static final Logger s_logger = Logger.getLogger(DirectTemplateDownloaderImpl.class.getName()); | ||||||
| 
 | 
 | ||||||
|     protected DirectTemplateDownloaderImpl(final String url, final String destPoolPath, final Long templateId, final String checksum) { |     protected DirectTemplateDownloaderImpl(final String url, final String destPoolPath, final Long templateId, final String checksum) { | ||||||
|         this.url = url; |         this.url = url; | ||||||
| @ -70,6 +73,10 @@ public abstract class DirectTemplateDownloaderImpl implements DirectTemplateDown | |||||||
|         return url; |         return url; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public void setUrl(String url) { | ||||||
|  |         this.url = url; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     public String getDestPoolPath() { |     public String getDestPoolPath() { | ||||||
|         return destPoolPath; |         return destPoolPath; | ||||||
|     } |     } | ||||||
| @ -86,6 +93,18 @@ public abstract class DirectTemplateDownloaderImpl implements DirectTemplateDown | |||||||
|         this.downloadedFilePath = filePath; |         this.downloadedFilePath = filePath; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public String getChecksum() { | ||||||
|  |         return checksum; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public void setChecksum(String checksum) { | ||||||
|  |         this.checksum = checksum; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public boolean isRedownload() { | ||||||
|  |         return redownload; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Return filename from url |      * Return filename from url | ||||||
|      */ |      */ | ||||||
| @ -155,14 +174,47 @@ public abstract class DirectTemplateDownloaderImpl implements DirectTemplateDown | |||||||
|     @Override |     @Override | ||||||
|     public boolean validateChecksum() { |     public boolean validateChecksum() { | ||||||
|         if (StringUtils.isNotBlank(checksum)) { |         if (StringUtils.isNotBlank(checksum)) { | ||||||
|  |             int retry = 3; | ||||||
|  |             boolean valid = false; | ||||||
|             try { |             try { | ||||||
|                 return DigestHelper.check(checksum, new FileInputStream(downloadedFilePath)); |                 while (!valid && retry > 0) { | ||||||
|  |                     retry--; | ||||||
|  |                     s_logger.info("Performing checksum validation for downloaded template " + templateId + " using " + checksum + ", retries left: " + retry); | ||||||
|  |                     valid = DigestHelper.check(checksum, new FileInputStream(downloadedFilePath)); | ||||||
|  |                     if (!valid && retry > 0) { | ||||||
|  |                         s_logger.info("Checksum validation failded, re-downloading template"); | ||||||
|  |                         redownload = true; | ||||||
|  |                         resetDownloadFile(); | ||||||
|  |                         downloadTemplate(); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 s_logger.info("Checksum validation for template " + templateId + ": " + (valid ? "succeeded" : "failed")); | ||||||
|  |                 return valid; | ||||||
|             } catch (IOException e) { |             } catch (IOException e) { | ||||||
|                 throw new CloudRuntimeException("could not check sum for file: " + downloadedFilePath,e); |                 throw new CloudRuntimeException("could not check sum for file: " + downloadedFilePath, e); | ||||||
|             } catch (NoSuchAlgorithmException e) { |             } catch (NoSuchAlgorithmException e) { | ||||||
|                 throw new CloudRuntimeException("Unknown checksum algorithm: " + checksum, e); |                 throw new CloudRuntimeException("Unknown checksum algorithm: " + checksum, e); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |         s_logger.info("No checksum provided, skipping checksum validation"); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Delete and create download file | ||||||
|  |      */ | ||||||
|  |     private void resetDownloadFile() { | ||||||
|  |         File f = new File(getDownloadedFilePath()); | ||||||
|  |         s_logger.info("Resetting download file: " + getDownloadedFilePath() + ", in order to re-download and persist template " + templateId + " on it"); | ||||||
|  |         try { | ||||||
|  |             if (f.exists()) { | ||||||
|  |                 f.delete(); | ||||||
|  |             } | ||||||
|  |             f.createNewFile(); | ||||||
|  |         } catch (IOException e) { | ||||||
|  |             s_logger.error("Error creating file to download on: " + getDownloadedFilePath() + " due to: " + e.getMessage()); | ||||||
|  |             throw new CloudRuntimeException("Failed to create download file for direct download"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
| } | } | ||||||
|  | |||||||
| @ -22,12 +22,9 @@ package com.cloud.agent.direct.download; | |||||||
| import com.cloud.utils.exception.CloudRuntimeException; | import com.cloud.utils.exception.CloudRuntimeException; | ||||||
| import org.apache.commons.collections.MapUtils; | import org.apache.commons.collections.MapUtils; | ||||||
| import org.apache.commons.httpclient.HttpClient; | import org.apache.commons.httpclient.HttpClient; | ||||||
| import org.apache.commons.httpclient.HttpMethod; |  | ||||||
| import org.apache.commons.httpclient.HttpMethodRetryHandler; |  | ||||||
| import org.apache.commons.httpclient.MultiThreadedHttpConnectionManager; | import org.apache.commons.httpclient.MultiThreadedHttpConnectionManager; | ||||||
| import org.apache.commons.httpclient.NoHttpResponseException; | import org.apache.commons.httpclient.HttpStatus; | ||||||
| import org.apache.commons.httpclient.methods.GetMethod; | import org.apache.commons.httpclient.methods.GetMethod; | ||||||
| import org.apache.commons.httpclient.params.HttpMethodParams; |  | ||||||
| import org.apache.commons.io.IOUtils; | import org.apache.commons.io.IOUtils; | ||||||
| import org.apache.log4j.Logger; | import org.apache.log4j.Logger; | ||||||
| 
 | 
 | ||||||
| @ -36,21 +33,22 @@ import java.io.FileOutputStream; | |||||||
| import java.io.InputStream; | import java.io.InputStream; | ||||||
| import java.io.OutputStream; | import java.io.OutputStream; | ||||||
| import java.io.IOException; | import java.io.IOException; | ||||||
|  | import java.util.HashMap; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| 
 | 
 | ||||||
| public class HttpDirectTemplateDownloader extends DirectTemplateDownloaderImpl { | public class HttpDirectTemplateDownloader extends DirectTemplateDownloaderImpl { | ||||||
| 
 | 
 | ||||||
|     private HttpClient client; |     protected HttpClient client; | ||||||
|     private static final MultiThreadedHttpConnectionManager s_httpClientManager = new MultiThreadedHttpConnectionManager(); |     private static final MultiThreadedHttpConnectionManager s_httpClientManager = new MultiThreadedHttpConnectionManager(); | ||||||
|     private static final int CHUNK_SIZE = 1024 * 1024; //1M |  | ||||||
|     protected HttpMethodRetryHandler myretryhandler; |  | ||||||
|     public static final Logger s_logger = Logger.getLogger(HttpDirectTemplateDownloader.class.getName()); |     public static final Logger s_logger = Logger.getLogger(HttpDirectTemplateDownloader.class.getName()); | ||||||
|     protected GetMethod request; |     protected GetMethod request; | ||||||
|  |     protected Map<String, String> reqHeaders = new HashMap<>(); | ||||||
| 
 | 
 | ||||||
|     public HttpDirectTemplateDownloader(String url, Long templateId, String destPoolPath, String checksum, Map<String, String> headers) { |     public HttpDirectTemplateDownloader(String url, Long templateId, String destPoolPath, String checksum, Map<String, String> headers) { | ||||||
|         super(url, destPoolPath, templateId, checksum); |         super(url, destPoolPath, templateId, checksum); | ||||||
|  |         s_httpClientManager.getParams().setConnectionTimeout(5000); | ||||||
|  |         s_httpClientManager.getParams().setSoTimeout(5000); | ||||||
|         client = new HttpClient(s_httpClientManager); |         client = new HttpClient(s_httpClientManager); | ||||||
|         myretryhandler = createRetryTwiceHandler(); |  | ||||||
|         request = createRequest(url, headers); |         request = createRequest(url, headers); | ||||||
|         String downloadDir = getDirectDownloadTempPath(templateId); |         String downloadDir = getDirectDownloadTempPath(templateId); | ||||||
|         createTemporaryDirectoryAndFile(downloadDir); |         createTemporaryDirectoryAndFile(downloadDir); | ||||||
| @ -64,50 +62,34 @@ public class HttpDirectTemplateDownloader extends DirectTemplateDownloaderImpl { | |||||||
| 
 | 
 | ||||||
|     protected GetMethod createRequest(String downloadUrl, Map<String, String> headers) { |     protected GetMethod createRequest(String downloadUrl, Map<String, String> headers) { | ||||||
|         GetMethod request = new GetMethod(downloadUrl); |         GetMethod request = new GetMethod(downloadUrl); | ||||||
|         request.getParams().setParameter(HttpMethodParams.RETRY_HANDLER, myretryhandler); |  | ||||||
|         request.setFollowRedirects(true); |         request.setFollowRedirects(true); | ||||||
|         if (MapUtils.isNotEmpty(headers)) { |         if (MapUtils.isNotEmpty(headers)) { | ||||||
|             for (String key : headers.keySet()) { |             for (String key : headers.keySet()) { | ||||||
|                 request.setRequestHeader(key, headers.get(key)); |                 request.setRequestHeader(key, headers.get(key)); | ||||||
|  |                 reqHeaders.put(key, headers.get(key)); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return request; |         return request; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     protected HttpMethodRetryHandler createRetryTwiceHandler() { |  | ||||||
|         return new HttpMethodRetryHandler() { |  | ||||||
|             @Override |  | ||||||
|             public boolean retryMethod(final HttpMethod method, final IOException exception, int executionCount) { |  | ||||||
|                 if (executionCount >= 2) { |  | ||||||
|                     // Do not retry if over max retry count |  | ||||||
|                     return false; |  | ||||||
|                 } |  | ||||||
|                 if (exception instanceof NoHttpResponseException) { |  | ||||||
|                     // Retry if the server dropped connection on us |  | ||||||
|                     return true; |  | ||||||
|                 } |  | ||||||
|                 if (!method.isRequestSent()) { |  | ||||||
|                     // Retry if the request has not been sent fully or |  | ||||||
|                     // if it's OK to retry methods that have been sent |  | ||||||
|                     return true; |  | ||||||
|                 } |  | ||||||
|                 // otherwise do not retry |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean downloadTemplate() { |     public boolean downloadTemplate() { | ||||||
|         try { |         try { | ||||||
|             client.executeMethod(request); |             int status = client.executeMethod(request); | ||||||
|         } catch (IOException e) { |             if (status != HttpStatus.SC_OK) { | ||||||
|             throw new CloudRuntimeException("Error on HTTP request: " + e.getMessage()); |                 s_logger.warn("Not able to download template, status code: " + status); | ||||||
|  |                 return false; | ||||||
|             } |             } | ||||||
|             return performDownload(); |             return performDownload(); | ||||||
|  |         } catch (IOException e) { | ||||||
|  |             throw new CloudRuntimeException("Error on HTTP request: " + e.getMessage()); | ||||||
|  |         } finally { | ||||||
|  |             request.releaseConnection(); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     protected boolean performDownload() { |     protected boolean performDownload() { | ||||||
|  |         s_logger.info("Downloading template " + getTemplateId() + " from " + getUrl() + " to: " + getDownloadedFilePath()); | ||||||
|         try ( |         try ( | ||||||
|                 InputStream in = request.getResponseBodyAsStream(); |                 InputStream in = request.getResponseBodyAsStream(); | ||||||
|                 OutputStream out = new FileOutputStream(getDownloadedFilePath()); |                 OutputStream out = new FileOutputStream(getDownloadedFilePath()); | ||||||
|  | |||||||
| @ -24,6 +24,8 @@ import com.cloud.utils.script.Script; | |||||||
| import org.apache.commons.io.IOUtils; | import org.apache.commons.io.IOUtils; | ||||||
| import org.apache.http.HttpEntity; | import org.apache.http.HttpEntity; | ||||||
| import org.apache.http.client.methods.CloseableHttpResponse; | import org.apache.http.client.methods.CloseableHttpResponse; | ||||||
|  | import org.apache.commons.collections.MapUtils; | ||||||
|  | import org.apache.http.client.config.RequestConfig; | ||||||
| import org.apache.http.client.methods.HttpGet; | import org.apache.http.client.methods.HttpGet; | ||||||
| import org.apache.http.client.methods.HttpUriRequest; | import org.apache.http.client.methods.HttpUriRequest; | ||||||
| import org.apache.http.conn.ssl.SSLConnectionSocketFactory; | import org.apache.http.conn.ssl.SSLConnectionSocketFactory; | ||||||
| @ -44,14 +46,15 @@ import java.security.KeyStore; | |||||||
| import java.security.KeyStoreException; | import java.security.KeyStoreException; | ||||||
| import java.security.NoSuchAlgorithmException; | import java.security.NoSuchAlgorithmException; | ||||||
| import java.security.cert.CertificateException; | import java.security.cert.CertificateException; | ||||||
|  | import java.util.Map; | ||||||
| 
 | 
 | ||||||
| public class HttpsDirectTemplateDownloader extends HttpDirectTemplateDownloader { | public class HttpsDirectTemplateDownloader extends HttpDirectTemplateDownloader { | ||||||
| 
 | 
 | ||||||
|     private CloseableHttpClient httpsClient; |     private CloseableHttpClient httpsClient; | ||||||
|     private HttpUriRequest req; |     private HttpUriRequest req; | ||||||
| 
 | 
 | ||||||
|     public HttpsDirectTemplateDownloader(String url, Long templateId, String destPoolPath, String checksum) { |     public HttpsDirectTemplateDownloader(String url, Long templateId, String destPoolPath, String checksum, Map<String, String> headers) { | ||||||
|         super(url, templateId, destPoolPath, checksum, null); |         super(url, templateId, destPoolPath, checksum, headers); | ||||||
|         SSLContext sslcontext = null; |         SSLContext sslcontext = null; | ||||||
|         try { |         try { | ||||||
|             sslcontext = getSSLContext(); |             sslcontext = getSSLContext(); | ||||||
| @ -59,12 +62,21 @@ public class HttpsDirectTemplateDownloader extends HttpDirectTemplateDownloader | |||||||
|             throw new CloudRuntimeException("Failure getting SSL context for HTTPS downloader: " + e.getMessage()); |             throw new CloudRuntimeException("Failure getting SSL context for HTTPS downloader: " + e.getMessage()); | ||||||
|         } |         } | ||||||
|         SSLConnectionSocketFactory factory = new SSLConnectionSocketFactory(sslcontext, SSLConnectionSocketFactory.ALLOW_ALL_HOSTNAME_VERIFIER); |         SSLConnectionSocketFactory factory = new SSLConnectionSocketFactory(sslcontext, SSLConnectionSocketFactory.ALLOW_ALL_HOSTNAME_VERIFIER); | ||||||
|         httpsClient = HttpClients.custom().setSSLSocketFactory(factory).build(); |         RequestConfig config = RequestConfig.custom() | ||||||
|         req = createUriRequest(url); |                 .setConnectTimeout(5000) | ||||||
|  |                 .setConnectionRequestTimeout(5000) | ||||||
|  |                 .setSocketTimeout(5000).build(); | ||||||
|  |         httpsClient = HttpClients.custom().setSSLSocketFactory(factory).setDefaultRequestConfig(config).build(); | ||||||
|  |         createUriRequest(url, headers); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     protected HttpUriRequest createUriRequest(String downloadUrl) { |     protected void createUriRequest(String downloadUrl, Map<String, String> headers) { | ||||||
|         return new HttpGet(downloadUrl); |         req = new HttpGet(downloadUrl); | ||||||
|  |         if (MapUtils.isNotEmpty(headers)) { | ||||||
|  |             for (String headerKey: headers.keySet()) { | ||||||
|  |                 req.setHeader(headerKey, headers.get(headerKey)); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private SSLContext getSSLContext() throws KeyStoreException, NoSuchAlgorithmException, CertificateException, IOException, KeyManagementException { |     private SSLContext getSSLContext() throws KeyStoreException, NoSuchAlgorithmException, CertificateException, IOException, KeyManagementException { | ||||||
| @ -98,6 +110,7 @@ public class HttpsDirectTemplateDownloader extends HttpDirectTemplateDownloader | |||||||
|      * Consume response and persist it on getDownloadedFilePath() file |      * Consume response and persist it on getDownloadedFilePath() file | ||||||
|      */ |      */ | ||||||
|     protected boolean consumeResponse(CloseableHttpResponse response) { |     protected boolean consumeResponse(CloseableHttpResponse response) { | ||||||
|  |         s_logger.info("Downloading template " + getTemplateId() + " from " + getUrl() + " to: " + getDownloadedFilePath()); | ||||||
|         if (response.getStatusLine().getStatusCode() != 200) { |         if (response.getStatusLine().getStatusCode() != 200) { | ||||||
|             throw new CloudRuntimeException("Error on HTTPS response"); |             throw new CloudRuntimeException("Error on HTTPS response"); | ||||||
|         } |         } | ||||||
|  | |||||||
| @ -18,32 +18,81 @@ | |||||||
| // | // | ||||||
| package com.cloud.agent.direct.download; | package com.cloud.agent.direct.download; | ||||||
| 
 | 
 | ||||||
| import com.cloud.utils.script.Script; | import com.cloud.utils.UriUtils; | ||||||
|  | import com.cloud.utils.exception.CloudRuntimeException; | ||||||
|  | import org.apache.commons.collections.CollectionUtils; | ||||||
|  | import org.apache.commons.lang.StringUtils; | ||||||
|  | import org.apache.log4j.Logger; | ||||||
| 
 | 
 | ||||||
| import java.io.File; | import java.io.File; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.Map; | ||||||
|  | import java.util.Random; | ||||||
| 
 | 
 | ||||||
| public class MetalinkDirectTemplateDownloader extends DirectTemplateDownloaderImpl { | public class MetalinkDirectTemplateDownloader extends HttpDirectTemplateDownloader { | ||||||
| 
 | 
 | ||||||
|     private String downloadDir; |     private String metalinkUrl; | ||||||
|  |     private List<String> metalinkUrls; | ||||||
|  |     private List<String> metalinkChecksums; | ||||||
|  |     private Random random = new Random(); | ||||||
|  |     private static final Logger s_logger = Logger.getLogger(MetalinkDirectTemplateDownloader.class.getName()); | ||||||
| 
 | 
 | ||||||
|     public MetalinkDirectTemplateDownloader(String url, String destPoolPath, Long templateId, String checksum) { |     public MetalinkDirectTemplateDownloader(String url, String destPoolPath, Long templateId, String checksum, Map<String, String> headers) { | ||||||
|         super(url, destPoolPath, templateId, checksum); |         super(url, templateId, destPoolPath, checksum, headers); | ||||||
|         String relativeDir = getDirectDownloadTempPath(templateId); |         metalinkUrl = url; | ||||||
|         downloadDir = getDestPoolPath() + File.separator + relativeDir; |         metalinkUrls = UriUtils.getMetalinkUrls(metalinkUrl); | ||||||
|         createFolder(downloadDir); |         metalinkChecksums = UriUtils.getMetalinkChecksums(metalinkUrl); | ||||||
|  |         if (CollectionUtils.isEmpty(metalinkUrls)) { | ||||||
|  |             throw new CloudRuntimeException("No urls found on metalink file: " + metalinkUrl + ". Not possible to download template " + templateId); | ||||||
|  |         } | ||||||
|  |         setUrl(metalinkUrls.get(0)); | ||||||
|  |         s_logger.info("Metalink downloader created, metalink url: " + metalinkUrl + " parsed - " + | ||||||
|  |                 metalinkUrls.size() + " urls and " + | ||||||
|  |                 (CollectionUtils.isNotEmpty(metalinkChecksums) ? metalinkChecksums.size() : "0") + " checksums found"); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|     public boolean downloadTemplate() { |     public boolean downloadTemplate() { | ||||||
|         String downloadCommand = "aria2c " + getUrl() + " -d " + downloadDir + " --check-integrity=true"; |         if (StringUtils.isBlank(getUrl())) { | ||||||
|         Script.runSimpleBashScript(downloadCommand); |             throw new CloudRuntimeException("Download url has not been set, aborting"); | ||||||
|         //Remove .metalink file |  | ||||||
|         Script.runSimpleBashScript("rm -f " + downloadDir + File.separator + getFileNameFromUrl()); |  | ||||||
|         String fileName = Script.runSimpleBashScript("ls " + downloadDir); |  | ||||||
|         if (fileName == null) { |  | ||||||
|             return false; |  | ||||||
|         } |         } | ||||||
|         setDownloadedFilePath(downloadDir + File.separator + fileName); |         String downloadDir = getDirectDownloadTempPath(getTemplateId()); | ||||||
|         return true; |         boolean downloaded = false; | ||||||
|  |         int i = 0; | ||||||
|  |         do { | ||||||
|  |             if (!isRedownload()) { | ||||||
|  |                 setUrl(metalinkUrls.get(i)); | ||||||
|  |             } | ||||||
|  |             s_logger.info("Trying to download template from url: " + getUrl()); | ||||||
|  |             try { | ||||||
|  |                 File f = new File(getDestPoolPath() + File.separator + downloadDir + File.separator + getFileNameFromUrl()); | ||||||
|  |                 if (f.exists()) { | ||||||
|  |                     f.delete(); | ||||||
|  |                     f.createNewFile(); | ||||||
|  |                 } | ||||||
|  |                 setDownloadedFilePath(f.getAbsolutePath()); | ||||||
|  |                 request = createRequest(getUrl(), reqHeaders); | ||||||
|  |                 downloaded = super.downloadTemplate(); | ||||||
|  |                 if (downloaded) { | ||||||
|  |                     s_logger.info("Successfully downloaded template from url: " + getUrl()); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |             } catch (Exception e) { | ||||||
|  |                 s_logger.error("Error downloading template: " + getTemplateId() + " from " + getUrl() + ": " + e.getMessage()); | ||||||
|  |             } | ||||||
|  |             i++; | ||||||
|  |         } | ||||||
|  |         while (!downloaded && !isRedownload() && i < metalinkUrls.size()); | ||||||
|  |         return downloaded; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public boolean validateChecksum() { | ||||||
|  |         if (StringUtils.isBlank(getChecksum()) && CollectionUtils.isNotEmpty(metalinkChecksums)) { | ||||||
|  |             String chk = metalinkChecksums.get(random.nextInt(metalinkChecksums.size())); | ||||||
|  |             setChecksum(chk); | ||||||
|  |             s_logger.info("Checksum not provided but " + metalinkChecksums.size() + " found on metalink file, performing checksum using one of them: " + chk); | ||||||
|  |         } | ||||||
|  |         return super.validateChecksum(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -581,6 +581,8 @@ public class EventTypes { | |||||||
|     public static final String EVENT_ANNOTATION_CREATE = "ANNOTATION.CREATE"; |     public static final String EVENT_ANNOTATION_CREATE = "ANNOTATION.CREATE"; | ||||||
|     public static final String EVENT_ANNOTATION_REMOVE = "ANNOTATION.REMOVE"; |     public static final String EVENT_ANNOTATION_REMOVE = "ANNOTATION.REMOVE"; | ||||||
| 
 | 
 | ||||||
|  |     public static final String EVENT_TEMPLATE_DIRECT_DOWNLOAD_FAILURE = "TEMPLATE.DIRECT.DOWNLOAD.FAILURE"; | ||||||
|  |     public static final String EVENT_ISO_DIRECT_DOWNLOAD_FAILURE = "ISO.DIRECT.DOWNLOAD.FAILURE"; | ||||||
| 
 | 
 | ||||||
|     static { |     static { | ||||||
| 
 | 
 | ||||||
| @ -972,6 +974,9 @@ public class EventTypes { | |||||||
| 
 | 
 | ||||||
|         entityEventDetails.put(EVENT_ANNOTATION_CREATE, Annotation.class); |         entityEventDetails.put(EVENT_ANNOTATION_CREATE, Annotation.class); | ||||||
|         entityEventDetails.put(EVENT_ANNOTATION_REMOVE, Annotation.class); |         entityEventDetails.put(EVENT_ANNOTATION_REMOVE, Annotation.class); | ||||||
|  | 
 | ||||||
|  |         entityEventDetails.put(EVENT_TEMPLATE_DIRECT_DOWNLOAD_FAILURE, VirtualMachineTemplate.class); | ||||||
|  |         entityEventDetails.put(EVENT_ISO_DIRECT_DOWNLOAD_FAILURE, "Iso"); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static String getEntityForEvent(String eventName) { |     public static String getEntityForEvent(String eventName) { | ||||||
|  | |||||||
| @ -67,9 +67,12 @@ public class UploadTemplateDirectDownloadCertificate extends BaseCmd { | |||||||
|             throw new ServerApiException(ApiErrorCode.PARAM_ERROR, "Currently supporting KVM hosts only"); |             throw new ServerApiException(ApiErrorCode.PARAM_ERROR, "Currently supporting KVM hosts only"); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         SuccessResponse response = new SuccessResponse(getCommandName()); | ||||||
|         try { |         try { | ||||||
|             directDownloadManager.uploadCertificateToHosts(certificate, name);; |             LOG.debug("Uploading certificate " + name + " to agents for Direct Download"); | ||||||
|             setResponseObject(new SuccessResponse(getCommandName())); |             boolean result = directDownloadManager.uploadCertificateToHosts(certificate, name, hypervisor); | ||||||
|  |             response.setSuccess(result); | ||||||
|  |             setResponseObject(response); | ||||||
|         } catch (Exception e) { |         } catch (Exception e) { | ||||||
|             throw new ServerApiException(ApiErrorCode.INTERNAL_ERROR, e.getMessage()); |             throw new ServerApiException(ApiErrorCode.INTERNAL_ERROR, e.getMessage()); | ||||||
|         } |         } | ||||||
| @ -77,7 +80,7 @@ public class UploadTemplateDirectDownloadCertificate extends BaseCmd { | |||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|     public String getCommandName() { |     public String getCommandName() { | ||||||
|         return UploadTemplateDirectDownloadCertificate.APINAME; |         return APINAME.toLowerCase() + BaseCmd.RESPONSE_SUFFIX; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|  | |||||||
| @ -19,37 +19,138 @@ | |||||||
| package com.cloud.storage.template; | package com.cloud.storage.template; | ||||||
| 
 | 
 | ||||||
| import com.cloud.storage.StorageLayer; | import com.cloud.storage.StorageLayer; | ||||||
| import com.cloud.utils.script.Script; | import com.cloud.utils.UriUtils; | ||||||
|  | import org.apache.commons.httpclient.HttpClient; | ||||||
|  | import org.apache.commons.httpclient.HttpMethod; | ||||||
|  | import org.apache.commons.httpclient.HttpMethodRetryHandler; | ||||||
|  | import org.apache.commons.httpclient.MultiThreadedHttpConnectionManager; | ||||||
|  | import org.apache.commons.httpclient.NoHttpResponseException; | ||||||
|  | import org.apache.commons.httpclient.methods.GetMethod; | ||||||
|  | import org.apache.commons.httpclient.params.HttpMethodParams; | ||||||
|  | import org.apache.commons.io.IOUtils; | ||||||
| import org.apache.log4j.Logger; | import org.apache.log4j.Logger; | ||||||
|  | import org.springframework.util.CollectionUtils; | ||||||
| 
 | 
 | ||||||
| import java.io.File; | import java.io.File; | ||||||
|  | import java.io.FileOutputStream; | ||||||
|  | import java.io.IOException; | ||||||
|  | import java.io.InputStream; | ||||||
|  | import java.io.OutputStream; | ||||||
|  | import java.util.List; | ||||||
| 
 | 
 | ||||||
| public class MetalinkTemplateDownloader extends TemplateDownloaderBase implements TemplateDownloader { | public class MetalinkTemplateDownloader extends TemplateDownloaderBase implements TemplateDownloader { | ||||||
| 
 | 
 | ||||||
|     private TemplateDownloader.Status status = TemplateDownloader.Status.NOT_STARTED; |     private TemplateDownloader.Status status = TemplateDownloader.Status.NOT_STARTED; | ||||||
|  |     protected HttpClient client; | ||||||
|  |     private static final MultiThreadedHttpConnectionManager s_httpClientManager = new MultiThreadedHttpConnectionManager(); | ||||||
|  |     protected HttpMethodRetryHandler myretryhandler; | ||||||
|  |     protected GetMethod request; | ||||||
|  |     private boolean toFileSet = false; | ||||||
| 
 | 
 | ||||||
|     private static final Logger LOGGER = Logger.getLogger(MetalinkTemplateDownloader.class.getName()); |     private static final Logger LOGGER = Logger.getLogger(MetalinkTemplateDownloader.class.getName()); | ||||||
| 
 | 
 | ||||||
|     public MetalinkTemplateDownloader(StorageLayer storageLayer, String downloadUrl, String toDir, DownloadCompleteCallback callback, long maxTemplateSize) { |     public MetalinkTemplateDownloader(StorageLayer storageLayer, String downloadUrl, String toDir, DownloadCompleteCallback callback, long maxTemplateSize) { | ||||||
|         super(storageLayer, downloadUrl, toDir, maxTemplateSize, callback); |         super(storageLayer, downloadUrl, toDir, maxTemplateSize, callback); | ||||||
|         String[] parts = _downloadUrl.split("/"); |         s_httpClientManager.getParams().setConnectionTimeout(5000); | ||||||
|         String filename = parts[parts.length - 1]; |         client = new HttpClient(s_httpClientManager); | ||||||
|         _toFile = toDir + File.separator + filename; |         myretryhandler = createRetryTwiceHandler(); | ||||||
|  |         request = createRequest(downloadUrl); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     protected GetMethod createRequest(String downloadUrl) { | ||||||
|  |         GetMethod request = new GetMethod(downloadUrl); | ||||||
|  |         request.getParams().setParameter(HttpMethodParams.RETRY_HANDLER, myretryhandler); | ||||||
|  |         request.setFollowRedirects(true); | ||||||
|  |         if (!toFileSet) { | ||||||
|  |             String[] parts = downloadUrl.split("/"); | ||||||
|  |             String filename = parts[parts.length - 1]; | ||||||
|  |             _toFile = _toDir + File.separator + filename; | ||||||
|  |             toFileSet = true; | ||||||
|  |         } | ||||||
|  |         return request; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected HttpMethodRetryHandler createRetryTwiceHandler() { | ||||||
|  |         return new HttpMethodRetryHandler() { | ||||||
|  |             @Override | ||||||
|  |             public boolean retryMethod(final HttpMethod method, final IOException exception, int executionCount) { | ||||||
|  |                 if (executionCount >= 2) { | ||||||
|  |                     // Do not retry if over max retry count | ||||||
|  |                     return false; | ||||||
|  |                 } | ||||||
|  |                 if (exception instanceof NoHttpResponseException) { | ||||||
|  |                     // Retry if the server dropped connection on us | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |                 if (!method.isRequestSent()) { | ||||||
|  |                     // Retry if the request has not been sent fully or | ||||||
|  |                     // if it's OK to retry methods that have been sent | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |                 // otherwise do not retry | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private boolean downloadTemplate() { | ||||||
|  |         try { | ||||||
|  |             client.executeMethod(request); | ||||||
|  |         } catch (IOException e) { | ||||||
|  |             LOGGER.error("Error on HTTP request: " + e.getMessage()); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         return performDownload(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private boolean performDownload() { | ||||||
|  |         try ( | ||||||
|  |                 InputStream in = request.getResponseBodyAsStream(); | ||||||
|  |                 OutputStream out = new FileOutputStream(_toFile); | ||||||
|  |         ) { | ||||||
|  |             IOUtils.copy(in, out); | ||||||
|  |         } catch (IOException e) { | ||||||
|  |             LOGGER.error("Error downloading template from: " + _downloadUrl + " due to: " + e.getMessage()); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|     @Override |     @Override | ||||||
|     public long download(boolean resume, DownloadCompleteCallback callback) { |     public long download(boolean resume, DownloadCompleteCallback callback) { | ||||||
|         if (!status.equals(Status.NOT_STARTED)) { |         if (_status == Status.ABORTED || _status == Status.UNRECOVERABLE_ERROR || _status == Status.DOWNLOAD_FINISHED) { | ||||||
|             // Only start downloading if we haven't started yet. |  | ||||||
|             LOGGER.debug("Template download is already started, not starting again. Template: " + _downloadUrl); |  | ||||||
|             return 0; |             return 0; | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         LOGGER.info("Starting metalink download from: " + _downloadUrl); | ||||||
|  |         _start = System.currentTimeMillis(); | ||||||
|  | 
 | ||||||
|         status = Status.IN_PROGRESS; |         status = Status.IN_PROGRESS; | ||||||
|         Script.runSimpleBashScript("aria2c " + _downloadUrl + " -d " + _toDir); |         List<String> metalinkUrls = UriUtils.getMetalinkUrls(_downloadUrl); | ||||||
|  |         if (CollectionUtils.isEmpty(metalinkUrls)) { | ||||||
|  |             LOGGER.error("No URLs found for metalink: " + _downloadUrl); | ||||||
|  |             status = Status.UNRECOVERABLE_ERROR; | ||||||
|  |             return 0; | ||||||
|  |         } | ||||||
|  |         boolean downloaded = false; | ||||||
|  |         int i = 0; | ||||||
|  |         while (!downloaded && i < metalinkUrls.size()) { | ||||||
|  |             String url = metalinkUrls.get(i); | ||||||
|  |             request = createRequest(url); | ||||||
|  |             downloaded = downloadTemplate(); | ||||||
|  |             i++; | ||||||
|  |         } | ||||||
|  |         if (!downloaded) { | ||||||
|  |             LOGGER.error("Template couldnt be downloaded"); | ||||||
|  |             status = Status.UNRECOVERABLE_ERROR; | ||||||
|  |             return 0; | ||||||
|  |         } | ||||||
|  |         LOGGER.info("Template downloaded successfully on: " + _toFile); | ||||||
|         status = Status.DOWNLOAD_FINISHED; |         status = Status.DOWNLOAD_FINISHED; | ||||||
|         String sizeResult = Script.runSimpleBashScript("ls -als " + _toFile + " | awk '{print $1}'"); |         _downloadTime = System.currentTimeMillis() - _start; | ||||||
|         long size = Long.parseLong(sizeResult); |         if (_callback != null) { | ||||||
|         return size; |             _callback.downloadComplete(status); | ||||||
|  |         } | ||||||
|  |         return _totalBytes; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
| @ -63,4 +164,13 @@ public class MetalinkTemplateDownloader extends TemplateDownloaderBase implement | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     @Override | ||||||
|  |     public Status getStatus() { | ||||||
|  |         return status; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public void setStatus(Status status) { | ||||||
|  |         this.status = status; | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -24,11 +24,13 @@ public class DirectDownloadAnswer extends Answer { | |||||||
| 
 | 
 | ||||||
|     private Long templateSize; |     private Long templateSize; | ||||||
|     private String installPath; |     private String installPath; | ||||||
|  |     private boolean retryOnOtherHosts; | ||||||
| 
 | 
 | ||||||
|     public DirectDownloadAnswer(final boolean result, final String msg) { |     public DirectDownloadAnswer(final boolean result, final String msg, final boolean retry) { | ||||||
|         super(null); |         super(null); | ||||||
|         this.result = result; |         this.result = result; | ||||||
|         this.details = msg; |         this.details = msg; | ||||||
|  |         this.retryOnOtherHosts = retry; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public DirectDownloadAnswer(final boolean result, final Long size, final String installPath) { |     public DirectDownloadAnswer(final boolean result, final Long size, final String installPath) { | ||||||
| @ -45,4 +47,8 @@ public class DirectDownloadAnswer extends Answer { | |||||||
|     public String getInstallPath() { |     public String getInstallPath() { | ||||||
|         return installPath; |         return installPath; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public boolean isRetryOnOtherHosts() { | ||||||
|  |         return retryOnOtherHosts; | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -22,6 +22,8 @@ package org.apache.cloudstack.agent.directdownload; | |||||||
| import org.apache.cloudstack.storage.command.StorageSubSystemCommand; | import org.apache.cloudstack.storage.command.StorageSubSystemCommand; | ||||||
| import org.apache.cloudstack.storage.to.PrimaryDataStoreTO; | import org.apache.cloudstack.storage.to.PrimaryDataStoreTO; | ||||||
| 
 | 
 | ||||||
|  | import java.util.Map; | ||||||
|  | 
 | ||||||
| public abstract class DirectDownloadCommand extends StorageSubSystemCommand { | public abstract class DirectDownloadCommand extends StorageSubSystemCommand { | ||||||
| 
 | 
 | ||||||
|     public enum DownloadProtocol { |     public enum DownloadProtocol { | ||||||
| @ -32,12 +34,14 @@ public abstract class DirectDownloadCommand extends StorageSubSystemCommand { | |||||||
|     private Long templateId; |     private Long templateId; | ||||||
|     private PrimaryDataStoreTO destPool; |     private PrimaryDataStoreTO destPool; | ||||||
|     private String checksum; |     private String checksum; | ||||||
|  |     private Map<String, String> headers; | ||||||
| 
 | 
 | ||||||
|     protected DirectDownloadCommand (final String url, final Long templateId, final PrimaryDataStoreTO destPool, final String checksum) { |     protected DirectDownloadCommand (final String url, final Long templateId, final PrimaryDataStoreTO destPool, final String checksum, final Map<String, String> headers) { | ||||||
|         this.url = url; |         this.url = url; | ||||||
|         this.templateId = templateId; |         this.templateId = templateId; | ||||||
|         this.destPool = destPool; |         this.destPool = destPool; | ||||||
|         this.checksum = checksum; |         this.checksum = checksum; | ||||||
|  |         this.headers = headers; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public String getUrl() { |     public String getUrl() { | ||||||
| @ -56,6 +60,10 @@ public abstract class DirectDownloadCommand extends StorageSubSystemCommand { | |||||||
|         return checksum; |         return checksum; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public Map<String, String> getHeaders() { | ||||||
|  |         return headers; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     @Override |     @Override | ||||||
|     public void setExecuteInSequence(boolean inSeq) { |     public void setExecuteInSequence(boolean inSeq) { | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -24,15 +24,8 @@ import java.util.Map; | |||||||
| 
 | 
 | ||||||
| public class HttpDirectDownloadCommand extends DirectDownloadCommand { | public class HttpDirectDownloadCommand extends DirectDownloadCommand { | ||||||
| 
 | 
 | ||||||
|     private Map<String, String> headers; |  | ||||||
| 
 |  | ||||||
|     public HttpDirectDownloadCommand(String url, Long templateId, PrimaryDataStoreTO destPool, String checksum, Map<String, String> headers) { |     public HttpDirectDownloadCommand(String url, Long templateId, PrimaryDataStoreTO destPool, String checksum, Map<String, String> headers) { | ||||||
|         super(url, templateId, destPool, checksum); |         super(url, templateId, destPool, checksum, headers); | ||||||
|         this.headers = headers; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     public Map<String, String> getHeaders() { |  | ||||||
|         return headers; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
|  | |||||||
| @ -26,6 +26,6 @@ import java.util.Map; | |||||||
| public class HttpsDirectDownloadCommand extends DirectDownloadCommand { | public class HttpsDirectDownloadCommand extends DirectDownloadCommand { | ||||||
| 
 | 
 | ||||||
|     public HttpsDirectDownloadCommand(String url, Long templateId, PrimaryDataStoreTO destPool, String checksum, Map<String, String> headers) { |     public HttpsDirectDownloadCommand(String url, Long templateId, PrimaryDataStoreTO destPool, String checksum, Map<String, String> headers) { | ||||||
|         super(url, templateId, destPool, checksum); |         super(url, templateId, destPool, checksum, headers); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @ -20,10 +20,12 @@ package org.apache.cloudstack.agent.directdownload; | |||||||
| 
 | 
 | ||||||
| import org.apache.cloudstack.storage.to.PrimaryDataStoreTO; | import org.apache.cloudstack.storage.to.PrimaryDataStoreTO; | ||||||
| 
 | 
 | ||||||
|  | import java.util.Map; | ||||||
|  | 
 | ||||||
| public class MetalinkDirectDownloadCommand extends DirectDownloadCommand { | public class MetalinkDirectDownloadCommand extends DirectDownloadCommand { | ||||||
| 
 | 
 | ||||||
|     public MetalinkDirectDownloadCommand(String url, Long templateId, PrimaryDataStoreTO destPool, String checksum) { |     public MetalinkDirectDownloadCommand(String url, Long templateId, PrimaryDataStoreTO destPool, String checksum, Map<String, String> headers) { | ||||||
|         super(url, templateId, destPool, checksum); |         super(url, templateId, destPool, checksum, headers); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
|  | |||||||
| @ -20,10 +20,12 @@ package org.apache.cloudstack.agent.directdownload; | |||||||
| 
 | 
 | ||||||
| import org.apache.cloudstack.storage.to.PrimaryDataStoreTO; | import org.apache.cloudstack.storage.to.PrimaryDataStoreTO; | ||||||
| 
 | 
 | ||||||
|  | import java.util.Map; | ||||||
|  | 
 | ||||||
| public class NfsDirectDownloadCommand extends DirectDownloadCommand { | public class NfsDirectDownloadCommand extends DirectDownloadCommand { | ||||||
| 
 | 
 | ||||||
|     public NfsDirectDownloadCommand(final String url, final Long templateId, final PrimaryDataStoreTO destPool, final String checksum) { |     public NfsDirectDownloadCommand(final String url, final Long templateId, final PrimaryDataStoreTO destPool, final String checksum, final Map<String, String> headers) { | ||||||
|         super(url, templateId, destPool, checksum); |         super(url, templateId, destPool, checksum, headers); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
|  | |||||||
| @ -27,5 +27,5 @@ public interface DirectDownloadService { | |||||||
|     /** |     /** | ||||||
|      * Upload client certificate to each running host |      * Upload client certificate to each running host | ||||||
|      */ |      */ | ||||||
|     boolean uploadCertificateToHosts(String certificateCer, String certificateName); |     boolean uploadCertificateToHosts(String certificateCer, String certificateName, String hypervisor); | ||||||
| } | } | ||||||
|  | |||||||
| @ -42,6 +42,7 @@ import com.cloud.agent.direct.download.HttpDirectTemplateDownloader; | |||||||
| import com.cloud.agent.direct.download.MetalinkDirectTemplateDownloader; | import com.cloud.agent.direct.download.MetalinkDirectTemplateDownloader; | ||||||
| import com.cloud.agent.direct.download.NfsDirectTemplateDownloader; | import com.cloud.agent.direct.download.NfsDirectTemplateDownloader; | ||||||
| import com.cloud.agent.direct.download.HttpsDirectTemplateDownloader; | import com.cloud.agent.direct.download.HttpsDirectTemplateDownloader; | ||||||
|  | import com.cloud.exception.InvalidParameterValueException; | ||||||
| import org.apache.cloudstack.agent.directdownload.HttpsDirectDownloadCommand; | import org.apache.cloudstack.agent.directdownload.HttpsDirectDownloadCommand; | ||||||
| import org.apache.cloudstack.agent.directdownload.DirectDownloadCommand; | import org.apache.cloudstack.agent.directdownload.DirectDownloadCommand; | ||||||
| import org.apache.cloudstack.agent.directdownload.HttpDirectDownloadCommand; | import org.apache.cloudstack.agent.directdownload.HttpDirectDownloadCommand; | ||||||
| @ -1071,19 +1072,9 @@ public class KVMStorageProcessor implements StorageProcessor { | |||||||
|         final DiskTO disk = cmd.getDisk(); |         final DiskTO disk = cmd.getDisk(); | ||||||
|         final TemplateObjectTO isoTO = (TemplateObjectTO)disk.getData(); |         final TemplateObjectTO isoTO = (TemplateObjectTO)disk.getData(); | ||||||
|         final DataStoreTO store = isoTO.getDataStore(); |         final DataStoreTO store = isoTO.getDataStore(); | ||||||
|         String dataStoreUrl = null; | 
 | ||||||
|         if (store instanceof NfsTO) { |  | ||||||
|             NfsTO nfsStore = (NfsTO)store; |  | ||||||
|             dataStoreUrl = nfsStore.getUrl(); |  | ||||||
|         } else if (store instanceof PrimaryDataStoreTO && ((PrimaryDataStoreTO) store).getPoolType().equals(StoragePoolType.NetworkFilesystem)) { |  | ||||||
|             //In order to support directly downloaded ISOs |  | ||||||
|             String psHost = ((PrimaryDataStoreTO) store).getHost(); |  | ||||||
|             String psPath = ((PrimaryDataStoreTO) store).getPath(); |  | ||||||
|             dataStoreUrl = "nfs://" + psHost + File.separator + psPath; |  | ||||||
|         } else { |  | ||||||
|             return new AttachAnswer("unsupported protocol"); |  | ||||||
|         } |  | ||||||
|         try { |         try { | ||||||
|  |             String dataStoreUrl = getDataStoreUrlFromStore(store); | ||||||
|             final Connect conn = LibvirtConnection.getConnectionByVmName(cmd.getVmName()); |             final Connect conn = LibvirtConnection.getConnectionByVmName(cmd.getVmName()); | ||||||
|             attachOrDetachISO(conn, cmd.getVmName(), dataStoreUrl + File.separator + isoTO.getPath(), true); |             attachOrDetachISO(conn, cmd.getVmName(), dataStoreUrl + File.separator + isoTO.getPath(), true); | ||||||
|         } catch (final LibvirtException e) { |         } catch (final LibvirtException e) { | ||||||
| @ -1092,6 +1083,8 @@ public class KVMStorageProcessor implements StorageProcessor { | |||||||
|             return new Answer(cmd, false, e.toString()); |             return new Answer(cmd, false, e.toString()); | ||||||
|         } catch (final InternalErrorException e) { |         } catch (final InternalErrorException e) { | ||||||
|             return new Answer(cmd, false, e.toString()); |             return new Answer(cmd, false, e.toString()); | ||||||
|  |         } catch (final InvalidParameterValueException e) { | ||||||
|  |             return new Answer(cmd, false, e.toString()); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         return new Answer(cmd); |         return new Answer(cmd); | ||||||
| @ -1102,24 +1095,45 @@ public class KVMStorageProcessor implements StorageProcessor { | |||||||
|         final DiskTO disk = cmd.getDisk(); |         final DiskTO disk = cmd.getDisk(); | ||||||
|         final TemplateObjectTO isoTO = (TemplateObjectTO)disk.getData(); |         final TemplateObjectTO isoTO = (TemplateObjectTO)disk.getData(); | ||||||
|         final DataStoreTO store = isoTO.getDataStore(); |         final DataStoreTO store = isoTO.getDataStore(); | ||||||
|         if (!(store instanceof NfsTO)) { | 
 | ||||||
|             return new AttachAnswer("unsupported protocol"); |  | ||||||
|         } |  | ||||||
|         final NfsTO nfsStore = (NfsTO)store; |  | ||||||
|         try { |         try { | ||||||
|  |             String dataStoreUrl = getDataStoreUrlFromStore(store); | ||||||
|             final Connect conn = LibvirtConnection.getConnectionByVmName(cmd.getVmName()); |             final Connect conn = LibvirtConnection.getConnectionByVmName(cmd.getVmName()); | ||||||
|             attachOrDetachISO(conn, cmd.getVmName(), nfsStore.getUrl() + File.separator + isoTO.getPath(), false); |             attachOrDetachISO(conn, cmd.getVmName(), dataStoreUrl + File.separator + isoTO.getPath(), false); | ||||||
|         } catch (final LibvirtException e) { |         } catch (final LibvirtException e) { | ||||||
|             return new Answer(cmd, false, e.toString()); |             return new Answer(cmd, false, e.toString()); | ||||||
|         } catch (final URISyntaxException e) { |         } catch (final URISyntaxException e) { | ||||||
|             return new Answer(cmd, false, e.toString()); |             return new Answer(cmd, false, e.toString()); | ||||||
|         } catch (final InternalErrorException e) { |         } catch (final InternalErrorException e) { | ||||||
|             return new Answer(cmd, false, e.toString()); |             return new Answer(cmd, false, e.toString()); | ||||||
|  |         } catch (final InvalidParameterValueException e) { | ||||||
|  |             return new Answer(cmd, false, e.toString()); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         return new Answer(cmd); |         return new Answer(cmd); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Return data store URL from store | ||||||
|  |      */ | ||||||
|  |     private String getDataStoreUrlFromStore(DataStoreTO store) { | ||||||
|  |         if (!(store instanceof NfsTO) && (!(store instanceof PrimaryDataStoreTO) || | ||||||
|  |                 store instanceof PrimaryDataStoreTO && !((PrimaryDataStoreTO) store).getPoolType().equals(StoragePoolType.NetworkFilesystem))) { | ||||||
|  |             throw new InvalidParameterValueException("unsupported protocol"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (store instanceof NfsTO) { | ||||||
|  |             NfsTO nfsStore = (NfsTO)store; | ||||||
|  |             return nfsStore.getUrl(); | ||||||
|  |         } else if (store instanceof PrimaryDataStoreTO && ((PrimaryDataStoreTO) store).getPoolType().equals(StoragePoolType.NetworkFilesystem)) { | ||||||
|  |             //In order to support directly downloaded ISOs | ||||||
|  |             String psHost = ((PrimaryDataStoreTO) store).getHost(); | ||||||
|  |             String psPath = ((PrimaryDataStoreTO) store).getPath(); | ||||||
|  |             return "nfs://" + psHost + File.separator + psPath; | ||||||
|  |         } | ||||||
|  |         return store.getUrl(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     protected synchronized String attachOrDetachDevice(final Connect conn, final boolean attach, final String vmName, final String xml) throws LibvirtException, InternalErrorException { |     protected synchronized String attachOrDetachDevice(final Connect conn, final boolean attach, final String vmName, final String xml) throws LibvirtException, InternalErrorException { | ||||||
|         Domain dm = null; |         Domain dm = null; | ||||||
|         try { |         try { | ||||||
| @ -1582,36 +1596,57 @@ public class KVMStorageProcessor implements StorageProcessor { | |||||||
|         return new Answer(cmd, false, "not implememented yet"); |         return new Answer(cmd, false, "not implememented yet"); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Get direct template downloader from direct download command and destination pool | ||||||
|  |      */ | ||||||
|  |     private DirectTemplateDownloader getDirectTemplateDownloaderFromCommand(DirectDownloadCommand cmd, KVMStoragePool destPool) { | ||||||
|  |         if (cmd instanceof HttpDirectDownloadCommand) { | ||||||
|  |             return new HttpDirectTemplateDownloader(cmd.getUrl(), cmd.getTemplateId(), destPool.getLocalPath(), cmd.getChecksum(), cmd.getHeaders()); | ||||||
|  |         } else if (cmd instanceof HttpsDirectDownloadCommand) { | ||||||
|  |             return new HttpsDirectTemplateDownloader(cmd.getUrl(), cmd.getTemplateId(), destPool.getLocalPath(), cmd.getChecksum(), cmd.getHeaders()); | ||||||
|  |         } else if (cmd instanceof NfsDirectDownloadCommand) { | ||||||
|  |             return new NfsDirectTemplateDownloader(cmd.getUrl(), destPool.getLocalPath(), cmd.getTemplateId(), cmd.getChecksum()); | ||||||
|  |         } else if (cmd instanceof MetalinkDirectDownloadCommand) { | ||||||
|  |             return new MetalinkDirectTemplateDownloader(cmd.getUrl(), destPool.getLocalPath(), cmd.getTemplateId(), cmd.getChecksum(), cmd.getHeaders()); | ||||||
|  |         } else { | ||||||
|  |             throw new IllegalArgumentException("Unsupported protocol, please provide HTTP(S), NFS or a metalink"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     @Override |     @Override | ||||||
|     public Answer handleDownloadTemplateToPrimaryStorage(DirectDownloadCommand cmd) { |     public Answer handleDownloadTemplateToPrimaryStorage(DirectDownloadCommand cmd) { | ||||||
|         final PrimaryDataStoreTO pool = cmd.getDestPool(); |         final PrimaryDataStoreTO pool = cmd.getDestPool(); | ||||||
|         if (!pool.getPoolType().equals(StoragePoolType.NetworkFilesystem)) { |         if (!pool.getPoolType().equals(StoragePoolType.NetworkFilesystem)) { | ||||||
|             return new DirectDownloadAnswer(false, "Unsupported pool type " + pool.getPoolType().toString()); |             return new DirectDownloadAnswer(false, "Unsupported pool type " + pool.getPoolType().toString(), true); | ||||||
|         } |         } | ||||||
|         KVMStoragePool destPool = storagePoolMgr.getStoragePool(pool.getPoolType(), pool.getUuid()); |         KVMStoragePool destPool = storagePoolMgr.getStoragePool(pool.getPoolType(), pool.getUuid()); | ||||||
|         DirectTemplateDownloader downloader; |         DirectTemplateDownloader downloader; | ||||||
| 
 | 
 | ||||||
|         if (cmd instanceof HttpDirectDownloadCommand) { |         try { | ||||||
|             downloader = new HttpDirectTemplateDownloader(cmd.getUrl(), cmd.getTemplateId(), destPool.getLocalPath(), cmd.getChecksum(), ((HttpDirectDownloadCommand) cmd).getHeaders()); |             downloader = getDirectTemplateDownloaderFromCommand(cmd, destPool); | ||||||
|         } else if (cmd instanceof HttpsDirectDownloadCommand) { |         } catch (IllegalArgumentException e) { | ||||||
|             downloader = new HttpsDirectTemplateDownloader(cmd.getUrl(), cmd.getTemplateId(), destPool.getLocalPath(), cmd.getChecksum()); |             return new DirectDownloadAnswer(false, "Unable to create direct downloader: " + e.getMessage(), true); | ||||||
|         } else if (cmd instanceof NfsDirectDownloadCommand) { |  | ||||||
|             downloader = new NfsDirectTemplateDownloader(cmd.getUrl(), destPool.getLocalPath(), cmd.getTemplateId(), cmd.getChecksum()); |  | ||||||
|         } else if (cmd instanceof MetalinkDirectDownloadCommand) { |  | ||||||
|             downloader = new MetalinkDirectTemplateDownloader(cmd.getUrl(), destPool.getLocalPath(), cmd.getTemplateId(), cmd.getChecksum()); |  | ||||||
|         } else { |  | ||||||
|             return new DirectDownloadAnswer(false, "Unsupported protocol, please provide HTTP(S), NFS or a metalink"); |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         try { | ||||||
|  |             s_logger.info("Trying to download template"); | ||||||
|             if (!downloader.downloadTemplate()) { |             if (!downloader.downloadTemplate()) { | ||||||
|             return new DirectDownloadAnswer(false, "Could not download template " + cmd.getTemplateId() + " on " + destPool.getLocalPath()); |                 s_logger.warn("Couldn't download template"); | ||||||
|  |                 return new DirectDownloadAnswer(false, "Unable to download template", true); | ||||||
|             } |             } | ||||||
|             if (!downloader.validateChecksum()) { |             if (!downloader.validateChecksum()) { | ||||||
|             return new DirectDownloadAnswer(false, "Checksum validation failed for template " + cmd.getTemplateId()); |                 s_logger.warn("Couldn't validate template checksum"); | ||||||
|  |                 return new DirectDownloadAnswer(false, "Checksum validation failed", false); | ||||||
|             } |             } | ||||||
|             if (!downloader.extractAndInstallDownloadedTemplate()) { |             if (!downloader.extractAndInstallDownloadedTemplate()) { | ||||||
|             return new DirectDownloadAnswer(false, "Template downloaded but there was an error on installation"); |                 s_logger.warn("Couldn't extract and install template"); | ||||||
|  |                 return new DirectDownloadAnswer(false, "Extraction and installation failed", false); | ||||||
|             } |             } | ||||||
|  |         } catch (CloudRuntimeException e) { | ||||||
|  |             s_logger.warn("Error downloading template " + cmd.getTemplateId() + " due to: " + e.getMessage()); | ||||||
|  |             return new DirectDownloadAnswer(false, "Unable to download template: " + e.getMessage(), true); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         DirectTemplateInformation info = downloader.getTemplateInformation(); |         DirectTemplateInformation info = downloader.getTemplateInformation(); | ||||||
|         return new DirectDownloadAnswer(true, info.getSize(), info.getInstallPath()); |         return new DirectDownloadAnswer(true, info.getSize(), info.getInstallPath()); | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -313,6 +313,7 @@ import com.cloud.vm.snapshot.VMSnapshotManager; | |||||||
| import com.cloud.vm.snapshot.VMSnapshotVO; | import com.cloud.vm.snapshot.VMSnapshotVO; | ||||||
| import com.cloud.vm.snapshot.dao.VMSnapshotDao; | import com.cloud.vm.snapshot.dao.VMSnapshotDao; | ||||||
| import com.cloud.storage.snapshot.SnapshotApiService; | import com.cloud.storage.snapshot.SnapshotApiService; | ||||||
|  | import com.cloud.storage.VMTemplateStorageResourceAssoc; | ||||||
| 
 | 
 | ||||||
| public class UserVmManagerImpl extends ManagerBase implements UserVmManager, VirtualMachineGuru, UserVmService, Configurable { | public class UserVmManagerImpl extends ManagerBase implements UserVmManager, VirtualMachineGuru, UserVmService, Configurable { | ||||||
|     private static final Logger s_logger = Logger.getLogger(UserVmManagerImpl.class); |     private static final Logger s_logger = Logger.getLogger(UserVmManagerImpl.class); | ||||||
| @ -6100,10 +6101,8 @@ public class UserVmManagerImpl extends ManagerBase implements UserVmManager, Vir | |||||||
|                     throw ex; |                     throw ex; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             TemplateDataStoreVO tmplStore = _templateStoreDao.findByTemplateZoneReady(template.getId(), vm.getDataCenterId()); | 
 | ||||||
|             if (tmplStore == null) { |             checkRestoreVmFromTemplate(vm, template); | ||||||
|                 throw new InvalidParameterValueException("Cannot restore the vm as the template " + template.getUuid() + " isn't available in the zone"); |  | ||||||
|             } |  | ||||||
| 
 | 
 | ||||||
|             if (needRestart) { |             if (needRestart) { | ||||||
|                 try { |                 try { | ||||||
| @ -6217,6 +6216,27 @@ public class UserVmManagerImpl extends ManagerBase implements UserVmManager, Vir | |||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Perform basic checkings to make sure restore is possible. If not, #InvalidParameterValueException is thrown | ||||||
|  |      * @param vm vm | ||||||
|  |      * @param template template | ||||||
|  |      * @throws InvalidParameterValueException if restore is not possible | ||||||
|  |      */ | ||||||
|  |     private void checkRestoreVmFromTemplate(UserVmVO vm, VMTemplateVO template) { | ||||||
|  |         TemplateDataStoreVO tmplStore; | ||||||
|  |         if (!template.isDirectDownload()) { | ||||||
|  |             tmplStore = _templateStoreDao.findByTemplateZoneReady(template.getId(), vm.getDataCenterId()); | ||||||
|  |             if (tmplStore == null) { | ||||||
|  |                 throw new InvalidParameterValueException("Cannot restore the vm as the template " + template.getUuid() + " isn't available in the zone"); | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             tmplStore = _templateStoreDao.findByTemplate(template.getId(), DataStoreRole.Image); | ||||||
|  |             if (tmplStore == null || (tmplStore != null && !tmplStore.getDownloadState().equals(VMTemplateStorageResourceAssoc.Status.BYPASSED))) { | ||||||
|  |                 throw new InvalidParameterValueException("Cannot restore the vm as the bypassed template " + template.getUuid() + " isn't available in the zone"); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     private void handleManagedStorage(UserVmVO vm, VolumeVO root) { |     private void handleManagedStorage(UserVmVO vm, VolumeVO root) { | ||||||
|         if (Volume.State.Allocated.equals(root.getState())) { |         if (Volume.State.Allocated.equals(root.getState())) { | ||||||
|             return; |             return; | ||||||
|  | |||||||
| @ -18,13 +18,18 @@ | |||||||
| // | // | ||||||
| package org.apache.cloudstack.direct.download; | package org.apache.cloudstack.direct.download; | ||||||
| 
 | 
 | ||||||
|  | import static com.cloud.storage.Storage.ImageFormat; | ||||||
|  | 
 | ||||||
| import com.cloud.agent.AgentManager; | import com.cloud.agent.AgentManager; | ||||||
| import com.cloud.agent.api.Answer; | import com.cloud.agent.api.Answer; | ||||||
|  | import com.cloud.event.ActionEventUtils; | ||||||
|  | import com.cloud.event.EventTypes; | ||||||
|  | import com.cloud.event.EventVO; | ||||||
| import com.cloud.host.Host; | import com.cloud.host.Host; | ||||||
| import com.cloud.host.HostVO; | import com.cloud.host.HostVO; | ||||||
| import com.cloud.host.Status; | import com.cloud.host.Status; | ||||||
| import com.cloud.host.dao.HostDao; | import com.cloud.host.dao.HostDao; | ||||||
| import com.cloud.hypervisor.Hypervisor; | import com.cloud.hypervisor.Hypervisor.HypervisorType; | ||||||
| import com.cloud.storage.DataStoreRole; | import com.cloud.storage.DataStoreRole; | ||||||
| import com.cloud.storage.VMTemplateStoragePoolVO; | import com.cloud.storage.VMTemplateStoragePoolVO; | ||||||
| import com.cloud.storage.VMTemplateStorageResourceAssoc; | import com.cloud.storage.VMTemplateStorageResourceAssoc; | ||||||
| @ -40,6 +45,8 @@ import java.util.ArrayList; | |||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  | import java.util.Arrays; | ||||||
|  | import java.util.Collections; | ||||||
| import java.util.stream.Collectors; | import java.util.stream.Collectors; | ||||||
| import javax.inject.Inject; | import javax.inject.Inject; | ||||||
| 
 | 
 | ||||||
| @ -51,6 +58,7 @@ import org.apache.cloudstack.agent.directdownload.MetalinkDirectDownloadCommand; | |||||||
| import org.apache.cloudstack.agent.directdownload.NfsDirectDownloadCommand; | import org.apache.cloudstack.agent.directdownload.NfsDirectDownloadCommand; | ||||||
| import org.apache.cloudstack.agent.directdownload.HttpsDirectDownloadCommand; | import org.apache.cloudstack.agent.directdownload.HttpsDirectDownloadCommand; | ||||||
| import org.apache.cloudstack.agent.directdownload.SetupDirectDownloadCertificate; | import org.apache.cloudstack.agent.directdownload.SetupDirectDownloadCertificate; | ||||||
|  | import org.apache.cloudstack.context.CallContext; | ||||||
| import org.apache.cloudstack.engine.subsystem.api.storage.DataStore; | import org.apache.cloudstack.engine.subsystem.api.storage.DataStore; | ||||||
| import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreManager; | import org.apache.cloudstack.engine.subsystem.api.storage.DataStoreManager; | ||||||
| import org.apache.cloudstack.engine.subsystem.api.storage.ObjectInDataStoreStateMachine; | import org.apache.cloudstack.engine.subsystem.api.storage.ObjectInDataStoreStateMachine; | ||||||
| @ -58,6 +66,7 @@ import org.apache.cloudstack.engine.subsystem.api.storage.PrimaryDataStore; | |||||||
| import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; | import org.apache.cloudstack.storage.datastore.db.PrimaryDataStoreDao; | ||||||
| import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; | import org.apache.cloudstack.storage.datastore.db.StoragePoolVO; | ||||||
| import org.apache.cloudstack.storage.to.PrimaryDataStoreTO; | import org.apache.cloudstack.storage.to.PrimaryDataStoreTO; | ||||||
|  | import org.apache.commons.collections.CollectionUtils; | ||||||
| import org.apache.commons.collections.MapUtils; | import org.apache.commons.collections.MapUtils; | ||||||
| import org.apache.log4j.Logger; | import org.apache.log4j.Logger; | ||||||
| 
 | 
 | ||||||
| @ -137,6 +146,46 @@ public class DirectDownloadManagerImpl extends ManagerBase implements DirectDown | |||||||
|         return headers; |         return headers; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Get running host IDs within the same hypervisor, cluster and datacenter than hostId. ID hostId is not included on the returned list | ||||||
|  |      */ | ||||||
|  |     protected List<Long> getRunningHostIdsInTheSameCluster(Long clusterId, long dataCenterId, HypervisorType hypervisorType, long hostId) { | ||||||
|  |         List<Long> list = hostDao.listByDataCenterIdAndHypervisorType(dataCenterId, hypervisorType) | ||||||
|  |                 .stream() | ||||||
|  |                 .filter(x -> x.getHypervisorType().equals(hypervisorType) && x.getStatus().equals(Status.Up) && | ||||||
|  |                         x.getType().equals(Host.Type.Routing) && x.getClusterId().equals(clusterId) && | ||||||
|  |                         x.getId() != hostId) | ||||||
|  |                 .map(x -> x.getId()) | ||||||
|  |                 .collect(Collectors.toList()); | ||||||
|  |         Collections.shuffle(list); | ||||||
|  |         return list; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Create host IDs array having hostId as the first element | ||||||
|  |      */ | ||||||
|  |     protected Long[] createHostIdsList(List<Long> hostIds, long hostId) { | ||||||
|  |         if (CollectionUtils.isEmpty(hostIds)) { | ||||||
|  |             return Arrays.asList(hostId).toArray(new Long[1]); | ||||||
|  |         } | ||||||
|  |         Long[] ids = new Long[hostIds.size() + 1]; | ||||||
|  |         ids[0] = hostId; | ||||||
|  |         int i = 1; | ||||||
|  |         for (Long id : hostIds) { | ||||||
|  |             ids[i] = id; | ||||||
|  |             i++; | ||||||
|  |         } | ||||||
|  |         return ids; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Get hosts to retry download having hostId as the first element | ||||||
|  |      */ | ||||||
|  |     protected Long[] getHostsToRetryOn(Long clusterId, long dataCenterId, HypervisorType hypervisorType, long hostId) { | ||||||
|  |         List<Long> hostIds = getRunningHostIdsInTheSameCluster(clusterId, dataCenterId, hypervisorType, hostId); | ||||||
|  |         return createHostIdsList(hostIds, hostId); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     @Override |     @Override | ||||||
|     public void downloadTemplate(long templateId, long poolId, long hostId) { |     public void downloadTemplate(long templateId, long poolId, long hostId) { | ||||||
|         VMTemplateVO template = vmTemplateDao.findById(templateId); |         VMTemplateVO template = vmTemplateDao.findById(templateId); | ||||||
| @ -167,11 +216,8 @@ public class DirectDownloadManagerImpl extends ManagerBase implements DirectDown | |||||||
| 
 | 
 | ||||||
|         DownloadProtocol protocol = getProtocolFromUrl(url); |         DownloadProtocol protocol = getProtocolFromUrl(url); | ||||||
|         DirectDownloadCommand cmd = getDirectDownloadCommandFromProtocol(protocol, url, templateId, to, checksum, headers); |         DirectDownloadCommand cmd = getDirectDownloadCommandFromProtocol(protocol, url, templateId, to, checksum, headers); | ||||||
|         Answer answer = agentManager.easySend(hostId, cmd); | 
 | ||||||
|         if (answer == null || !answer.getResult()) { |         Answer answer = sendDirectDownloadCommand(cmd, template, poolId, host); | ||||||
|             throw new CloudRuntimeException("Host " + hostId + " could not download template " + |  | ||||||
|                     templateId + " on pool " + poolId); |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         VMTemplateStoragePoolVO sPoolRef = vmTemplatePoolDao.findByPoolTemplate(poolId, templateId); |         VMTemplateStoragePoolVO sPoolRef = vmTemplatePoolDao.findByPoolTemplate(poolId, templateId); | ||||||
|         if (sPoolRef == null) { |         if (sPoolRef == null) { | ||||||
| @ -190,13 +236,56 @@ public class DirectDownloadManagerImpl extends ManagerBase implements DirectDown | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Send direct download command for downloading template with ID templateId on storage pool with ID poolId.<br/> | ||||||
|  |      * At first, cmd is sent to host, in case of failure it will retry on other hosts before failing | ||||||
|  |      * @param cmd direct download command | ||||||
|  |      * @param template template | ||||||
|  |      * @param poolId pool id | ||||||
|  |      * @param host first host to which send the command | ||||||
|  |      * @return download answer from any host which could handle cmd | ||||||
|  |      */ | ||||||
|  |     private Answer sendDirectDownloadCommand(DirectDownloadCommand cmd, VMTemplateVO template, long poolId, HostVO host) { | ||||||
|  |         boolean downloaded = false; | ||||||
|  |         int retry = 3; | ||||||
|  |         Long[] hostsToRetry = getHostsToRetryOn(host.getClusterId(), host.getDataCenterId(), host.getHypervisorType(), host.getId()); | ||||||
|  |         int hostIndex = 0; | ||||||
|  |         Answer answer = null; | ||||||
|  |         Long hostToSendDownloadCmd = hostsToRetry[hostIndex]; | ||||||
|  |         boolean continueRetrying = true; | ||||||
|  |         while (!downloaded && retry > 0 && continueRetrying) { | ||||||
|  |             s_logger.debug("Sending Direct download command to host " + hostToSendDownloadCmd); | ||||||
|  |             answer = agentManager.easySend(hostToSendDownloadCmd, cmd); | ||||||
|  |             if (answer != null) { | ||||||
|  |                 DirectDownloadAnswer ans = (DirectDownloadAnswer)answer; | ||||||
|  |                 downloaded = answer.getResult(); | ||||||
|  |                 continueRetrying = ans.isRetryOnOtherHosts(); | ||||||
|  |             } | ||||||
|  |             hostToSendDownloadCmd = hostsToRetry[(hostIndex + 1) % hostsToRetry.length]; | ||||||
|  |             retry --; | ||||||
|  |         } | ||||||
|  |         if (!downloaded) { | ||||||
|  |             logUsageEvent(template, poolId); | ||||||
|  |             throw new CloudRuntimeException("Template " + template.getId() + " could not be downloaded on pool " + poolId + ", failing after trying on several hosts"); | ||||||
|  |         } | ||||||
|  |         return answer; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Log and persist event for direct download failure | ||||||
|  |      */ | ||||||
|  |     private void logUsageEvent(VMTemplateVO template, long poolId) { | ||||||
|  |         String event = EventTypes.EVENT_TEMPLATE_DIRECT_DOWNLOAD_FAILURE; | ||||||
|  |         if (template.getFormat() == ImageFormat.ISO) { | ||||||
|  |             event = EventTypes.EVENT_ISO_DIRECT_DOWNLOAD_FAILURE; | ||||||
|  |         } | ||||||
|  |         String description = "Direct Download for template Id: " + template.getId() + " on pool Id: " + poolId + " failed"; | ||||||
|  |         s_logger.error(description); | ||||||
|  |         ActionEventUtils.onCompletedActionEvent(CallContext.current().getCallingUserId(), template.getAccountId(), EventVO.LEVEL_INFO, event, description, 0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * Return DirectDownloadCommand according to the protocol |      * Return DirectDownloadCommand according to the protocol | ||||||
|      * @param protocol |  | ||||||
|      * @param url |  | ||||||
|      * @param templateId |  | ||||||
|      * @param destPool |  | ||||||
|      * @return |  | ||||||
|      */ |      */ | ||||||
|     private DirectDownloadCommand getDirectDownloadCommandFromProtocol(DownloadProtocol protocol, String url, Long templateId, PrimaryDataStoreTO destPool, |     private DirectDownloadCommand getDirectDownloadCommandFromProtocol(DownloadProtocol protocol, String url, Long templateId, PrimaryDataStoreTO destPool, | ||||||
|                                                                        String checksum, Map<String, String> httpHeaders) { |                                                                        String checksum, Map<String, String> httpHeaders) { | ||||||
| @ -205,26 +294,36 @@ public class DirectDownloadManagerImpl extends ManagerBase implements DirectDown | |||||||
|         } else if (protocol.equals(DownloadProtocol.HTTPS)) { |         } else if (protocol.equals(DownloadProtocol.HTTPS)) { | ||||||
|             return new HttpsDirectDownloadCommand(url, templateId, destPool, checksum, httpHeaders); |             return new HttpsDirectDownloadCommand(url, templateId, destPool, checksum, httpHeaders); | ||||||
|         } else if (protocol.equals(DownloadProtocol.NFS)) { |         } else if (protocol.equals(DownloadProtocol.NFS)) { | ||||||
|             return new NfsDirectDownloadCommand(url, templateId, destPool, checksum); |             return new NfsDirectDownloadCommand(url, templateId, destPool, checksum, httpHeaders); | ||||||
|         } else if (protocol.equals(DownloadProtocol.METALINK)) { |         } else if (protocol.equals(DownloadProtocol.METALINK)) { | ||||||
|             return new MetalinkDirectDownloadCommand(url, templateId, destPool, checksum); |             return new MetalinkDirectDownloadCommand(url, templateId, destPool, checksum, httpHeaders); | ||||||
|         } else { |         } else { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     @Override |     /** | ||||||
|     public boolean uploadCertificateToHosts(String certificateCer, String certificateName) { |      * Return the list of running hosts to which upload certificates for Direct Download | ||||||
|         List<HostVO> hosts = hostDao.listAllHostsByType(Host.Type.Routing) |      */ | ||||||
|  |     private List<HostVO> getRunningHostsToUploadCertificate(HypervisorType hypervisorType) { | ||||||
|  |         return hostDao.listAllHostsByType(Host.Type.Routing) | ||||||
|                 .stream() |                 .stream() | ||||||
|                 .filter(x -> x.getStatus().equals(Status.Up) && |                 .filter(x -> x.getStatus().equals(Status.Up) && | ||||||
|                             x.getHypervisorType().equals(Hypervisor.HypervisorType.KVM)) |                         x.getHypervisorType().equals(hypervisorType)) | ||||||
|                 .collect(Collectors.toList()); |                 .collect(Collectors.toList()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public boolean uploadCertificateToHosts(String certificateCer, String certificateName, String hypervisor) { | ||||||
|  |         HypervisorType hypervisorType = HypervisorType.getType(hypervisor); | ||||||
|  |         List<HostVO> hosts = getRunningHostsToUploadCertificate(hypervisorType); | ||||||
|  |         if (CollectionUtils.isNotEmpty(hosts)) { | ||||||
|             for (HostVO host : hosts) { |             for (HostVO host : hosts) { | ||||||
|                 if (!uploadCertificate(certificateCer, certificateName, host.getId())) { |                 if (!uploadCertificate(certificateCer, certificateName, host.getId())) { | ||||||
|                     throw new CloudRuntimeException("Uploading certificate " + certificateName + " failed on host: " + host.getId()); |                     throw new CloudRuntimeException("Uploading certificate " + certificateName + " failed on host: " + host.getId()); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |         } | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -36,6 +36,7 @@ import java.util.ListIterator; | |||||||
| import java.util.StringTokenizer; | import java.util.StringTokenizer; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
|  | import java.util.Comparator; | ||||||
| 
 | 
 | ||||||
| import javax.xml.parsers.DocumentBuilder; | import javax.xml.parsers.DocumentBuilder; | ||||||
| import javax.xml.parsers.DocumentBuilderFactory; | import javax.xml.parsers.DocumentBuilderFactory; | ||||||
| @ -63,6 +64,7 @@ import org.w3c.dom.Document; | |||||||
| import org.w3c.dom.Element; | import org.w3c.dom.Element; | ||||||
| import org.w3c.dom.Node; | import org.w3c.dom.Node; | ||||||
| import org.w3c.dom.NodeList; | import org.w3c.dom.NodeList; | ||||||
|  | import org.w3c.dom.NamedNodeMap; | ||||||
| 
 | 
 | ||||||
| public class UriUtils { | public class UriUtils { | ||||||
| 
 | 
 | ||||||
| @ -297,6 +299,74 @@ public class UriUtils { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Add element to priority list examining node attributes: priority (for urls) and type (for checksums) | ||||||
|  |      */ | ||||||
|  |     protected static void addPriorityListElementExaminingNode(String tagName, Node node, List<Pair<String, Integer>> priorityList) { | ||||||
|  |         Integer priority = Integer.MAX_VALUE; | ||||||
|  |         String first = node.getTextContent(); | ||||||
|  |         if (node.hasAttributes()) { | ||||||
|  |             NamedNodeMap attributes = node.getAttributes(); | ||||||
|  |             for (int k=0; k<attributes.getLength(); k++) { | ||||||
|  |                 Node attr = attributes.item(k); | ||||||
|  |                 if (tagName.equals("url") && attr.getNodeName().equals("priority")) { | ||||||
|  |                     String prio = attr.getNodeValue().replace("\"", ""); | ||||||
|  |                     priority = Integer.parseInt(prio); | ||||||
|  |                     break; | ||||||
|  |                 } else if (tagName.equals("hash") && attr.getNodeName().equals("type")) { | ||||||
|  |                     first = "{" + attr.getNodeValue() + "}" + first; | ||||||
|  |                     break; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         priorityList.add(new Pair<>(first, priority)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Return the list of first elements on the list of pairs | ||||||
|  |      */ | ||||||
|  |     protected static List<String> getListOfFirstElements(List<Pair<String, Integer>> priorityList) { | ||||||
|  |         List<String> values = new ArrayList<>(); | ||||||
|  |         for (Pair<String, Integer> pair : priorityList) { | ||||||
|  |             values.add(pair.first()); | ||||||
|  |         } | ||||||
|  |         return values; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Return HttpClient with connection timeout | ||||||
|  |      */ | ||||||
|  |     private static HttpClient getHttpClient() { | ||||||
|  |         MultiThreadedHttpConnectionManager s_httpClientManager = new MultiThreadedHttpConnectionManager(); | ||||||
|  |         s_httpClientManager.getParams().setConnectionTimeout(5000); | ||||||
|  |         return new HttpClient(s_httpClientManager); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static List<String> getMetalinkChecksums(String url) { | ||||||
|  |         HttpClient httpClient = getHttpClient(); | ||||||
|  |         GetMethod getMethod = new GetMethod(url); | ||||||
|  |         try { | ||||||
|  |             if (httpClient.executeMethod(getMethod) == HttpStatus.SC_OK) { | ||||||
|  |                 InputStream is = getMethod.getResponseBodyAsStream(); | ||||||
|  |                 Map<String, List<String>> checksums = getMultipleValuesFromXML(is, new String[] {"hash"}); | ||||||
|  |                 if (checksums.containsKey("hash")) { | ||||||
|  |                     List<String> listChksum = new ArrayList<>(); | ||||||
|  |                     for (String chk : checksums.get("hash")) { | ||||||
|  |                         listChksum.add(chk.replaceAll("\n", "").replaceAll(" ", "").trim()); | ||||||
|  |                     } | ||||||
|  |                     return listChksum; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } catch (IOException e) { | ||||||
|  |             e.printStackTrace(); | ||||||
|  |         } finally { | ||||||
|  |             getMethod.releaseConnection(); | ||||||
|  |         } | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |     /** | ||||||
|  |      * Retrieve values from XML documents ordered by ascending priority for each tag name | ||||||
|  |      */ | ||||||
|     protected static Map<String, List<String>> getMultipleValuesFromXML(InputStream is, String[] tagNames) { |     protected static Map<String, List<String>> getMultipleValuesFromXML(InputStream is, String[] tagNames) { | ||||||
|         Map<String, List<String>> returnValues = new HashMap<String, List<String>>(); |         Map<String, List<String>> returnValues = new HashMap<String, List<String>>(); | ||||||
|         try { |         try { | ||||||
| @ -307,14 +377,15 @@ public class UriUtils { | |||||||
|             for (int i = 0; i < tagNames.length; i++) { |             for (int i = 0; i < tagNames.length; i++) { | ||||||
|                 NodeList targetNodes = rootElement.getElementsByTagName(tagNames[i]); |                 NodeList targetNodes = rootElement.getElementsByTagName(tagNames[i]); | ||||||
|                 if (targetNodes.getLength() <= 0) { |                 if (targetNodes.getLength() <= 0) { | ||||||
|                     s_logger.error("no " + tagNames[i] + " tag in XML response...returning null"); |                     s_logger.error("no " + tagNames[i] + " tag in XML response..."); | ||||||
|                 } else { |                 } else { | ||||||
|                     List<String> valueList = new ArrayList<String>(); |                     List<Pair<String, Integer>> priorityList = new ArrayList<>(); | ||||||
|                     for (int j = 0; j < targetNodes.getLength(); j++) { |                     for (int j = 0; j < targetNodes.getLength(); j++) { | ||||||
|                         Node node = targetNodes.item(j); |                         Node node = targetNodes.item(j); | ||||||
|                         valueList.add(node.getTextContent()); |                         addPriorityListElementExaminingNode(tagNames[i], node, priorityList); | ||||||
|                     } |                     } | ||||||
|                     returnValues.put(tagNames[i], valueList); |                     priorityList.sort(Comparator.comparing(x -> x.second())); | ||||||
|  |                     returnValues.put(tagNames[i], getListOfFirstElements(priorityList)); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } catch (Exception ex) { |         } catch (Exception ex) { | ||||||
| @ -329,7 +400,7 @@ public class UriUtils { | |||||||
|      * @return true if at least one existent URL defined on metalink, false if not |      * @return true if at least one existent URL defined on metalink, false if not | ||||||
|      */ |      */ | ||||||
|     protected static boolean checkUrlExistenceMetalink(String url) { |     protected static boolean checkUrlExistenceMetalink(String url) { | ||||||
|         HttpClient httpClient = new HttpClient(new MultiThreadedHttpConnectionManager()); |         HttpClient httpClient = getHttpClient(); | ||||||
|         GetMethod getMethod = new GetMethod(url); |         GetMethod getMethod = new GetMethod(url); | ||||||
|         try { |         try { | ||||||
|             if (httpClient.executeMethod(getMethod) == HttpStatus.SC_OK) { |             if (httpClient.executeMethod(getMethod) == HttpStatus.SC_OK) { | ||||||
| @ -356,23 +427,30 @@ public class UriUtils { | |||||||
|             } |             } | ||||||
|         } catch (IOException e) { |         } catch (IOException e) { | ||||||
|             s_logger.warn(e.getMessage()); |             s_logger.warn(e.getMessage()); | ||||||
|  |         } finally { | ||||||
|  |             getMethod.releaseConnection(); | ||||||
|         } |         } | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Get list of urls on metalink |      * Get list of urls on metalink ordered by ascending priority (for those which priority tag is not defined, highest priority value is assumed) | ||||||
|      * @param metalinkUrl |  | ||||||
|      * @return |  | ||||||
|      */ |      */ | ||||||
|     public static List<String> getMetalinkUrls(String metalinkUrl) { |     public static List<String> getMetalinkUrls(String metalinkUrl) { | ||||||
|         HttpClient httpClient = new HttpClient(new MultiThreadedHttpConnectionManager()); |         HttpClient httpClient = getHttpClient(); | ||||||
|         GetMethod getMethod = new GetMethod(metalinkUrl); |         GetMethod getMethod = new GetMethod(metalinkUrl); | ||||||
|         List<String> urls = new ArrayList<>(); |         List<String> urls = new ArrayList<>(); | ||||||
|         try ( |         int status; | ||||||
|                 InputStream is = getMethod.getResponseBodyAsStream() |         try { | ||||||
|         ) { |             status = httpClient.executeMethod(getMethod); | ||||||
|             if (httpClient.executeMethod(getMethod) == HttpStatus.SC_OK) { |         } catch (IOException e) { | ||||||
|  |             s_logger.error("Error retrieving urls form metalink: " + metalinkUrl); | ||||||
|  |             getMethod.releaseConnection(); | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         try { | ||||||
|  |             InputStream is = getMethod.getResponseBodyAsStream(); | ||||||
|  |             if (status == HttpStatus.SC_OK) { | ||||||
|                 Map<String, List<String>> metalinkUrlsMap = getMultipleValuesFromXML(is, new String[] {"url"}); |                 Map<String, List<String>> metalinkUrlsMap = getMultipleValuesFromXML(is, new String[] {"url"}); | ||||||
|                 if (metalinkUrlsMap.containsKey("url")) { |                 if (metalinkUrlsMap.containsKey("url")) { | ||||||
|                     List<String> metalinkUrls = metalinkUrlsMap.get("url"); |                     List<String> metalinkUrls = metalinkUrlsMap.get("url"); | ||||||
| @ -381,6 +459,8 @@ public class UriUtils { | |||||||
|             } |             } | ||||||
|         } catch (IOException e) { |         } catch (IOException e) { | ||||||
|             s_logger.warn(e.getMessage()); |             s_logger.warn(e.getMessage()); | ||||||
|  |         } finally { | ||||||
|  |             getMethod.releaseConnection(); | ||||||
|         } |         } | ||||||
|         return urls; |         return urls; | ||||||
|     } |     } | ||||||
| @ -388,7 +468,7 @@ public class UriUtils { | |||||||
|     // use http HEAD method to validate url |     // use http HEAD method to validate url | ||||||
|     public static void checkUrlExistence(String url) { |     public static void checkUrlExistence(String url) { | ||||||
|         if (url.toLowerCase().startsWith("http") || url.toLowerCase().startsWith("https")) { |         if (url.toLowerCase().startsWith("http") || url.toLowerCase().startsWith("https")) { | ||||||
|             HttpClient httpClient = new HttpClient(new MultiThreadedHttpConnectionManager()); |             HttpClient httpClient = getHttpClient(); | ||||||
|             HeadMethod httphead = new HeadMethod(url); |             HeadMethod httphead = new HeadMethod(url); | ||||||
|             try { |             try { | ||||||
|                 if (httpClient.executeMethod(httphead) != HttpStatus.SC_OK) { |                 if (httpClient.executeMethod(httphead) != HttpStatus.SC_OK) { | ||||||
| @ -398,9 +478,11 @@ public class UriUtils { | |||||||
|                     throw new IllegalArgumentException("Invalid URLs defined on metalink: " + url); |                     throw new IllegalArgumentException("Invalid URLs defined on metalink: " + url); | ||||||
|                 } |                 } | ||||||
|             } catch (HttpException hte) { |             } catch (HttpException hte) { | ||||||
|                 throw new IllegalArgumentException("Cannot reach URL: " + url); |                 throw new IllegalArgumentException("Cannot reach URL: " + url + " due to: " + hte.getMessage()); | ||||||
|             } catch (IOException ioe) { |             } catch (IOException ioe) { | ||||||
|                 throw new IllegalArgumentException("Cannot reach URL: " + url); |                 throw new IllegalArgumentException("Cannot reach URL: " + url + " due to: " + ioe.getMessage()); | ||||||
|  |             } finally { | ||||||
|  |                 httphead.releaseConnection(); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @ -444,7 +526,8 @@ public class UriUtils { | |||||||
|                 && (!uripath.toLowerCase().endsWith("ova") |                 && (!uripath.toLowerCase().endsWith("ova") | ||||||
|                         && !uripath.toLowerCase().endsWith("ova.zip") |                         && !uripath.toLowerCase().endsWith("ova.zip") | ||||||
|                         && !uripath.toLowerCase().endsWith("ova.bz2") |                         && !uripath.toLowerCase().endsWith("ova.bz2") | ||||||
|                         && !uripath.toLowerCase().endsWith("ova.gz"))) |                         && !uripath.toLowerCase().endsWith("ova.gz") | ||||||
|  |                         && !uripath.toLowerCase().endsWith("metalink"))) | ||||||
|                 || (format.equalsIgnoreCase("tar") |                 || (format.equalsIgnoreCase("tar") | ||||||
|                 && (!uripath.toLowerCase().endsWith("tar") |                 && (!uripath.toLowerCase().endsWith("tar") | ||||||
|                         && !uripath.toLowerCase().endsWith("tar.zip") |                         && !uripath.toLowerCase().endsWith("tar.zip") | ||||||
| @ -468,7 +551,8 @@ public class UriUtils { | |||||||
|                 && (!uripath.toLowerCase().endsWith("iso") |                 && (!uripath.toLowerCase().endsWith("iso") | ||||||
|                         && !uripath.toLowerCase().endsWith("iso.zip") |                         && !uripath.toLowerCase().endsWith("iso.zip") | ||||||
|                         && !uripath.toLowerCase().endsWith("iso.bz2") |                         && !uripath.toLowerCase().endsWith("iso.bz2") | ||||||
|                         && !uripath.toLowerCase().endsWith("iso.gz")))) { |                         && !uripath.toLowerCase().endsWith("iso.gz")) | ||||||
|  |                         && !uripath.toLowerCase().endsWith("metalink"))) { | ||||||
|             throw new IllegalArgumentException("Please specify a valid URL. URL:" + uripath + " is an invalid for the format " + format.toLowerCase()); |             throw new IllegalArgumentException("Please specify a valid URL. URL:" + uripath + " is an invalid for the format " + format.toLowerCase()); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user