mirror of
https://github.com/yacy/yacy_search_server.git
synced 2025-09-13 16:46:13 -04:00
This is a complete re-design of the serverObjects data structure which holds all data that is submitted during http post requests to YaCy. Before the change, post attributes had been stored to Strings which cannot be larger than 2GB. Furthermore, byte[] uploads had been encoded to b64 Strings to fit into this data structure. Those strings are now replaced by a new data structure, ChunkedBytes which is an object that can hold more than 2GB data using a list of byte[] objects. All required streaming functions are implemented and streaming from http post upload into this data structure works. The b64 encoding has been removed. The ZIM and WARC reader make use of the new data structure.
120 lines
5.5 KiB
Java
120 lines
5.5 KiB
Java
// linkstructure.java
|
|
// ------------
|
|
// (C) 2014 by Michael Peter Christen; mc@yacy.net
|
|
// first published 02.04.2014 on http://yacy.net
|
|
//
|
|
// 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
|
|
|
|
package net.yacy.htroot.api;
|
|
|
|
import java.io.IOException;
|
|
import java.net.MalformedURLException;
|
|
|
|
import net.yacy.cora.document.encoding.ASCII;
|
|
import net.yacy.cora.document.id.DigestURL;
|
|
import net.yacy.cora.order.Base64Order;
|
|
import net.yacy.cora.protocol.HeaderFramework;
|
|
import net.yacy.cora.protocol.RequestHeader;
|
|
import net.yacy.cora.protocol.ResponseHeader;
|
|
import net.yacy.cora.util.ConcurrentLog;
|
|
import net.yacy.search.Switchboard;
|
|
import net.yacy.search.index.Fulltext;
|
|
import net.yacy.search.schema.HyperlinkEdge;
|
|
import net.yacy.search.schema.HyperlinkGraph;
|
|
import net.yacy.search.schema.HyperlinkType;
|
|
import net.yacy.server.serverObjects;
|
|
import net.yacy.server.serverSwitch;
|
|
import net.yacy.server.servletProperties;
|
|
|
|
public class linkstructure {
|
|
|
|
public static serverObjects respond(final RequestHeader header, final serverObjects post, final serverSwitch env) {
|
|
final servletProperties prop = new servletProperties();
|
|
|
|
final String ext = header.get(HeaderFramework.CONNECTION_PROP_EXT, "");
|
|
//final boolean json = ext.equals("json");
|
|
final boolean xml = ext.equals("xml");
|
|
|
|
final Switchboard sb = (Switchboard) env;
|
|
final Fulltext fulltext = sb.index.fulltext();
|
|
if (post == null) return prop;
|
|
final boolean authenticated = sb.adminAuthenticated(header) >= 2;
|
|
final int maxtime = Math.min(post.getInt("maxtime", 60000), authenticated ? 300000 : 1000);
|
|
final int maxnodes = Math.min(post.getInt("maxnodes", 10000), authenticated ? 10000000 : 100);
|
|
final HyperlinkGraph hlg = new HyperlinkGraph();
|
|
int maxdepth = 0;
|
|
|
|
if (post.get("about", "").length() > 0) try {
|
|
// get link structure within a host
|
|
final String about = post.get("about", ""); // may be a URL, a URL hash or a domain hash
|
|
DigestURL url = null;
|
|
String hostname = null;
|
|
if (about.length() == 12 && Base64Order.enhancedCoder.wellformed(ASCII.getBytes(about))) {
|
|
final byte[] urlhash = ASCII.getBytes(about);
|
|
try {
|
|
final String u = authenticated ? sb.getURL(urlhash) : null;
|
|
url = u == null ? null : new DigestURL(u);
|
|
} catch (final IOException e) {
|
|
ConcurrentLog.logException(e);
|
|
}
|
|
} else if (url == null && about.length() > 0) { // consider "about" as url or hostname
|
|
url = new DigestURL(about.indexOf("://") >= 0 ? about : "http://" + about); // accept also domains
|
|
hostname = url.getHost();
|
|
}
|
|
if (hostname == null) return prop;
|
|
|
|
// now collect _all_ documents inside the domain until a timeout appears
|
|
hlg.fill(fulltext.getDefaultConnector(), hostname, null, maxtime, maxnodes);
|
|
maxdepth = hlg.findLinkDepth();
|
|
} catch (final MalformedURLException e) {}
|
|
else if (post.get("to", "").length() > 0) try {
|
|
// get link structure between two links
|
|
final DigestURL to = new DigestURL(post.get("to", ""), null); // must be an url
|
|
final DigestURL from = post.get("from", "").length() == 0 ? null : new DigestURL(post.get("from", "")); // can be null or must be an url
|
|
hlg.path(sb.index, from, to, maxtime, maxnodes);
|
|
} catch (final MalformedURLException e) {}
|
|
|
|
// finally just write out the edge array
|
|
writeGraph(prop, hlg, maxdepth);
|
|
|
|
// Adding CORS Access header for xml output
|
|
if (xml) {
|
|
final ResponseHeader outgoingHeader = new ResponseHeader(200);
|
|
outgoingHeader.put(HeaderFramework.CORS_ALLOW_ORIGIN, "*");
|
|
prop.setOutgoingHeader(outgoingHeader);
|
|
}
|
|
|
|
// return rewrite properties
|
|
return prop;
|
|
}
|
|
|
|
private static void writeGraph(final servletProperties prop, final HyperlinkGraph hlg, final int maxdepth) {
|
|
int c = 0;
|
|
for (final HyperlinkEdge e: hlg) {
|
|
prop.putJSON("edges_" + c + "_source", e.source.getPath());
|
|
prop.putJSON("edges_" + c + "_target", e.target.type.equals(HyperlinkType.Outbound) ? e.target.toNormalform(true) : e.target.getPath());
|
|
prop.putJSON("edges_" + c + "_type", e.target.type.name());
|
|
final Integer depth_source = hlg.getDepth(e.source);
|
|
final Integer depth_target = hlg.getDepth(e.target);
|
|
prop.put("edges_" + c + "_depthSource", depth_source == null ? -1 : depth_source.intValue());
|
|
prop.put("edges_" + c + "_depthTarget", depth_target == null ? -1 : depth_target.intValue());
|
|
c++;
|
|
}
|
|
prop.put("edges", c);
|
|
prop.put("maxdepth", maxdepth);
|
|
}
|
|
|
|
}
|