Files
SCM-Manager/scm-webapp/src/main/java/sonia/scm/admin/ReleaseFeedParser.java

120 lines
4.2 KiB
Java
Raw Normal View History

2020-09-21 17:04:30 +02:00
/*
* MIT License
*
* Copyright (c) 2020-present Cloudogu GmbH and Contributors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
package sonia.scm.admin;
2020-09-24 10:26:07 +02:00
import com.google.common.base.Strings;
2020-09-21 17:04:30 +02:00
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import sonia.scm.net.ahc.AdvancedHttpClient;
2020-09-24 10:26:07 +02:00
import sonia.scm.version.Version;
2020-09-21 17:04:30 +02:00
import javax.inject.Inject;
import javax.inject.Singleton;
2020-09-21 17:04:30 +02:00
import java.io.IOException;
import java.util.Comparator;
import java.util.Optional;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
2020-09-21 17:04:30 +02:00
@Singleton
public class ReleaseFeedParser {
2020-09-21 17:04:30 +02:00
public static final int DEFAULT_TIMEOUT_IN_MILLIS = 1000;
private static final Logger LOG = LoggerFactory.getLogger(ReleaseFeedParser.class);
2020-09-21 17:04:30 +02:00
private final AdvancedHttpClient client;
private final ExecutorService executorService;
private final long timeoutInMillis;
private Future<Optional<UpdateInfo>> updateInfoFuture;
2020-09-21 17:04:30 +02:00
@Inject
public ReleaseFeedParser(AdvancedHttpClient client) {
this(client, DEFAULT_TIMEOUT_IN_MILLIS);
}
public ReleaseFeedParser(AdvancedHttpClient client, long timeoutInMillis) {
2020-09-21 17:04:30 +02:00
this.client = client;
this.timeoutInMillis = timeoutInMillis;
this.executorService = Executors.newSingleThreadExecutor();
2020-09-21 17:04:30 +02:00
}
2020-09-24 10:26:07 +02:00
Optional<UpdateInfo> findLatestRelease(String url) {
Future<Optional<UpdateInfo>> currentUpdateInfoFuture;
boolean updateInfoFutureCreated = false;
try {
synchronized (this) {
currentUpdateInfoFuture = this.updateInfoFuture;
if (currentUpdateInfoFuture == null) {
currentUpdateInfoFuture = submitRequest(url);
this.updateInfoFuture = currentUpdateInfoFuture;
updateInfoFutureCreated = true;
}
}
try {
return currentUpdateInfoFuture.get(timeoutInMillis, TimeUnit.MILLISECONDS);
} catch (Exception e) {
LOG.error("Could not read release feed", e);
return Optional.empty();
}
} finally {
if (updateInfoFutureCreated) {
synchronized (this) {
this.updateInfoFuture = null;
}
}
}
}
private Future<Optional<UpdateInfo>> submitRequest(String url) {
return executorService.submit(() -> {
LOG.info("Search for newer versions of SCM-Manager");
Optional<ReleaseFeedDto.Release> latestRelease = parseLatestReleaseFromRssFeed(url);
return latestRelease.map(release -> new UpdateInfo(release.getTitle(), release.getLink()));
});
2020-09-22 17:08:22 +02:00
}
private Optional<ReleaseFeedDto.Release> parseLatestReleaseFromRssFeed(String url) {
try {
2020-09-24 10:26:07 +02:00
if (Strings.isNullOrEmpty(url)) {
return Optional.empty();
}
2020-09-22 17:08:22 +02:00
ReleaseFeedDto releaseFeed = client.get(url).request().contentFromXml(ReleaseFeedDto.class);
return filterForLatestRelease(releaseFeed);
} catch (IOException e) {
2020-09-24 10:26:07 +02:00
LOG.error("Could not parse release feed from {}", url, e);
2020-09-22 17:08:22 +02:00
return Optional.empty();
2020-09-21 17:04:30 +02:00
}
}
private Optional<ReleaseFeedDto.Release> filterForLatestRelease(ReleaseFeedDto releaseFeed) {
return releaseFeed.getChannel().getReleases()
2020-09-21 17:04:30 +02:00
.stream()
2020-09-24 10:26:07 +02:00
.min(Comparator.comparing(release -> Version.parse(release.getTitle().trim())));
2020-09-21 17:04:30 +02:00
}
}