You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
339 lines
15 KiB
339 lines
15 KiB
// plasmaDHTChunk.java
|
|
// ------------------------------
|
|
// part of YaCy
|
|
// (C) by Michael Peter Christen; mc@anomic.de
|
|
// first published on http://www.anomic.de
|
|
// Frankfurt, Germany, 2006
|
|
// created: 18.02.2006
|
|
//
|
|
// $LastChangedDate$
|
|
// $LastChangedRevision$
|
|
// $LastChangedBy$
|
|
//
|
|
// This program is free software; you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation; either version 2 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program; if not, write to the Free Software
|
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
//
|
|
// Using this software in any meaning (reading, learning, copying, compiling,
|
|
// running) means that you agree that the Author(s) is (are) not responsible
|
|
// for cost, loss of data or any harm that may be caused directly or indirectly
|
|
// by usage of this softare or this documentation. The usage of this software
|
|
// is on your own risk. The installation and usage (starting/running) of this
|
|
// software may allow other people or application to access your computer and
|
|
// any attached devices and is highly dependent on the configuration of the
|
|
// software which must be done by the user of the software; the author(s) is
|
|
// (are) also not responsible for proper configuration and usage of the
|
|
// software, even if provoked by documentation provided together with
|
|
// the software.
|
|
//
|
|
// Any changes to this file according to the GPL as documented in the file
|
|
// gpl.txt aside this file in the shipment you received can be done to the
|
|
// lines that follows this copyright notice here, but changes must not be
|
|
// done inside the copyright notive above. A re-distribution must contain
|
|
// the intact and unchanged copyright notice.
|
|
// Contributions and changes to the program code must be marked as such.
|
|
|
|
package de.anomic.plasma;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.HashMap;
|
|
import java.util.HashSet;
|
|
import java.util.Iterator;
|
|
|
|
import de.anomic.index.indexContainer;
|
|
import de.anomic.index.indexRWIEntry;
|
|
import de.anomic.index.indexRWIRowEntry;
|
|
import de.anomic.index.indexURLEntry;
|
|
import de.anomic.kelondro.kelondroBase64Order;
|
|
import de.anomic.kelondro.kelondroException;
|
|
import de.anomic.server.serverCodings;
|
|
import de.anomic.server.logging.serverLog;
|
|
import de.anomic.yacy.yacyCore;
|
|
import de.anomic.yacy.yacyDHTAction;
|
|
import de.anomic.yacy.yacySeedDB;
|
|
|
|
public class plasmaDHTChunk {
|
|
|
|
public static final int chunkStatus_UNDEFINED = -1;
|
|
public static final int chunkStatus_FAILED = 0;
|
|
public static final int chunkStatus_FILLED = 1;
|
|
public static final int chunkStatus_RUNNING = 2;
|
|
public static final int chunkStatus_INTERRUPTED = 3;
|
|
public static final int chunkStatus_COMPLETE = 4;
|
|
|
|
public static final int peerRedundancy = 3;
|
|
|
|
private plasmaWordIndex wordIndex;
|
|
private serverLog log;
|
|
|
|
private int status = chunkStatus_UNDEFINED;
|
|
private String startPointHash = "AAAAAAAAAAAA";
|
|
private indexContainer[] indexContainers = null;
|
|
private HashMap<String, indexURLEntry> urlCache; // String (url-hash) / plasmaCrawlLURL.Entry
|
|
private int idxCount;
|
|
|
|
private long selectionStartTime = 0;
|
|
private long selectionEndTime = 0;
|
|
|
|
private int transferFailedCounter = 0;
|
|
|
|
public indexContainer firstContainer() {
|
|
return indexContainers[0];
|
|
}
|
|
|
|
public indexContainer lastContainer() {
|
|
return indexContainers[indexContainers.length - 1];
|
|
}
|
|
|
|
public indexContainer[] containers() {
|
|
return indexContainers;
|
|
}
|
|
|
|
public int containerSize() {
|
|
return indexContainers.length;
|
|
}
|
|
|
|
public int indexCount() {
|
|
return this.idxCount;
|
|
}
|
|
|
|
private int indexCounter() {
|
|
int c = 0;
|
|
for (int i = 0; i < indexContainers.length; i++) {
|
|
c += indexContainers[i].size();
|
|
}
|
|
return c;
|
|
}
|
|
|
|
public HashMap<String, indexURLEntry> urlCacheMap() {
|
|
return urlCache;
|
|
}
|
|
|
|
public void setStatus(int newStatus) {
|
|
this.status = newStatus;
|
|
}
|
|
|
|
public int getStatus() {
|
|
return this.status;
|
|
}
|
|
|
|
public plasmaDHTChunk(serverLog log, plasmaWordIndex wordIndex, int minCount, int maxCount, int maxtime) {
|
|
try {
|
|
this.log = log;
|
|
this.wordIndex = wordIndex;
|
|
this.startPointHash = selectTransferStart();
|
|
log.logFine("Selected hash " + this.startPointHash + " as start point for index distribution, distance = " + yacyDHTAction.dhtDistance(yacyCore.seedDB.mySeed().hash, this.startPointHash));
|
|
selectTransferContainers(this.startPointHash, minCount, maxCount, maxtime);
|
|
|
|
// count the indexes, can be smaller as expected
|
|
this.idxCount = indexCounter();
|
|
if (this.idxCount < minCount) {
|
|
log.logFine("Too few (" + this.idxCount + ") indexes selected for transfer.");
|
|
this.status = chunkStatus_FAILED;
|
|
}
|
|
} catch (InterruptedException e) {
|
|
this.status = chunkStatus_INTERRUPTED;
|
|
}
|
|
}
|
|
|
|
public plasmaDHTChunk(serverLog log, plasmaWordIndex wordIndex, int minCount, int maxCount, int maxtime, String startHash) {
|
|
try {
|
|
this.log = log;
|
|
this.wordIndex = wordIndex;
|
|
log.logFine("Demanded hash " + startHash + " as start point for index distribution, distance = " + yacyDHTAction.dhtDistance(yacyCore.seedDB.mySeed().hash, this.startPointHash));
|
|
selectTransferContainers(startHash, minCount, maxCount, maxtime);
|
|
|
|
// count the indexes, can be smaller as expected
|
|
this.idxCount = indexCounter();
|
|
if (this.idxCount < minCount) {
|
|
log.logFine("Too few (" + this.idxCount + ") indexes selected for transfer.");
|
|
this.status = chunkStatus_FAILED;
|
|
}
|
|
} catch (InterruptedException e) {
|
|
this.status = chunkStatus_INTERRUPTED;
|
|
}
|
|
}
|
|
|
|
private String selectTransferStart() {
|
|
String startPointHash;
|
|
// first try to select with increasing probality a good start point
|
|
double minimumDistance = ((double) peerRedundancy) / ((double) yacyCore.seedDB.sizeConnected());
|
|
double d, bestDistance = 0.0;
|
|
String bestHash = null;
|
|
for (int i = yacyCore.seedDB.sizeConnected() / 8; i > 0; i--) {
|
|
startPointHash = kelondroBase64Order.enhancedCoder.encode(serverCodings.encodeMD5Raw(Long.toString(i + System.currentTimeMillis()))).substring(2, 2 + yacySeedDB.commonHashLength);
|
|
d = yacyDHTAction.dhtDistance(yacyCore.seedDB.mySeed().hash, startPointHash);
|
|
if (d > (minimumDistance + ((double) i / (double) 10))) {
|
|
return startPointHash;
|
|
}
|
|
if (d > bestDistance) {
|
|
bestDistance = d;
|
|
bestHash = startPointHash;
|
|
}
|
|
}
|
|
// if that fails, take simply the best start point
|
|
if (bestHash == null) {
|
|
return yacyCore.seedDB.mySeed().hash.substring(0, 11) + "z";
|
|
} else {
|
|
return bestHash;
|
|
}
|
|
}
|
|
|
|
private void selectTransferContainers(String hash, int mincount, int maxcount, int maxtime) throws InterruptedException {
|
|
try {
|
|
this.selectionStartTime = System.currentTimeMillis();
|
|
int refcountRAM = selectTransferContainersResource(hash, true, maxcount, maxtime);
|
|
if (refcountRAM >= mincount) {
|
|
log.logFine("DHT selection from RAM: " + refcountRAM + " entries");
|
|
return;
|
|
}
|
|
int refcountFile = selectTransferContainersResource(hash, false, maxcount, maxtime);
|
|
log.logFine("DHT selection from FILE: " + refcountFile + " entries, RAM provided only " + refcountRAM + " entries");
|
|
return;
|
|
} finally {
|
|
this.selectionEndTime = System.currentTimeMillis();
|
|
}
|
|
}
|
|
|
|
private int selectTransferContainersResource(String hash, boolean ram, int maxcount, int maxtime) throws InterruptedException {
|
|
// if (maxcount > 500) { maxcount = 500; } // flooding & OOM reduce
|
|
// the hash is a start hash from where the indexes are picked
|
|
final ArrayList<indexContainer> tmpContainers = new ArrayList<indexContainer>(maxcount);
|
|
try {
|
|
final Iterator<indexContainer> indexContainerIterator = wordIndex.indexContainerSet(hash, ram, true, maxcount).iterator();
|
|
indexContainer container;
|
|
Iterator<indexRWIRowEntry> urlIter;
|
|
indexRWIEntry iEntry;
|
|
indexURLEntry lurl;
|
|
int refcount = 0;
|
|
int wholesize;
|
|
|
|
urlCache = new HashMap<String, indexURLEntry>();
|
|
final double maximumDistance = ((double) peerRedundancy * 2) / ((double) yacyCore.seedDB.sizeConnected());
|
|
final long timeout = (maxtime < 0) ? Long.MAX_VALUE : System.currentTimeMillis() + maxtime;
|
|
while (
|
|
(maxcount > refcount) &&
|
|
(indexContainerIterator.hasNext()) &&
|
|
((container = (indexContainer) indexContainerIterator.next()) != null) &&
|
|
(container.size() > 0) &&
|
|
((tmpContainers.size() == 0) ||
|
|
(yacyDHTAction.dhtDistance(container.getWordHash(), tmpContainers.get(0).getWordHash()) < maximumDistance)) &&
|
|
(System.currentTimeMillis() < timeout)
|
|
) {
|
|
// check for interruption
|
|
if (Thread.currentThread().isInterrupted()) throw new InterruptedException("Shutdown in progress");
|
|
|
|
// make an on-the-fly entity and insert values
|
|
int notBoundCounter = 0;
|
|
try {
|
|
wholesize = container.size();
|
|
urlIter = container.entries();
|
|
// iterate over indexes to fetch url entries and store them in the urlCache
|
|
while ((urlIter.hasNext()) && (maxcount > refcount) && (System.currentTimeMillis() < timeout)) {
|
|
// CPU & IO reduce
|
|
// try { Thread.sleep(50); } catch (InterruptedException e) { }
|
|
|
|
iEntry = (indexRWIEntry) urlIter.next();
|
|
if ((iEntry == null) || (iEntry.urlHash() == null)) {
|
|
urlIter.remove();
|
|
continue;
|
|
}
|
|
lurl = wordIndex.loadedURL.load(iEntry.urlHash(), iEntry, 0);
|
|
if ((lurl == null) || (lurl.comp() == null) || (lurl.comp().url() == null)) {
|
|
//yacyCore.log.logFine("DEBUG selectTransferContainersResource: not-bound url hash '" + iEntry.urlHash() + "' for word hash " + container.getWordHash());
|
|
notBoundCounter++;
|
|
urlIter.remove();
|
|
wordIndex.removeEntry(container.getWordHash(), iEntry.urlHash());
|
|
} else {
|
|
urlCache.put(iEntry.urlHash(), lurl);
|
|
//yacyCore.log.logFine("DEBUG selectTransferContainersResource: added url hash '" + iEntry.urlHash() + "' to urlCache for word hash " + container.getWordHash());
|
|
refcount++;
|
|
}
|
|
}
|
|
|
|
// remove all remaining; we have enough
|
|
while (urlIter.hasNext()) {
|
|
iEntry = (indexRWIEntry) urlIter.next();
|
|
urlIter.remove();
|
|
}
|
|
|
|
// use whats left
|
|
log.logFine("Selected partial index (" + container.size() + " from " + wholesize + " URLs, " + notBoundCounter + " not bound) for word " + container.getWordHash());
|
|
tmpContainers.add(container);
|
|
} catch (kelondroException e) {
|
|
log.logSevere("plasmaWordIndexDistribution/2: deleted DB for word " + container.getWordHash(), e);
|
|
wordIndex.deleteContainer(container.getWordHash());
|
|
}
|
|
}
|
|
// create result
|
|
indexContainers = (indexContainer[]) tmpContainers.toArray(new indexContainer[tmpContainers.size()]);
|
|
//[C[16GwGuFzwffp] has 1 entries, C[16hGKMAl0w97] has 9 entries, C[17A8cDPF6SfG] has 9 entries, C[17Kdj__WWnUy] has 1 entries, C[1
|
|
if ((indexContainers == null) || (indexContainers.length == 0)) {
|
|
log.logFine("No index available for index transfer, hash start-point " + startPointHash);
|
|
this.status = chunkStatus_FAILED;
|
|
return 0;
|
|
}
|
|
|
|
this.status = chunkStatus_FILLED;
|
|
return refcount;
|
|
} catch (kelondroException e) {
|
|
log.logSevere("selectTransferIndexes database corrupted: " + e.getMessage(), e);
|
|
indexContainers = new indexContainer[0];
|
|
urlCache = new HashMap<String, indexURLEntry>();
|
|
this.status = chunkStatus_FAILED;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
public synchronized String deleteTransferIndexes() {
|
|
Iterator<indexRWIRowEntry> urlIter;
|
|
indexRWIEntry iEntry;
|
|
HashSet<String> urlHashes;
|
|
String count = "0";
|
|
|
|
for (int i = 0; i < this.indexContainers.length; i++) {
|
|
// delete entries separately
|
|
if (this.indexContainers[i] == null) {
|
|
log.logFine("Deletion of partial index #" + i + " not possible, entry is null");
|
|
continue;
|
|
}
|
|
int c = this.indexContainers[i].size();
|
|
urlHashes = new HashSet<String>(this.indexContainers[i].size());
|
|
urlIter = this.indexContainers[i].entries();
|
|
while (urlIter.hasNext()) {
|
|
iEntry = urlIter.next();
|
|
urlHashes.add(iEntry.urlHash());
|
|
}
|
|
String wordHash = indexContainers[i].getWordHash();
|
|
count = wordIndex.removeEntriesExpl(this.indexContainers[i].getWordHash(), urlHashes);
|
|
if (log.isFine())
|
|
log.logFine("Deleted partial index (" + c + " URLs) for word " + wordHash + "; " + this.wordIndex.indexSize(wordHash) + " entries left");
|
|
this.indexContainers[i] = null;
|
|
}
|
|
return count;
|
|
}
|
|
|
|
public long getSelectionTime() {
|
|
if (this.selectionStartTime == 0 || this.selectionEndTime == 0) return -1;
|
|
return this.selectionEndTime-this.selectionStartTime;
|
|
}
|
|
|
|
public void incTransferFailedCounter() {
|
|
this.transferFailedCounter++;
|
|
}
|
|
|
|
public int getTransferFailedCounter() {
|
|
return transferFailedCounter;
|
|
}
|
|
}
|