org.jmeterplugins.repository.JARSourceHTTP Maven / Gradle / Ivy
package org.jmeterplugins.repository;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.InterruptedIOException;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.NetworkInterface;
import java.net.SocketException;
import java.net.URI;
import java.net.URLEncoder;
import java.net.UnknownHostException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Date;
import java.util.Enumeration;
import java.util.List;
import java.util.Locale;
import net.sf.json.JSON;
import net.sf.json.JSONArray;
import net.sf.json.JSONObject;
import net.sf.json.JSONSerializer;
import net.sf.json.JsonConfig;
import org.apache.commons.codec.digest.DigestUtils;
import org.apache.commons.io.FilenameUtils;
import org.apache.commons.io.output.ByteArrayOutputStream;
import org.apache.http.Header;
import org.apache.http.HeaderElement;
import org.apache.http.HttpEntity;
import org.apache.http.HttpHost;
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.auth.AuthScope;
import org.apache.http.auth.NTCredentials;
import org.apache.http.client.ServiceUnavailableRetryStrategy;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.methods.HttpRequestBase;
import org.apache.http.client.methods.HttpUriRequest;
import org.apache.http.conn.params.ConnRoutePNames;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.AbstractHttpClient;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.impl.client.DefaultHttpRequestRetryHandler;
import org.apache.http.params.CoreConnectionPNames;
import org.apache.http.params.HttpParams;
import org.apache.http.protocol.BasicHttpContext;
import org.apache.http.protocol.ExecutionContext;
import org.apache.http.protocol.HttpContext;
import org.apache.http.util.EntityUtils;
import org.apache.jmeter.JMeter;
import org.apache.jmeter.gui.GuiPackage;
import org.apache.jmeter.util.JMeterUtils;
import org.apache.jorphan.logging.LoggingManager;
import org.apache.log.Logger;
import org.jmeterplugins.repository.cache.PluginsRepo;
import org.jmeterplugins.repository.http.HttpRetryStrategy;
import java.util.zip.GZIPInputStream;
public class JARSourceHTTP extends JARSource {
private static final Logger log = LoggingManager.getLoggerForClass();
private static final int RETRY_COUNT = 1;
public static final SimpleDateFormat dateFormat = new SimpleDateFormat("EEE, dd MMM yyyy HH:mm:ss z", Locale.US);
private static final long CACHE_MAX_AGE = 60 * 60 * 1000;
private final String[] addresses;
protected AbstractHttpClient httpClient;
private int timeout = Integer.parseInt(JMeterUtils.getPropDefault("jpgc.repo.timeout", "30000"));
private final ServiceUnavailableRetryStrategy retryStrategy = new HttpRetryStrategy(RETRY_COUNT, 5000);
private File cacheDir;
public JARSourceHTTP(String jmProp) {
this.addresses = jmProp.split("[;]");
httpClient = getHTTPClient();
cacheDir = getCacheDir();
}
@Override
public Object clone() throws CloneNotSupportedException {
JARSourceHTTP clone = (JARSourceHTTP) super.clone();
log.debug("Clone HTTP client");
clone.httpClient = getHTTPClient();
return clone;
}
private File getCacheDir() {
String tmpDirPath = System.getProperty("java.io.tmpdir");
File cacheFolder = new File(tmpDirPath, "pmgr_cache");
if (!cacheFolder.isDirectory()) {
cacheFolder.delete();
}
if (!cacheFolder.exists()) {
cacheFolder.mkdirs();
}
return cacheFolder;
}
private AbstractHttpClient getHTTPClient() {
AbstractHttpClient client = new DefaultHttpClient();
String proxyHost = System.getProperty("https.proxyHost", "");
if (!proxyHost.isEmpty()) {
int proxyPort = Integer.parseInt(System.getProperty("https.proxyPort", "-1"));
log.info("Using proxy " + proxyHost + ":" + proxyPort);
HttpParams params = client.getParams();
HttpHost proxy = new HttpHost(proxyHost, proxyPort);
params.setParameter(ConnRoutePNames.DEFAULT_PROXY, proxy);
String proxyUser = System.getProperty(JMeter.HTTP_PROXY_USER, org.apache.jmeter.util.JMeterUtils.getProperty(JMeter.HTTP_PROXY_USER));
if (proxyUser != null) {
log.info("Using authenticated proxy with username: " + proxyUser);
String proxyPass = System.getProperty(JMeter.HTTP_PROXY_PASS, JMeterUtils.getProperty(JMeter.HTTP_PROXY_PASS));
String localHost;
try {
localHost = InetAddress.getLocalHost().getCanonicalHostName();
} catch (Throwable e) {
log.error("Failed to get local host name, defaulting to 'localhost'", e);
localHost = "localhost";
}
AuthScope authscope = new AuthScope(proxyHost, proxyPort);
String proxyDomain = JMeterUtils.getPropDefault("http.proxyDomain", "");
NTCredentials credentials = new NTCredentials(proxyUser, proxyPass, localHost, proxyDomain);
client.getCredentialsProvider().setCredentials(authscope, credentials);
}
}
client.setHttpRequestRetryHandler(new DefaultHttpRequestRetryHandler(RETRY_COUNT, true));
return client;
}
protected JSON getJSON(String uri) throws IOException {
log.info("Requesting " + uri);
HttpRequestBase get = new HttpGet(uri);
HttpParams requestParams = get.getParams();
get.setHeader("Accept-Encoding", "gzip");
requestParams.setIntParameter(CoreConnectionPNames.SO_TIMEOUT, timeout);
requestParams.setIntParameter(CoreConnectionPNames.CONNECTION_TIMEOUT, timeout);
HttpResponse result = execute(get);
ByteArrayOutputStream bos = new ByteArrayOutputStream();
HttpEntity entity = result.getEntity();
try {
entity.writeTo(bos);
byte[] bytes = bos.toByteArray();
if (bytes == null) {
bytes = "null".getBytes();
}
String response = isGZIPResponse(result) ? convertGZIPToString(bytes) : new String(bytes);
int statusCode = result.getStatusLine().getStatusCode();
if (statusCode >= 300) {
log.warn("Response with code " + result + ": " + response);
throw new IOException("Repository responded with wrong status code: " + statusCode);
} else {
log.debug("Response with code " + result + ": " + response);
}
cacheRepo(response, result, uri);
return JSONSerializer.toJSON(response, new JsonConfig());
} finally {
get.abort();
try {
entity.getContent().close();
} catch (IOException | IllegalStateException e) {
log.warn("Exception in finalizing request", e);
}
}
}
private PluginsRepo getRepoCache(String uri) {
File file = generateCacheFile(uri);
if (!file.exists()) {
return null;
}
return PluginsRepo.fromFile(file);
}
private void cacheRepo(String repoJSON, HttpResponse response, String uri) {
// default cache expire is 1 hour since now
long maxAge = CACHE_MAX_AGE;
long date = System.currentTimeMillis();
long lastModified = System.currentTimeMillis();
Header[] allHeaders = response.getAllHeaders();
for (Header header : allHeaders) {
if ("date".equals(header.getName().toLowerCase())) {
date = parseDateHeader(header);
} else if ("cache-control".equals(header.getName().toLowerCase())) {
maxAge = parseCacheControlHeader(header);
} else if ("last-modified".equals(header.getName().toLowerCase())) {
lastModified = parseDateHeader(header);
}
}
long expirationTime = date + maxAge;
PluginsRepo repo = new PluginsRepo(repoJSON, expirationTime, lastModified);
repo.saveToFile(generateCacheFile(uri));
}
private File generateCacheFile(String uri) {
return new File(cacheDir, generateFileName(uri));
}
private String generateFileName(String uri) {
return DigestUtils.md5Hex(System.getProperty("user.name") + uri);
}
private long parseCacheControlHeader(Header header) {
HeaderElement[] elements = header.getElements();
for (HeaderElement el : elements) {
if ("max-age".equals(el.getName().toLowerCase())) {
String value = el.getValue();
try {
int i = Integer.parseInt(value);
return i * 1000; // because max-age store in seconds
} catch (NumberFormatException e) {
log.warn("Cannot parse 'max-age' value", e);
return CACHE_MAX_AGE;
}
}
}
return CACHE_MAX_AGE;
}
private long parseDateHeader(Header header) {
try {
Date d = dateFormat.parse(header.getValue());
return d.getTime();
} catch (ParseException e) {
log.warn("Cannot parse date header", e);
return System.currentTimeMillis();
}
}
private boolean isGZIPResponse(HttpResponse result) {
Header encoding = result.getFirstHeader("Content-Encoding");
return encoding != null && "gzip".equals(encoding.getValue().toLowerCase());
}
private String convertGZIPToString(byte[] bytes) throws IOException {
GZIPInputStream gzipInputStream = new GZIPInputStream(new ByteArrayInputStream(bytes));
InputStreamReader reader = new InputStreamReader(gzipInputStream);
BufferedReader in = new BufferedReader(reader);
final StringBuilder buffer = new StringBuilder();
String line;
while ((line = in.readLine()) != null) {
buffer.append(line);
}
return buffer.toString();
}
protected JSONArray getRepositories(String path) throws IOException {
final List repositories = new ArrayList<>(addresses.length);
for (String address : addresses) {
PluginsRepo repo = getRepoCache(address + path);
if (repo != null && repo.isActual()) {
log.info("Found cached repo");
repositories.add(JSONSerializer.toJSON(repo.getRepoJSON(), new JsonConfig()));
} else {
repositories.add(getJSON(address + path));
}
}
final JSONArray result = new JSONArray();
final List pluginsIDs = new ArrayList<>();
for (JSON json : repositories) {
if (!(json instanceof JSONArray)) {
throw new RuntimeException("Result is not array");
}
for (Object elm : (JSONArray) json) {
// resolve plugin-id conflicts
String id = ((JSONObject) elm).getString("id");
if (!pluginsIDs.contains(id)) {
pluginsIDs.add(id);
result.add(elm);
} else {
log.info("Plugin " + id + " will be skipped, because it is duplicated.");
}
}
}
return result;
}
@Override
public JSON getRepo() throws IOException {
return getRepositories("?installID=" + getInstallID());
}
/**
* This function makes sure anonymous identifier sent
*
* @return unique ID for installation
*/
public String getInstallID() {
StringBuilder str = new StringBuilder();
str.append(getClass().getProtectionDomain().getCodeSource().getLocation().getFile());
try {
str.append("\t").append(InetAddress.getLocalHost().getHostName());
} catch (UnknownHostException e) {
log.warn("Cannot get local host name", e);
}
try {
Enumeration ifs = NetworkInterface.getNetworkInterfaces();
for (NetworkInterface netint : Collections.list(ifs)) {
str.append("\t").append(Arrays.toString(netint.getHardwareAddress()));
}
} catch (SocketException e) {
log.warn("Failed to get network addresses", e);
}
return getPlatformName() + '-' + DigestUtils.md5Hex(str.toString()) + '-' + getGuiMode();
}
private String getGuiMode() {
return (GuiPackage.getInstance() == null) ? "nongui" : "gui";
}
protected String getPlatformName() {
if (containsEnvironment("JENKINS_HOME")) {
return "jenkins";
} else if (containsEnvironment("TRAVIS")) {
return "travis";
} else if (containsEnvironmentPrefix("bamboo")) {
return "bamboo";
} else if (containsEnvironment("TEAMCITY_VERSION")) {
return "teamcity";
} else if (containsEnvironment("DOCKER_HOST")) {
return "docker";
} else if (containsEnvironmentPrefix("AWS_")) {
return "amazon";
} else if (containsEnvironment("GOOGLE_APPLICATION_CREDENTIALS") || containsEnvironment("CLOUDSDK_CONFIG")) {
return "google_cloud";
} else if (containsEnvironment("WEBJOBS_NAME")) {
return "azure";
} else {
return getOSName();
}
}
private boolean containsEnvironment(String key) {
return System.getenv().containsKey(key);
}
private boolean containsEnvironmentPrefix(String prefix) {
for (String key : System.getenv().keySet()) {
if (key.toLowerCase().startsWith(prefix.toLowerCase())) {
return true;
}
}
return false;
}
private String getOSName() {
return System.getProperty("os.name").toLowerCase().replace(' ', '_');
}
@Override
public void setTimeout(int timeout) {
this.timeout = timeout;
}
@Override
public DownloadResult getJAR(final String id, String location, final GenericCallback callback) throws IOException {
URI url = URI.create(location);
log.info("Downloading: " + url);
callback.notify("Downloading " + id + "...");
HttpGet httpget = new HttpGet(url);
HttpContext context = new BasicHttpContext();
HttpResponse response = execute(httpget, context);
if (response.getStatusLine().getStatusCode() != HttpStatus.SC_OK) {
log.error("Error downloading url:" + url + " got response code:" + response.getStatusLine().getStatusCode());
EntityUtils.consumeQuietly(response.getEntity());
throw new IOException(response.getStatusLine().toString());
}
HttpEntity entity = response.getEntity();
File tempFile = File.createTempFile(id, ".jar");
final long size = entity.getContentLength();
try (InputStream inputStream = entity.getContent();
OutputStream fos = new FileOutputStream(tempFile);
BufferedOutputStream bos = new BufferedOutputStream(fos)) {
copyLarge(inputStream, bos, new GenericCallback() {
@Override
public void notify(Long progress) {
callback.notify(String.format("Downloading %s: %d%%", id, 100 * progress / size));
}
});
callback.notify("Downloaded " + id + "...");
Header cd = response.getLastHeader("Content-Disposition");
String filename;
if (cd != null) {
filename = cd.getValue().split(";")[1].split("=")[1];
if (filename.length() > 2 && filename.startsWith("\"") && filename.endsWith("\"")) {
filename = filename.substring(1, filename.length() - 1);
}
} else {
HttpUriRequest currentReq = (HttpUriRequest) context.getAttribute(ExecutionContext.HTTP_REQUEST);
HttpHost currentHost = (HttpHost) context.getAttribute(ExecutionContext.HTTP_TARGET_HOST);
String currentUrl = (currentReq.getURI().isAbsolute()) ? currentReq.getURI().toString() : (currentHost.toURI() + currentReq.getURI());
filename = FilenameUtils.getName(currentUrl);
}
return new DownloadResult(tempFile.getPath(), filename);
}
}
@Override
public void reportStats(String[] usageStats) throws IOException {
ArrayList stats = new ArrayList<>();
stats.add(getInstallID());
Collections.addAll(stats, usageStats);
for (String uri : addresses) {
HttpPost post = null;
try {
post = new HttpPost(uri);
post.setHeader("Content-Type", "application/x-www-form-urlencoded");
post.setHeader("Accept-Encoding", "gzip");
HttpEntity body = new StringEntity("stats=" + URLEncoder.encode(Arrays.toString(stats.toArray(new String[0])), "UTF-8"));
post.setEntity(body);
HttpParams requestParams = post.getParams();
requestParams.setIntParameter(CoreConnectionPNames.SO_TIMEOUT, 3000);
requestParams.setIntParameter(CoreConnectionPNames.CONNECTION_TIMEOUT, 1000);
log.debug("Requesting " + uri);
HttpResponse res = execute(post);
checkCacheValidity(uri + "?installID=" + getInstallID(), res);
} finally {
if (post != null) {
try {
post.abort();
} catch (Exception e) {
log.warn("Failure while aborting POST", e);
}
}
}
}
}
private void checkCacheValidity(String uri, HttpResponse res) {
PluginsRepo repo = getRepoCache(uri);
if (repo != null) {
Header hdr = res.getFirstHeader("last-modified");
if (hdr != null) {
if (!repo.isActual(parseDateHeader(hdr))) {
File fname = generateCacheFile(uri);
log.info("Cache file is not valid anymore, will drop it: " + fname.getAbsolutePath());
fname.deleteOnExit();
}
}
}
}
private static long copyLarge(InputStream input, OutputStream output, GenericCallback progressCallback) throws IOException {
byte[] buffer = new byte[4096];
long count = 0L;
int n;
for (; -1 != (n = input.read(buffer)); count += (long) n) {
output.write(buffer, 0, n);
progressCallback.notify(count);
}
return count;
}
public HttpResponse execute(HttpUriRequest request) throws IOException {
return execute(request, null);
}
public HttpResponse execute(HttpUriRequest request, HttpContext context) throws IOException {
for (int c = 1; ; c++) {
HttpResponse response = httpClient.execute(request, context);
try {
if (retryStrategy.retryRequest(response, c, context)) {
EntityUtils.consume(response.getEntity());
long nextInterval = retryStrategy.getRetryInterval();
try {
log.debug("Wait for " + nextInterval);
Thread.sleep(nextInterval);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new InterruptedIOException();
}
} else {
return response;
}
} catch (RuntimeException ex) {
try {
EntityUtils.consume(response.getEntity());
} catch (IOException ioex) {
log.warn("I/O error consuming response content", ioex);
}
throw ex;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy