-
Notifications
You must be signed in to change notification settings - Fork 26
/
Copy pathsearch_scrape_parse_threaded.py
118 lines (93 loc) · 2.91 KB
/
search_scrape_parse_threaded.py
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
#!/bin/python
# Inspired by: http://www.ibm.com/developerworks/aix/library/au-threadingpython/
# searches using DuckDuckGo
import Queue
import threading
import urllib2
import time
import re
from BeautifulSoup import BeautifulSoup
queries = ["Bakunawa", "Shen-Lung", "Fafnir"]
search_queue = Queue.Queue()
url_queue = Queue.Queue()
out_queue = Queue.Queue()
start = time.time()
class SearcherThread(threading.Thread):
def __init__(self, search_queue, url_queue):
threading.Thread.__init__(self)
self.search_queue = search_queue
self.url_queue = url_queue
def run(self):
while True:
query = self.search_queue.get()
url = 'http://duckduckgo.com/html/?q='+query
site = urllib2.urlopen(url)
data = site.read()
parsed = BeautifulSoup(data)
for link in parsed.findAll('div', {'class': re.compile('url')}):
self.url_queue.put(link.text)
# signals to search_queue job is done
self.search_queue.task_done()
class ScraperThread(threading.Thread):
def __init__(self, url_queue, out_queue):
threading.Thread.__init__(self)
self.url_queue = url_queue
self.out_queue = out_queue
def run(self):
while True:
# grabs host from queue and crafts url
host = self.url_queue.get()
url = "http://"+str(host)
# grabs urls of hosts and then grabs chunk of webpage
try:
url = urllib2.urlopen(url)
chunk = url.read()
# place chunk into out queue
self.out_queue.put(chunk)
except:
print '** %s failed to resolve' % (url)
# signals to queue job is done
self.url_queue.task_done()
class ParserThread(threading.Thread):
def __init__(self, out_queue):
threading.Thread.__init__(self)
self.out_queue = out_queue
def run(self):
while True:
# grabs host from queue
chunk = self.out_queue.get()
try:
# parse the chunk
soup = BeautifulSoup(chunk)
# Heres where you can scrape any data you want
print soup.find(['title']).text
#print soup.find(['body'])
#print soup.findAll('div', {'class': re.compile('content')})
except:
print '** parsing error'
# signals to queue job is done
self.out_queue.task_done()
def main():
for query in queries:
search_queue.put(query)
#spawn a pool of DuckDuckGo query threads
for i in range(3):
g = SearcherThread(search_queue, url_queue)
g.setDaemon(True)
g.start()
# spawn a pool of threads, and pass them queue instance
for i in range(10):
t = ScraperThread(url_queue, out_queue)
t.setDaemon(True)
t.start()
for i in range(10):
dt = ParserThread(out_queue)
dt.setDaemon(True)
dt.start()
# wait on the queue until everything has been processed
search_queue.join()
url_queue.join()
out_queue.join()
print "Elapsed Time: %s" % (time.time() - start)
if __name__ == '__main__':
main()