从URL获取协议+主机名


161

在我的Django应用中,我需要从引荐来源网址中获取主机名request.META.get('HTTP_REFERER')及其协议,以便从类似以下网址的网址中获取:

我应该得到:

我查看了其他相关问题,并找到了有关urlparse的信息,但这并没有成功

>>> urlparse(request.META.get('HTTP_REFERER')).hostname
'docs.google.com'

Answers:


297

您应该能够做到urlparse(docs:python2python3):

from urllib.parse import urlparse
# from urlparse import urlparse  # Python 2
parsed_uri = urlparse('http://stackoverflow.com/questions/1234567/blah-blah-blah-blah' )
result = '{uri.scheme}://{uri.netloc}/'.format(uri=parsed_uri)
print(result)

# gives
'http://stackoverflow.com/'

此答案在第三个示例中添加了,但我认为这可能是问题的缺点,而不是答案的缺点。/http://www.domain.com
SingleNegationElimination 2012年

@TokenMacGuy:雅,是我不好......没有注意到缺少/
杰拉德

8
urlparse.urlparse()返回类似namedtuple的结果;您可以{uri.scheme}://{uri.netloc}/'.format(uri=parsed_uri)用来提高可读性。
2013年

12
我认为这不是一个很好的解决方案,因为netloc它不是域:尝试urlparse.urlparse('http://user:pass@example.com:8080')发现它提供了类似的部分,'user:pass@'并且':8080'
斯塔克化了2014年

22
urlparse模块在Python 3中被重命名为urllib.parse。因此,from urllib.parse import urlparse
SparkAndShine 2015年

86

https://github.com/john-kurkowski/tldextract

这是urlparse的详细版本。它会为您检测域和子域。

从他们的文档中:

>>> import tldextract
>>> tldextract.extract('http://forums.news.cnn.com/')
ExtractResult(subdomain='forums.news', domain='cnn', suffix='com')
>>> tldextract.extract('http://forums.bbc.co.uk/') # United Kingdom
ExtractResult(subdomain='forums', domain='bbc', suffix='co.uk')
>>> tldextract.extract('http://www.worldbank.org.kg/') # Kyrgyzstan
ExtractResult(subdomain='www', domain='worldbank', suffix='org.kg')

ExtractResult 是一个namedtuple,因此可以轻松访问所需的部件。

>>> ext = tldextract.extract('http://forums.bbc.co.uk')
>>> ext.domain
'bbc'
>>> '.'.join(ext[:2]) # rejoin subdomain and domain
'forums.bbc'

2
这是所写问题(如何获得DOMAIN名称)的正确答案。选择的解决方案提供了HOSTNAME,我相信这是作者首先想要的。
烤饼

49

Python3使用urlsplit

from urllib.parse import urlsplit
url = "http://stackoverflow.com/questions/9626535/get-domain-name-from-url"
base_url = "{0.scheme}://{0.netloc}/".format(urlsplit(url))
print(base_url)
# http://stackoverflow.com/

23

纯字符串操作:):

>>> url = "http://stackoverflow.com/questions/9626535/get-domain-name-from-url"
>>> url.split("//")[-1].split("/")[0].split('?')[0]
'stackoverflow.com'
>>> url = "stackoverflow.com/questions/9626535/get-domain-name-from-url"
>>> url.split("//")[-1].split("/")[0].split('?')[0]
'stackoverflow.com'
>>> url = "http://foo.bar?haha/whatever"
>>> url.split("//")[-1].split("/")[0].split('?')[0]
'foo.bar'

就是这样,伙计们。


2
好的简单选择,但是在某些情况下会失败,例如foo.bar?haha
Simon Steinberger

1
@SimonSteinberger :-)怎么样::url.split("//")[-1].split("/")[0].split('?')[0]
SebMa

22
>>> import urlparse
>>> url = 'http://stackoverflow.com/questions/1234567/blah-blah-blah-blah'
>>> urlparse.urljoin(url, '/')
'http://stackoverflow.com/'

2
对于Python 3,导入为from urllib.parse import urlparse
杰夫·鲍文

7

如果您认为自己的网址有效,那么它将一直有效

domain = "http://google.com".split("://")[1].split("/")[0] 

最后一个split是错误的,没有更多的正斜杠可以分割。
CONvid19,5

2
没问题,如果没有更多的斜杠,列表将返回一个元素。因此无论是否存在斜线都可以使用
ZeroErr0r

1
我编辑了您的答案,可以删除不赞成票。很好的解释。Tks。
CONvid19

5

纯字符串操作有什么问题吗:

url = 'http://stackoverflow.com/questions/9626535/get-domain-name-from-url'
parts = url.split('//', 1)
print parts[0]+'//'+parts[1].split('/', 1)[0]
>>> http://stackoverflow.com

如果您希望在末尾加上斜杠,请将该脚本扩展如下:

parts = url.split('//', 1)
base = parts[0]+'//'+parts[1].split('/', 1)[0]
print base + (len(url) > len(base) and url[len(base)]=='/'and'/' or '')

可能可以优化一点...


7
没错,但是我们有一个已经可以完成工作的工具,让我们不要重新发明轮子;)
Gerard 2013年

5

这是一个稍微改进的版本:

urls = [
    "http://stackoverflow.com:8080/some/folder?test=/questions/9626535/get-domain-name-from-url",
    "Stackoverflow.com:8080/some/folder?test=/questions/9626535/get-domain-name-from-url",
    "http://stackoverflow.com/some/folder?test=/questions/9626535/get-domain-name-from-url",
    "https://StackOverflow.com:8080?test=/questions/9626535/get-domain-name-from-url",
    "stackoverflow.com?test=questions&v=get-domain-name-from-url"]
for url in urls:
    spltAr = url.split("://");
    i = (0,1)[len(spltAr)>1];
    dm = spltAr[i].split("?")[0].split('/')[0].split(':')[0].lower();
    print dm

输出量

stackoverflow.com
stackoverflow.com
stackoverflow.com
stackoverflow.com
stackoverflow.com

小提琴:https ://pyfiddle.io/fiddle/23e4976e-88d2-4757-993e-532aa41b7bf0/ ? i = true


恕我直言,最好的解决方案,因为它很简单,它考虑了各种罕见的情况。谢谢!
Simon Steinberger

2
既不简单也不改善
Corey Goldberg

这不是解决问题的方法,因为您没有提供协议(https://或http://)
Alexei Marinichenko,

2

这有点钝,但是urlparse在两个方向上都使用:

import urlparse
def uri2schemehostname(uri):
    urlparse.urlunparse(urlparse.urlparse(uri)[:2] + ("",) * 4)

该奇数("",) * 4位是因为urlparse期望精确地 等于len(urlparse.ParseResult._fields)6 的序列


2

我知道这是一个老问题,但是今天我也遇到了。用单线解决了这个问题:

import re
result = re.sub(r'(.*://)?([^/?]+).*', '\g<1>\g<2>', url)

2

您只需要标准库函数urllib.parse.urlsplit()。这是Python3的示例:

>>> import urllib.parse
>>> o = urllib.parse.urlsplit('https://user:pass@www.example.com:8080/dir/page.html?q1=test&q2=a2#anchor1')
>>> o.scheme
'https'
>>> o.netloc
'user:pass@www.example.com:8080'
>>> o.hostname
'www.example.com'
>>> o.port
8080
>>> o.path
'/dir/page.html'
>>> o.query
'q1=test&q2=a2'
>>> o.fragment
'anchor1'
>>> o.username
'user'
>>> o.password
'pass'

1

可以通过re.search()解决

import re
url = 'https://docs.google.com/spreadsheet/ccc?key=blah-blah-blah-blah#gid=1'
result = re.search(r'^http[s]*:\/\/[\w\.]*', url).group()
print(result)

#result
'https://docs.google.com'

0

获取域名/主机名和来源*

url = '/programming/9626535/get-protocol-host-name-from-url'
hostname = url.split('/')[2] # stackoverflow.com
origin = '/'.join(url.split('/')[:3]) # https://stackoverflow.com

* Origin用于XMLHttpRequest标题


0

您可以简单地使用带有相对根“ /”的urljoin作为第二个参数:

try:
    from urlparse import urljoin  # Python2
except ImportError:
    from urllib.parse import urljoin  # Python3


url = '/programming/9626535/get-protocol-host-name-from-url'

root_url = urljoin(url, '/')

-1

如果它包含的斜线少于3个,则说明您已经得到了,如果不是,那么我们可以发现它之间的出现:

import re

link = http://forum.unisoftdev.com/something

slash_count = len(re.findall("/", link))
print slash_count # output: 3

if slash_count > 2:
   regex = r'\:\/\/(.*?)\/'
   pattern  = re.compile(regex)
   path = re.findall(pattern, url)

   print path
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.