I am scraping some websites to extract the Facebook page URLs in Python 3 using Beautiful Soup. I am interested to extract only one URL for each website and that redirects into a Facebook page profile and that is not a share-type one.
For this reason I am attempting to use regular expression to extract the URLs from the href
attribute in <a>
tags. With Beautiful Soup I extract the first <a>
tag for each website, filtering by value of the href
attribute that contains a Facebook page URL.
My code is the following:
# coding=utf-8
from bs4 import BeautifulSoup
import requests
import re
def makeSoup(website):
if 'http' in website:
page = requests.get(website)
else:
page = requests.get('http://' + website)
soup = BeautifulSoup(page.content, 'html.parser')
page.close()
return soup
def facebookProfileScraper(soup):
link = soup.find('a', {'href': re.compile("https?://(www\\.)?facebook\\.com/[^(share)]?(\\w+\\.?)+")})
if link is None:
return "NaN"
return link['href']
Examples of the <a>
tags from which I'd like to extract the URLs are the following (I put numbers to identify each website also for the results I had with my attempts):
1. <a class="rss fb" href="http://www.facebook.com/gironafc" target="_blank">Facebook</a>
2. <a href="https://www.facebook.com/waterworld.parcaquatic" target="_blank"><i class="fa fa-facebook"></i></a>
3. <a class="social facebook" target="_blank" href="https://www.facebook.com/aquabrava"><span class="fa fa-facebook"></span></a>
4. <a href="https://www.facebook.com/UEO1921" target="_blank"><img alt="Facebook" height="32" src="http://www.ueolot.com/wp-content/themes/realsoccer/images/light/social-icon/facebook.png" width="32"/>
</a>
5. <a href="https://www.facebook.com/Roc%C3%B2drom-Girona-187271461378780/">Facebook</a>
6. <a class="fb_share" href="https://www.facebook.com/pages/Skydive-Empuriabrava/44214266003?fref=ts" target="_blank"></a>
https?://(www\\.)?facebook\\.com/[^(share)]?(\\w+\\.?)+
But I got these </a>
tags:
1. <a href="http://facebook.com/share.php?src=bm&v=3&u=" target="_blank"><span class="fa fa-facebook"></span></a>
2. <a href="https://www.facebook.com/waterworld.parcaquatic" target="_blank"><i class="fa fa-facebook"></i></a>
3. <a class="social facebook" href="https://www.facebook.com/aquabrava" target="_blank"><span class="fa fa-facebook"></span></a>
4. <a href="https://www.facebook.com/UEO1921" target="_blank"><img alt="Facebook" height="32" src="http://www.ueolot.com/wp-content/themes/realsoccer/images/light/social-icon/facebook.png" width="32"/>
</a>
5. <a href="https://www.facebook.com/Roc%C3%B2drom-Girona-187271461378780/">Facebook</a>
6. <a class="fb_share" href="https://www.facebook.com/pages/Skydive-Empuriabrava/44214266003?fref=ts" target="_blank"></a>
From website 1. I get the wrong <a>
tag.
https?://(www\\.)?facebook\\.com/[^(share)](\\w+\\.?)+
I tried removing the ?
after [^share]
but I got the following tags:
1. <a class="rss fb" href="http://www.facebook.com/gironafc" target="_blank">Facebook</a>
2. <a href="https://www.facebook.com/waterworld.parcaquatic" target="_blank"><i class="fa fa-facebook"></i></a>
3. None
4. <a href="https://www.facebook.com/UEO1921" target="_blank"><img alt="Facebook" height="32" src="http://www.ueolot.com/wp-content/themes/realsoccer/images/light/social-icon/facebook.png" width="32"/>
</a>
5. <a href="https://www.facebook.com/Roc%C3%B2drom-Girona-187271461378780/">Facebook</a>
6.<a class="fb_share" href="https://www.facebook.com/pages/Skydive-Empuriabrava/44214266003?fref=ts" target="_blank"></a>
From website 3. I don't extract any <a>
tag because by [^(share)]
I am negating any url with a
(or any of s
, h
, r
,e
) after http://www.facebook.com/
.
https?://(www\\.)?facebook\\.com/(\\w+\\.?)+
I tried removing [^share]
but the tags I got were the same of the first attempt, therefore getting share type URL too.
How can I select only the a
tags that they don't have a share-type Facebook URL as href
value?
答案 0 :(得分:2)
def foo(url):
l = []
soup = BeautifulSoup(requests.get(url).text, "html.parser")
links = soup.find_all("a")
for link in links:
if not "share" in link.get("href").lower():
l.append(link)
return l
此函数检查链接中的share
,并返回其中没有share
的链接。
答案 1 :(得分:1)
我通过改进正则表达式找到了解决方案。 This这个问题对我有很大帮助。 这是我的案例的正则表达式:
https?://(www\.)?facebook\.com/(?!share\.php).(\S+\.?)+
它将所有<a>
标签与Facebook页面URL匹配为href
值。
使用正则表达式(?!anywordorexpression).
不会提取任何包含anywordorexpression
子字符串的字符串。
答案 2 :(得分:0)
您可以通过bs4 4.7.1将更有效的css属性选择器与:not和:contains伪类一起使用
links = [item['href'] for item in soup.select("[href^='https://www.facebook.com/']:not(:contains(share))")]
仅用于第一个链接
link = soup.select_one("[href^='https://www.facebook.com/']:not(:contains(share))")['href']