mirror of
https://github.com/Athou/commafeed.git
synced 2026-03-21 21:37:29 +00:00
199 lines
5.3 KiB
Java
199 lines
5.3 KiB
Java
package com.commafeed.backend.feeds;
|
|
|
|
import java.util.Date;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.Queue;
|
|
import java.util.concurrent.ExecutorService;
|
|
import java.util.concurrent.Executors;
|
|
|
|
import javax.annotation.PostConstruct;
|
|
import javax.annotation.PreDestroy;
|
|
import javax.enterprise.context.ApplicationScoped;
|
|
import javax.inject.Inject;
|
|
|
|
import org.apache.commons.codec.digest.DigestUtils;
|
|
import org.apache.commons.lang.time.DateUtils;
|
|
import org.slf4j.Logger;
|
|
import org.slf4j.LoggerFactory;
|
|
|
|
import com.commafeed.backend.MetricsBean;
|
|
import com.commafeed.backend.dao.FeedDAO;
|
|
import com.commafeed.backend.model.Feed;
|
|
import com.commafeed.backend.services.ApplicationSettingsService;
|
|
import com.google.api.client.util.Maps;
|
|
import com.google.common.collect.Lists;
|
|
import com.google.common.collect.Queues;
|
|
|
|
/**
|
|
* Infinite loop fetching feeds from the database and queuing them to the {@link FeedRefreshWorker} pool. Also handles feed database updates
|
|
* at the end of the cycle through {@link #giveBack(Feed)}.
|
|
*
|
|
*/
|
|
@ApplicationScoped
|
|
public class FeedRefreshTaskGiver {
|
|
|
|
protected static final Logger log = LoggerFactory.getLogger(FeedRefreshTaskGiver.class);
|
|
|
|
@Inject
|
|
FeedDAO feedDAO;
|
|
|
|
@Inject
|
|
ApplicationSettingsService applicationSettingsService;
|
|
|
|
@Inject
|
|
MetricsBean metricsBean;
|
|
|
|
@Inject
|
|
FeedRefreshWorker worker;
|
|
|
|
private int backgroundThreads;
|
|
|
|
private Queue<Feed> addQueue = Queues.newConcurrentLinkedQueue();
|
|
private Queue<Feed> takeQueue = Queues.newConcurrentLinkedQueue();
|
|
private Queue<Feed> giveBackQueue = Queues.newConcurrentLinkedQueue();
|
|
|
|
private ExecutorService executor;
|
|
|
|
@PostConstruct
|
|
public void init() {
|
|
backgroundThreads = applicationSettingsService.get().getBackgroundThreads();
|
|
executor = Executors.newFixedThreadPool(1);
|
|
}
|
|
|
|
@PreDestroy
|
|
public void shutdown() {
|
|
executor.shutdownNow();
|
|
while (!executor.isTerminated()) {
|
|
try {
|
|
Thread.sleep(100);
|
|
} catch (InterruptedException e) {
|
|
log.error("interrupted while waiting for threads to finish.");
|
|
}
|
|
}
|
|
}
|
|
|
|
public void start() {
|
|
try {
|
|
// sleeping for a little while, let everything settle
|
|
Thread.sleep(5000);
|
|
} catch (InterruptedException e) {
|
|
log.error("interrupted while sleeping");
|
|
}
|
|
log.info("starting feed refresh task giver");
|
|
|
|
executor.execute(new Runnable() {
|
|
@Override
|
|
public void run() {
|
|
while (!executor.isShutdown()) {
|
|
try {
|
|
Feed feed = take();
|
|
if (feed != null) {
|
|
metricsBean.feedRefreshed();
|
|
worker.updateFeed(feed);
|
|
} else {
|
|
log.debug("nothing to do, sleeping for 15s");
|
|
metricsBean.threadWaited();
|
|
try {
|
|
Thread.sleep(15000);
|
|
} catch (InterruptedException e) {
|
|
log.error("interrupted while sleeping");
|
|
}
|
|
}
|
|
} catch (Exception e) {
|
|
log.error(e.getMessage(), e);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* take a feed from the refresh queue
|
|
*/
|
|
private Feed take() {
|
|
Feed feed = takeQueue.poll();
|
|
|
|
if (feed == null) {
|
|
refill();
|
|
feed = takeQueue.poll();
|
|
}
|
|
return feed;
|
|
}
|
|
|
|
public Long getUpdatableCount() {
|
|
return feedDAO.getUpdatableCount(getThreshold());
|
|
}
|
|
|
|
private Date getThreshold() {
|
|
boolean heavyLoad = applicationSettingsService.get().isHeavyLoad();
|
|
Date threshold = DateUtils.addMinutes(new Date(), heavyLoad ? -15 : -5);
|
|
return threshold;
|
|
}
|
|
|
|
/**
|
|
* add a feed to the refresh queue
|
|
*/
|
|
public void add(Feed feed) {
|
|
Date threshold = getThreshold();
|
|
if (feed.getDisabledUntil() == null || feed.getDisabledUntil().before(threshold)) {
|
|
addQueue.add(feed);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* refills the refresh queue and empties the giveBack queue while at it
|
|
*/
|
|
private void refill() {
|
|
int count = Math.min(100, 3 * backgroundThreads);
|
|
|
|
// first, get feeds that are up to refresh from the database
|
|
List<Feed> feeds = null;
|
|
if (applicationSettingsService.get().isCrawlingPaused()) {
|
|
feeds = Lists.newArrayList();
|
|
} else {
|
|
feeds = feedDAO.findNextUpdatable(count, getThreshold());
|
|
}
|
|
|
|
// then, add to those the feeds we got from the add() method. We add them at the beginning of the list as they probably have a
|
|
// higher priority
|
|
int size = addQueue.size();
|
|
for (int i = 0; i < size; i++) {
|
|
feeds.add(0, addQueue.poll());
|
|
}
|
|
|
|
// set the disabledDate to now as we use the disabledDate in feedDAO to decide what to refresh next. We also use a map to remove
|
|
// duplicates.
|
|
Map<Long, Feed> map = Maps.newLinkedHashMap();
|
|
for (Feed f : feeds) {
|
|
f.setDisabledUntil(new Date());
|
|
map.put(f.getId(), f);
|
|
}
|
|
|
|
// refill the queue
|
|
takeQueue.addAll(map.values());
|
|
|
|
// add feeds from the giveBack queue to the map, overriding duplicates
|
|
size = giveBackQueue.size();
|
|
for (int i = 0; i < size; i++) {
|
|
Feed f = giveBackQueue.poll();
|
|
map.put(f.getId(), f);
|
|
}
|
|
|
|
// update all feeds in the database
|
|
feedDAO.saveOrUpdate(map.values());
|
|
}
|
|
|
|
/**
|
|
* give a feed back, updating it to the database during the next refill()
|
|
*/
|
|
public void giveBack(Feed feed) {
|
|
String normalized = FeedUtils.normalizeURL(feed.getUrl());
|
|
feed.setNormalizedUrl(normalized);
|
|
feed.setNormalizedUrlHash(DigestUtils.sha1Hex(normalized));
|
|
feed.setLastUpdated(new Date());
|
|
giveBackQueue.add(feed);
|
|
}
|
|
|
|
}
|