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.

64 lines
2.4 KiB

8 years ago
  1. import urllib
  2. from fake_useragent import UserAgent
  3. from bs4 import BeautifulSoup as BS
  4. import re
  5. import datetime
  6. user_agent = UserAgent().chrome
  7. re_words = lambda n: re.compile(r"( ?[^ ]+ ?)"+"{0,"+str(n-1)+"}"+r"[^ ]+")
  8. def get_page(url):
  9. page = None
  10. while not page:
  11. page = urllib.request.Request(url,headers = {"User-Agent":user_agent})
  12. page = str(urllib.request.urlopen(page).read())
  13. return page
  14. def get_BS(url):
  15. return BS(get_page(url),"lxml")
  16. class price_finder:
  17. page_funcs = {
  18. "www.amazon.com":{
  19. "name":lambda page: re.sub(r"( {2,}|\n|\\n)","",page.find("span",id="productTitle").text),
  20. "price":lambda page: page.find(name = "span",id = re.compile("priceblock.*")).text
  21. },
  22. "www.banggood.com":{
  23. "name":lambda page: page.find("h1",attrs = {"itemprop":"name"}).text,
  24. "price":lambda page: page.find("div",attrs = {"class":"now"}).get("oriprice")
  25. },
  26. "www.dalprops.com":{
  27. "name":lambda page: page.find("h1",attrs = {"class":"product_title"}).text,
  28. "price":lambda page: page.find("meta",attrs = {"itemprop":"price"}).get("content")
  29. },
  30. "www.gearbest.com":{
  31. "name":lambda page: page.find("div",attrs = {"class":"goods-info-top"}).find("h1").text,
  32. "price":lambda page: page.find(id="unit_price").get("data-orgp")
  33. },
  34. "hobbyking.com":{
  35. "name":lambda page: page.find("h1",attrs={"class":"product-name"}).text,
  36. "price":lambda page: page.find("span",id = re.compile(r"product-price.*")).find("span",attrs={"class":"price"}).text
  37. }
  38. }
  39. def __init__(self,url,space_seperated_categories = 7,bs=None):
  40. self.url=url
  41. self.info_url = urllib.parse.urlparse(url)
  42. if self.info_url.netloc not in price_finder.page_funcs.keys():
  43. raise NotImplementedError("Not implemented for {}".format(self.info_url.netloc))
  44. if bs:
  45. self.bs= bs
  46. else:
  47. self.bs = get_BS(url)
  48. self.words = re_words(space_seperated_categories)
  49. self.time = datetime.datetime.today()
  50. self.info_product = self._get_product_info_()
  51. def _get_product_info_(self):
  52. funcs = price_finder.page_funcs[self.info_url.netloc]
  53. return {
  54. "product_name":self.words.match(
  55. funcs["name"](self.bs)
  56. ).group(0),
  57. "price":funcs["price"](self.bs).replace("$",""),
  58. }