Uzyskaj protokół + nazwę hosta z adresu URL

162

W mojej aplikacji Django muszę pobrać nazwę hosta z odsyłacza request.META.get('HTTP_REFERER')wraz z jego protokołem, aby z adresów URL, takich jak:

Powinienem dostać:

Przejrzałem inne powiązane pytania i znalazłem informacje o urlparse, ale od tego czasu to nie pomogło

>>> urlparse(request.META.get('HTTP_REFERER')).hostname
'docs.google.com'
Gerard
źródło

Odpowiedzi:

297

Powinieneś być w stanie to zrobić za pomocą urlparse(docs: python2 , python3 ):

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/'
kgr
źródło
ta odpowiedź dodaje a /do trzeciego przykładu http://www.domain.com, ale myślę, że może to być wada pytania, a nie odpowiedzi.
SingleNegationElimination
@TokenMacGuy: ya, mój zły ... nie zauważył brakujący /
Gerard
8
urlparse.urlparse()zwraca nazwany wynik podobny do dwukrotnego; możesz użyć {uri.scheme}://{uri.netloc}/'.format(uri=parsed_uri)dla czytelności.
jfs
12
Nie sądzę, żeby to było dobre rozwiązanie, ponieważ netlocnie jest domeną: spróbuj urlparse.urlparse('http://user:[email protected]:8080')znaleźć części takie jak 'user:pass@'i':8080'
starrify
22
Moduł urlparse zostaje zmieniona na urllib.parse w Pythonie 3. Więcfrom urllib.parse import urlparse
SparkAndShine
86

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

To jest bardziej szczegółowa wersja urlparse. Wykrywa domeny i subdomeny za Ciebie.

Z ich dokumentacji:

>>> 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 jest nazwany trzykrotnie, więc dostęp do potrzebnych części jest prosty.

>>> ext = tldextract.extract('http://forums.bbc.co.uk')
>>> ext.domain
'bbc'
>>> '.'.join(ext[:2]) # rejoin subdomain and domain
'forums.bbc'
dm03514
źródło
2
To jest poprawna odpowiedź na napisane pytanie, jak uzyskać nazwę DOMENY. Wybrane rozwiązanie zapewnia NAZWĘ HOSTA, która moim zdaniem jest tym, czego autor chciał w pierwszej kolejności.
Scone
49

Python3 przy użyciu 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/
Marc SJ
źródło
23

Czyste operacje na strunach :):

>>> 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'

To wszystko, ludzie.

SebMa
źródło
2
Dobra i prosta opcja, ale w niektórych przypadkach zawodzi, np. Foo.bar?haha
Simon Steinberger
1
@SimonSteinberger :-) A co z tym: 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/'
png
źródło
2
W przypadku Pythona 3 import to from urllib.parse import urlparse.
Jeff Bowen
8

jeśli uważasz, że Twój adres URL jest prawidłowy, to będzie działać przez cały czas

domain = "http://google.com".split("://")[1].split("/")[0] 
ZeroErr0r
źródło
To ostatnie splitjest złe, nie ma więcej ukośników do podziału.
CONvid19
2
nie będzie problemu, jeśli nie będzie więcej ukośników, lista wróci z jednym elementem. więc zadziała bez względu na to, czy jest ukośnik, czy nie
ZeroErr0r
1
Zredagowałem twoją odpowiedź, aby móc usunąć głos negatywny. Niezłe wyjaśnienie. Tks.
CONvid19
5

Czy jest coś złego w operacjach na czystych napisach:

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

Jeśli wolisz, aby na końcu był dołączony ukośnik, rozszerz ten skrypt trochę tak:

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

To prawdopodobnie można nieco zoptymalizować ...

Simon Steinberger
źródło
7
nie jest źle, ale mamy narzędzie, które już działa, nie wymyślajmy koła na nowo;)
Gerard
5

Oto nieco ulepszona wersja:

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

Wynik

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

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

Faiz
źródło
IMHO to najlepsze rozwiązanie, ponieważ jest proste i uwzględnia wszystkie rzadkie przypadki. Dzięki!
Simon Steinberger
2
ani proste, ani ulepszone
Corey Goldberg
To nie jest rozwiązanie tego pytania, ponieważ nie podajesz protokołu (https: // lub http: //)
Alexei Marinichenko
2

Jest to trochę tępe, ale używa urlparsew obu kierunkach:

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

ten nieparzysty ("",) * 4bit jest taki, że urlparse oczekuje sekwencji dokładnie len(urlparse.ParseResult._fields) = 6

SingleNegationElimination
źródło
2

Wiem, że to stare pytanie, ale ja też się z nim dzisiaj spotkałem. Rozwiązałem to za pomocą jednej linijki:

import re
result = re.sub(r'(.*://)?([^/?]+).*', '\g<1>\g<2>', url)
Orix Au Yeung
źródło
2

Wystarczy standardowa funkcja biblioteki urllib.parse.urlsplit () . Oto przykład dla Python3:

>>> import urllib.parse
>>> o = urllib.parse.urlsplit('https://user:[email protected]:8080/dir/page.html?q1=test&q2=a2#anchor1')
>>> o.scheme
'https'
>>> o.netloc
'user:[email protected]: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'
famzah
źródło
1

Można to rozwiązać przez 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'
kiwi
źródło
0

aby uzyskać domenę / nazwę hosta i Origin *

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

* Originjest używany w XMLHttpRequestnagłówkach

cieunteung
źródło
0

Możesz po prostu użyć urljoin z względnym korzeniem „/” jako drugim argumentem:

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, '/')
Mirko
źródło
-1

Jeśli zawiera mniej niż 3 ukośniki, więc masz, a jeśli nie, możemy znaleźć wystąpienie między nim:

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
Juraj
źródło