forked from simplecrawler/simplecrawler
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
682 lines (544 loc) · 21.4 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
// Simplecrawler
// Christopher Giffard, 2011
//
// http://www.github.com/cgiffard/node-simplecrawler
// Queue Dependency
var FetchQueue = require("./queue.js").queue;
var Cache = require("./cache.js").Cache;
var EventEmitter = require('events').EventEmitter;
var http = require("http"),
https = require("https");
// Crawler Constructor
var Crawler = function(hostname,initialPath,initialPort,interval) {
// SETTINGS TO STUFF WITH (not here! Do it when you create a `new Crawler()`)
// hostname to crawl
this.hostname = hostname || "";
// Gotta start crawling *somewhere*
this.initialPath = initialPath || "/";
this.initialPort = initialPort || 80;
this.initialProtocol = "http";
// Internal 'tick' interval for spawning new requests (as long as concurrency is under cap)
// One request will be spooled per tick, up to the concurrency threshold.
this.interval = interval || 250;
// Maximum request concurrency. Be sensible. Five ties in with node's default maxSockets value.
this.maxConcurrency = 5;
// Maximum time we'll wait for headers
this.timeout = 5 * 60 * 1000;
// User Agent
this.userAgent = "Node/SimpleCrawler 0.1 (http://www.github.com/cgiffard/node-simplecrawler)";
// Queue for requests - FetchQueue gives us stats and other sugar (but it's basically just an array)
this.queue = new FetchQueue();
// Do we filter by hostname?
// Unless you want to be crawling the entire internet, I would recommend leaving this on!
this.filterByhostname = true;
// Do we scan subhostnames?
this.scanSubhostnames = false;
// Treat WWW subhostname the same as the main hostname (and don't count it as a separate subhostname)
this.ignoreWWWhostname = true;
// Or go even further and strip WWW subhostname from hostnames altogether!
this.stripWWWhostname = false;
// Use simplecrawler's internal resource discovery function (switch it off if you'd prefer to discover and queue resources yourself!)
this.discoverResources = true;
// Internal cachestore
this.cache = null;
// Use an HTTP Proxy?
this.useProxy = false;
this.proxyHostname = "127.0.0.1";
this.proxyPort = 8123;
// Support for HTTP basic auth
this.needsAuth = false;
this.authUser = "";
this.authPass = "";
// hostname Whitelist
// We allow hostnames to be whitelisted, so cross-hostname requests can be made.
this.hostnameWhitelist = [];
// Supported Protocols
this.allowedProtocols = [
/^http(s)?\:/ig, // HTTP & HTTPS
/^(rss|atom|feed)(\+xml)?\:/ig // RSS / XML
];
// Max file size to download/store
this.maxResourceSize = 1024 * 1024 * 16; // 16mb
// Supported MIME-types
// Matching MIME-types will be scanned for links
this.supportedMimeTypes = [
/^text\//i,
/^application\/(rss)?[\+\/\-]?xml/i,
/^application\/javascript/i,
/^xml/i
];
// Download linked, but unsupported files (binary - images, documents, etc)
this.downloadUnsupported = true;
// STATE (AND OTHER) VARIABLES NOT TO STUFF WITH
this.commenced = false;
var crawler = this;
var openRequests = 0;
this.fetchConditions = [];
// Initialise our queue by pushing the initial request data into it...
this.queue.add(this.initialProtocol,this.hostname,this.initialPort,this.initialPath);
// Takes a URL, and extracts the protocol, hostname, port, and resource
function processURL(URL,URLContext) {
var split, protocol = "http", hostname = crawler.hostname, port = 80, path = "/";
var hostData = "", pathStack, relativePathStack, invalidPath = false;
if (URLContext) {
port = URLContext.port;
hostname = URLContext.hostname;
protocol = URLContext.protocol;
path = URLContext.path;
}
// Trim URL
URL = URL.replace(/^\s+/,"").replace(/\s+$/,"");
// Check whether we're global, hostname-absolute or relative
if (URL.match(/^http(s)?:\/\//i)) {
// We're global. Try and extract hostname and port
split = URL.replace(/^http(s)?:\/\//i,"").split(/\//g);
hostData = split[0] && split[0].length ? split[0] : hostname;
if (hostData.split(":").length > 0) {
hostData = hostData.split(":");
hostname = hostData[0];
port = hostData.pop();
port = isNaN(port) ? 80 : port;
}
if (URL.match(/^https:\/\//i)) {
protocol = "https";
}
path = "/" + split.slice(1).join("/");
} else if (URL.match(/^\//)) {
// Absolute URL. Easy to handle!
path = URL;
} else {
// Relative URL
// Split into a stack and walk it up and down to calculate the absolute path
var processedPathContext = URLContext.path;
processedPathContext = processedPathContext.split(/\?/).shift();
processedPathContext = processedPathContext.split(/\#/).shift();
pathStack = processedPathContext.split("/");
if (!URLContext.path.match(/\/\s*$/)) {
pathStack = pathStack.slice(0,pathStack.length-1);
}
relativePathStack = URL.split(/\//g);
invalidPath = false;
relativePathStack.forEach(function(pathChunk) {
if (!invalidPath) {
if (pathChunk.match(/^\.\./)) {
if (pathStack.length) {
pathStack = pathStack.slice(0,pathStack.length-1);
} else {
// URL tries to go too deep. Ignore it - it's invalid.
invalidPath = true;
}
} else if (pathChunk.match(/^\./)) {
// Ignore this chunk - it just points to the same directory...
} else {
pathStack.push(pathChunk);
}
}
});
// This relative URL is junky. Kill it
if (invalidPath) {
return false;
}
// Filter blank path chunks
pathStack = pathStack.filter(function(item) {
return !!item.length;
});
path = "/" + pathStack.join("/");
}
// Strip the www subhostname out if required
if (crawler.stripWWWhostname) {
hostname = hostname.replace(/^www\./ig,"");
}
// Replace problem entities...
path = path.replace(/&/ig,"&");
// Ensure hostname is always lower-case
hostname = hostname.toLowerCase();
return {
"protocol": protocol,
"hostname": hostname,
"port": port,
"path": path
};
}
// Make this function available externally
crawler.processURL = processURL;
// Determines whether the protocol is supported, given a URL
function protocolSupported(URL) {
var supported = false;
if (URL.match(/^[a-z0-9]+\:/i)) {
crawler.allowedProtocols.forEach(function(protocolCheck) {
if (!!protocolCheck.exec(URL)) {
supported = true;
}
});
return supported;
} else {
return true;
}
}
// Determines whether the mimetype is supported, given a... mimetype
function mimeTypeSupported(MIMEType) {
var supported = false;
crawler.supportedMimeTypes.forEach(function(mimeCheck) {
if (!!mimeCheck.exec(MIMEType)) {
supported = true;
}
});
return supported;
}
// Input some text/html and this function will return a bunch of URLs for queueing
// (if there are actually any in the resource, otherwise it'll return an empty array)
function discoverResources(resourceData,queueItem) {
var resources = [], resourceText = resourceData.toString("utf8");
// Clean links
function cleanAndQueue(urlMatch) {
if (urlMatch) {
urlMatch.forEach(function(URL) {
URL = URL.replace(/^(\s?href|\s?src)=['"]?/i,"").replace(/^\s*/,"");
URL = URL.replace(/^url\(['"]*/i,"");
URL = URL.replace(/^javascript\:[a-z0-9]+\(['"]/i,"");
URL = URL.replace(/["'\)]$/i,"");
URL = URL.split(/\s+/g).shift();
if (URL.match(/^\s*#/)) {
// Bookmark URL
return false;
}
URL = URL.split("#").shift();
if (URL.replace(/\s+/,"").length && protocolSupported(URL)) {
if (!resources.reduce(function(prev,current) {
return prev || current === URL;
},false)) {
resources.push(URL);
}
}
});
}
}
// Rough scan for URLs
cleanAndQueue(resourceText.match(/(\shref\s?=\s?|\ssrc\s?=\s?|url\()['"]?([^"'\s>\)]+)/ig));
cleanAndQueue(resourceText.match(/http(s)?\:\/\/[^?\s><\'\"]+/ig));
cleanAndQueue(resourceText.match(/url\([^)]+/ig));
// This might be a bit of a gamble... but get hard-coded strings out of javacript: URLs
// They're often popup-image or preview windows, which would otherwise be unavailable to us
cleanAndQueue(resourceText.match(/^javascript\:[a-z0-9]+\(['"][^'"\s]+/ig));
return resources;
}
// Checks to see whether hostname is valid for crawling.
function hostnameValid(hostname) {
function hostnameInWhitelist(hostname) {
// If there's no whitelist, or the whitelist is of zero length, just return false.
if (!crawler.hostnameWhitelist || !crawler.hostnameWhitelist.length) return false;
// Otherwise, scan through it.
return !!crawler.hostnameWhitelist.reduce(function(prev,cur,index,array) {
// If we already located the relevant hostname in the whitelist...
if (prev) return prev;
// If the hostname is just equal, return true.
if (hostname === cur) return true;
// If we're ignoring WWW subhostnames, and both hostnames, less www. are the same, return true.
if (crawler.ignoreWWWhostname && hostname.replace(/^www\./i,"") === cur.replace(/^www\./i,"")) return true;
// Otherwise, sorry. No dice.
return false;
},false);
}
// Checks if the first hostname is a subhostname of the second
function isSubhostnameOf(subhostname,hostname) {
hostnameParts = hostname.split(/\./g);
subhostnameParts = subhostname.split(/\./g);
// If we're ignoring www, remove it from both (if www is the first hostname component...)
if (crawler.ignoreWWWhostname) {
if (hostnameParts[0].match(/^www\./i)) hostnameParts = hostnameParts.slice(1);
if (subhostnameParts[0].match(/^www\./i)) hostnameParts = hostnameParts.slice(1);
}
// Can't have a subhostname that's shorter than its parent.
if (subhostname.length < hostname.length) return false;
// Loop through subhostname backwards, from TLD to least significant hostname, break on first error.
var index = subhostnameParts.length - 1;
while (index >= 0 && index >= subhostnameParts.length - hostnameParts.length) {
if (subhostnameParts[index] !== hostnameParts[index]) return false;
index --;
}
return true;
}
// If we're not filtering by hostname, just return true.
return (!crawler.filterByhostname ||
// Or if the hostname is just the right one, return true.
(hostname === crawler.hostname) ||
// Or if we're ignoring WWW subhostnames, and both hostnames, less www. are the same, return true.
(crawler.ignoreWWWhostname && crawler.hostname.replace(/^www\./i,"") === hostname.replace(/^www\./i,"")) ||
// Or if the hostname in question exists in the hostname whitelist, return true.
hostnameInWhitelist(hostname) ||
// Or if we're scanning subhostnames, and this hostname is a subhostname of the crawler's set hostname, return true.
(crawler.scanSubhostnames && isSubhostnameOf(hostname,crawler.hostname)));
}
// Make available externally to this scope
crawler.ishostnameValid = hostnameValid;
// Externally accessible function for auditing the number of open requests...
crawler.openRequests = function() {
return openRequests;
};
// Input some text/html and this function will delegate resource discovery, check link validity
// and queue up resources for downloading!
function queueLinkedItems(resourceData,queueItem) {
discoverResources(resourceData,queueItem).forEach(function(url){ queueURL(url,queueItem); });
}
// Clean and queue a single URL...
function queueURL(url,queueItem) {
var parsedURL = typeof(url) === "object" ? url : processURL(url,queueItem);
// URL Parser decided this URL was junky. Next please!
if (!parsedURL) {
return false;
}
// Pass this URL past fetch conditions to ensure the user thinks it's valid
var fetchDenied = false;
fetchDenied = crawler.fetchConditions.reduce(function(prev,callback) {
return fetchDenied || !callback(parsedURL);
},false);
if (fetchDenied) {
// Fetch Conditions conspired to block URL
return false;
}
// Check the hostname is valid before adding it to the queue
if (hostnameValid(parsedURL.hostname)) {
try {
crawler.queue.add(
parsedURL.protocol,
parsedURL.hostname,
parsedURL.port,
parsedURL.path,
function queueAddCallback(error,newQueueItem) {
if (error) {
// We received an error condition when adding the callback
crawler.emit("queueerror",error,parsedURL);
} else {
crawler.emit("queueadd",newQueueItem,parsedURL);
newQueueItem.referrer = queueItem.url;
}
}
);
} catch(error) {
// If we caught an error, emit queueerror
crawler.emit("queueerror",error,parsedURL);
}
}
}
// Fetch a queue item
function fetchQueueItem(queueItem) {
openRequests ++;
// Emit fetchstart event
crawler.emit("fetchstart",queueItem);
// Variable declarations
var fetchData = false, requestOptions, clientRequest, timeCommenced, timeHeadersReceived, timeDataReceived, parsedURL;
var responseBuffer, responseLength, responseLengthReceived, contentType;
// Mark as spooled
queueItem.status = "spooled";
client = (queueItem.protocol === "https" ? https : http);
// Extract request options from queue;
var requestHost = queueItem.hostname,
requestPort = queueItem.port,
requestPath = queueItem.path;
// Are we passing through an HTTP proxy?
if (crawler.useProxy) {
requestHost = crawler.proxyHostname;
requestPort = crawler.proxyPort;
requestPath = queueItem.url;
}
// Load in request options
requestOptions = {
host: requestHost,
port: requestPort,
path: requestPath,
headers: {
"User-Agent": crawler.userAgent
}
};
if(crawler.needsAuth) {
var auth = 'Basic ' + new Buffer(crawler.authUser + ":" + crawler.authPass).toString('base64');
requestOptions.headers['Authorization'] = auth;
}
// Record what time we started this request
timeCommenced = (new Date().getTime());
// Get the resource!
clientRequest = client.get(requestOptions,function(response) {
var dataReceived = false;
responseLengthReceived = 0;
// Record what time we first received the header information
timeHeadersReceived = (new Date().getTime());
responseLength = parseInt(response.headers["content-length"],10);
responseLength = !isNaN(responseLength) ? responseLength : 0;
// Save timing and content some header information into queue
queueItem.stateData.requestLatency = (timeHeadersReceived - timeCommenced);
queueItem.stateData.requestTime = (timeHeadersReceived - timeCommenced);
queueItem.stateData.contentLength = responseLength;
queueItem.stateData.contentType = contentType = response.headers["content-type"];
queueItem.stateData.code = response.statusCode;
// Save entire headers, in less scannable way
queueItem.stateData.headers = response.headers;
// Emit header receive event
crawler.emit("fetchheaders",queueItem,response);
// Ensure response length is reasonable...
responseLength = responseLength > 0 ? responseLength : crawler.maxResourceSize;
queueItem.stateData.contentLength = responseLength;
// Function for dealing with 200 responses
function processReceivedData() {
if (!queueItem.fetched) {
timeDataReceived = (new Date().getTime());
queueItem.fetched = true;
queueItem.status = "downloaded";
queueItem.stateData.downloadTime = (timeDataReceived - timeHeadersReceived);
queueItem.stateData.requestTime = (timeDataReceived - timeCommenced);
queueItem.stateData.actualDataSize = responseBuffer.length;
queueItem.stateData.sentIncorrectSize = responseBuffer.length !== responseLength;
crawler.emit("fetchcomplete",queueItem,responseBuffer,response);
// First, save item to cache (if we're using a cache!)
if (crawler.cache !== null && crawler.cache.setCacheData instanceof Function) {
crawler.cache.setCacheData(queueItem,responseBuffer);
}
// We only process the item if it's of a valid mimetype
// and only if the crawler is set to discover its own resources
if (mimeTypeSupported(contentType) && crawler.discoverResources) {
queueLinkedItems(responseBuffer,queueItem);
}
openRequests --;
}
}
function receiveData(chunk) {
if (chunk && chunk.length && !dataReceived) {
if (responseLengthReceived + chunk.length > responseBuffer.length) {
// Oh dear. We've been sent more data than we were initially told.
// This could be a mis-calculation, or a streaming resource.
// Let's increase the size of our buffer to match, as long as it isn't
// larger than our maximum resource size.
if (responseLengthReceived + chunk.length <= crawler.maxResourceSize) {
// Start by creating a new buffer, which will be our main buffer going forward...
var tmpNewBuffer = new Buffer(responseLengthReceived + chunk.length);
// Copy all our old data into it...
responseBuffer.copy(tmpNewBuffer,0,0,responseBuffer.length);
// And now the new chunk
chunk.copy(tmpNewBuffer,responseBuffer.length,0,chunk.length);
// And now make the response buffer our new buffer, leaving the original for GC
responseBuffer = tmpNewBuffer;
} else {
// Oh dear oh dear! The response is not only more data than we were initially told,
// but it also exceeds the maximum amount of data we're prepared to download per resource.
// Throw error event and ignore.
//
// We'll then deal with the data that we have.
crawler.emit("fetchdataerror",queueItem,response);
}
} else {
// Copy the chunk data into our main buffer
chunk.copy(responseBuffer,responseLengthReceived,0,chunk.length);
}
// Increment our data received counter
responseLengthReceived += chunk.length;
}
if ((responseLengthReceived >= responseLength || response.complete) && !dataReceived) {
// Slice the buffer to chop off any unused space
responseBuffer = responseBuffer.slice(0,responseLengthReceived);
dataReceived = true;
processReceivedData();
}
}
// If we should just go ahead and get the data
if (response.statusCode >= 200 && response.statusCode < 300 && responseLength <= crawler.maxResourceSize) {
queueItem.status = "headers";
// Create a buffer with our response length
responseBuffer = new Buffer(responseLength);
response.on("data",receiveData);
response.on("end",receiveData);
// We've got a not-modified response back
} else if (response.statusCode === 304) {
if (crawler.cache !== null && crawler.cache.getCacheData) {
// We've got access to a cache
crawler.cache.getCacheData(queueItem,function(cacheObject) {
crawler.emit("notmodified",queueItem,response,cacheObject);
});
} else {
// Emit notmodified event. We don't have a cache available, so we don't send any data.
crawler.emit("notmodified",queueItem,response);
}
// If we should queue a redirect
} else if (response.statusCode >= 300 && response.statusCode < 400 && response.headers.location) {
queueItem.fetched = true;
queueItem.status = "redirected";
// Parse the redirect URL ready for adding to the queue...
parsedURL = processURL(response.headers.location,queueItem);
// Emit redirect event
crawler.emit("fetchredirect",queueItem,parsedURL,response);
// Clean URL, add to queue...
queueURL(parsedURL,queueItem);
openRequests --;
// Ignore this request, but record that we had a 404
} else if (response.statusCode === 404) {
queueItem.fetched = true;
queueItem.status = "notfound";
// Emit 404 event
crawler.emit("fetch404",queueItem,response);
openRequests --;
// And oh dear. Handle this one as well. (other 400s, 500s, etc)
} else {
queueItem.fetched = true;
queueItem.status = "failed";
// Emit 5xx / 4xx event
crawler.emit("fetcherror",queueItem,response);
openRequests --;
}
});
clientRequest.on("error",function(errorData) {
openRequests --;
// Emit 5xx / 4xx event
crawler.emit("fetchclienterror",queueItem,errorData);
queueItem.fetched = true;
queueItem.stateData.code = 599;
queueItem.status = "failed";
});
}
// Crawl init
this.crawl = function() {
if (openRequests < crawler.maxConcurrency) {
crawler.queue.oldestUnfetchedItem(function(err,queueItem) {
if (queueItem) {
fetchQueueItem(queueItem);
} else if (openRequests === 0) {
crawler.queue.complete(function(err,completeCount) {
if (completeCount === crawler.queue.length) {
crawler.emit("complete");
crawler.stop();
}
});
}
});
}
};
};
Crawler.prototype = new EventEmitter();
Crawler.prototype.start = function() {
this.crawlIntervalID = setInterval(this.crawl,this.interval);
this.crawl();
this.running = true;
};
Crawler.prototype.stop = function() {
clearInterval(this.crawlIntervalID);
this.running = false;
};
Crawler.prototype.addFetchCondition = function(callback) {
if (callback instanceof Function) {
this.fetchConditions.push(callback);
return this.fetchConditions.length - 1;
} else {
throw new Error("Fetch Condition must be a function.");
}
};
Crawler.prototype.removeFetchCondition = function(index) {
if (this.fetchConditions[index] && this.fetchConditions[index] instanceof Function) {
var tmpArray = this.fetchConditions.slice(0,index);
tmpArray = this.fetchConditions.length-1 > index ? tmpArray.concat(this.fetchConditions.slice(0,index+1)) : tmpArray;
this.fetchConditions = tmpArray;
return true;
} else {
throw new Error("Unable to find indexed Fetch Condition.");
}
};
// EXPORTS
exports.FetchQueue = FetchQueue;
exports.Cache = Cache;
exports.Crawler = Crawler;