Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for DNS #2214

Open
wants to merge 25 commits into
base: py3
Choose a base branch
from
Open
Changes from 1 commit
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
fd11d4f
Add support for DNS
filips123 Sep 28, 2019
586f2dd
Add plugin description for Zeroname
filips123 Sep 28, 2019
08b84a9
Install plugin dependencies in CI
filips123 Sep 30, 2019
79759a3
Install plugin dependencies in Docker and Vagrant
filips123 Sep 30, 2019
060c66a
Use DNSPython temporary fork until PR is merged
filips123 Oct 1, 2019
476f612
Update DNSLink to allow Python 3.4
filips123 Oct 1, 2019
b17a365
Use hard-coded commit hash for DNSPython
filips123 Oct 1, 2019
460a103
Resolve requested changes
filips123 Oct 1, 2019
66516c0
Remove trailing commas
filips123 Oct 1, 2019
649cd90
Remove UiRequestPlugin from DNS plugin
filips123 Oct 16, 2019
a22b074
Merge branch 'py3' of https://github.com/HelloZeroNet/ZeroNet into py3
filips123 Oct 16, 2019
9104f42
Merge branch 'py3' of https://github.com/HelloZeroNet/ZeroNet into dns
filips123 Oct 29, 2019
2caf10b
Update DNSPython
filips123 Oct 29, 2019
1c099d5
Add wrapperInfo command
purplesyringa Oct 29, 2019
bcbd27e
Always support transparent proxies (including classic DNS)
purplesyringa Oct 29, 2019
071f1c6
Bundle plugin dependencies along source code
filips123 Oct 31, 2019
e679f81
Update dependencies
filips123 Dec 14, 2019
db3f586
Update dependencies
filips123 Feb 3, 2020
0bda4b4
Requests with `HOST` in `ui_host` shouldn't be proxy requests
filips123 Feb 21, 2020
1fe1e06
Also check if `HOST` is in `ui_host` before changing route path
filips123 Feb 21, 2020
633aaea
Fix checking for UI host where config is not set
filips123 Feb 22, 2020
c9883a1
Allow setting custom WS URL for proxy requests
filips123 Feb 22, 2020
5cff378
Merge branch 'py3' into dns
filips123 Apr 7, 2020
4b3e8cd
Install plugin dependencies in GitHub Actions
filips123 Apr 7, 2020
2078a48
Replace SecureDNS and Dnswarden with LibreDNS
filips123 Apr 21, 2020
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Next Next commit
Add support for DNS
  • Loading branch information
filips123 committed Sep 28, 2019
commit fd11d4f9b51fc1e5cff5f755bb3659f939053635
21 changes: 21 additions & 0 deletions plugins/DNS/ConfigPlugin.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
from Plugin import PluginManager

@PluginManager.registerTo('ConfigPlugin')
class ConfigPlugin:
def createArguments(self):
nameservers = [
'https://doh.securedns.eu/dns-query',

'https://doh-de.blahdns.com/dns-query',
'https://doh-jp.blahdns.com/dns-query',
'https://doh-ch.blahdns.com/dns-query',

'https://doh.dnswarden.com/uncensored',
]

group = self.parser.add_argument_group('DNS plugin')

group.add_argument('--dns_nameservers', help='Nameservers for DNS plugin', default=nameservers, metavar='address', nargs='*')
group.add_argument('--dns_configure', help='Configure resolver with system config for DNS plugin', action='store_true', default=False)

return super(ConfigPlugin, self).createArguments()
103 changes: 103 additions & 0 deletions plugins/DNS/DNSResolver.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,103 @@
from Config import config

import logging
import time
import json
import re
import os

log = logging.getLogger('DNSPlugin')

class DNSResolver:
loaded = False
cache = {}

def __init__(self, site_manager, nameservers, configure):
self.site_manager = site_manager
self.nameservers = nameservers
self.configure = configure

def load(self):
if not self.loaded:
self.loadModule()
self.loadCache()

self.resolver = dns.resolver.Resolver(configure=self.configure)

if not self.configure:
self.resolver.nameservers = self.nameservers

self.loaded = True

def loadModule(self):
import dns.resolver
import dnslink

if config.tor == 'always':
response = type('lamdbaobject', (object,), {})()
response.flags = dns.flags.TC
filips123 marked this conversation as resolved.
Show resolved Hide resolved

query = lambda *x, **y: response
dns.query.udp = query

globals()['dns'] = locals()['dns']
globals()['dnslink'] = locals()['dnslink']
filips123 marked this conversation as resolved.
Show resolved Hide resolved

def loadCache(self, path=os.path.join(config.data_dir, 'dns_cache.json')):
if os.path.isfile(path):
try: self.cache = json.load(open(path))
except json.decoder.JSONDecodeError: pass
filips123 marked this conversation as resolved.
Show resolved Hide resolved

def saveCache(self, path=os.path.join(config.data_dir, 'dns_cache.json')):
json.dump(self.cache, open(path, 'w'), indent=2)
filips123 marked this conversation as resolved.
Show resolved Hide resolved

def isDomain(self, address):
return re.match(r'(.*?)([A-Za-z0-9_-]+\.[A-Za-z0-9_-]+)$', address)

def resolveDomain(self, domain):
if not self.loaded:
self.load()

domain = domain.lower()

cache_entry = self.cache.get(domain)
if cache_entry and time.time() < cache_entry['timeout']:
log.info('cache: %s -> %s', domain, cache_entry['address'])
return cache_entry['address']

try:
resolver_record = dnslink.resolve(domain, protocol='zeronet', resolver=self.resolver)[0]
resolver_entry = {'domain': domain, 'address': resolver_record.split('/', 2)[2]}
except IndexError:
resolver_entry = None

resolver_error = None
try:
self.resolver.query(domain, 'TXT')
except dns.resolver.NXDOMAIN:
pass
except dns.resolver.NoAnswer:
pass
except dns.exception.DNSException as err:
resolver_error = err

if resolver_entry and not resolver_error:
log.info('resolver: %s -> %s', domain, resolver_entry['address'])
self.saveInCache(resolver_entry)
return resolver_entry['address']

if cache_entry and resolver_error:
log.info('fallback: %s -> %s', domain, cache_entry['address'])
self.extendInCache(cache_entry)
return cache_entry['address']

def saveInCache(self, entry):
entry['timeout'] = time.time() + 60 * 60
self.cache[entry['domain']] = entry

self.saveCache()

def extendInCache(self, entry):
self.cache[entry['domain']]['timeout'] = time.time() + 60 * 15

self.saveCache()
34 changes: 34 additions & 0 deletions plugins/DNS/SiteManagerPlugin.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
from Config import config
from Plugin import PluginManager

from .DNSResolver import DNSResolver

allow_reload = False

@PluginManager.registerTo('SiteManager')
class SiteManagerPlugin:
_dns_resolver = None

@property
def dns_resolver(self):
if not self._dns_resolver:
nameservers = config.dns_nameservers
configure = config.dns_configure

self._dns_resolver = DNSResolver(
site_manager=self,
nameservers=nameservers,
configure=configure,
filips123 marked this conversation as resolved.
Show resolved Hide resolved
)

return self._dns_resolver

def load(self, *args, **kwargs):
super(SiteManagerPlugin, self).load(*args, **kwargs)
self.dns_resolver.load()

def isDomain(self, address):
return self.dns_resolver.isDomain(address) or super(SiteManagerPlugin, self).isDomain(address)

def resolveDomain(self, domain):
return self.dns_resolver.resolveDomain(domain) or super(SiteManagerPlugin, self).resolveDomain(domain)
23 changes: 23 additions & 0 deletions plugins/DNS/UiRequestPlugin.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
from Plugin import PluginManager

import re

@PluginManager.registerTo('UiRequest')
class UiRequestPlugin:
def __init__(self, *args, **kwargs):
from Site import SiteManager
self.site_manager = SiteManager.site_manager

super(UiRequestPlugin, self).__init__(*args, **kwargs)

def actionSiteMedia(self, path, **kwargs):
match = re.match(r'/media/(?P<address>[A-Za-z0-9-]+\.[A-Za-z0-9\.-]+)(?P<inner_path>/.*|$)', path)

if match:
domain = match.group('address')
address = self.site_manager.dns_resolver.resolveDomain(domain)

if address:
path = '/media/' + address + match.group('inner_path')

return super(UiRequestPlugin, self).actionSiteMedia(path, **kwargs)
3 changes: 3 additions & 0 deletions plugins/DNS/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
from . import ConfigPlugin
from . import UiRequestPlugin
from . import SiteManagerPlugin
5 changes: 5 additions & 0 deletions plugins/DNS/plugin_info.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"name": "DNS",
"description": "Support classic DNS as domain system.",
"default": "enabled"
}
2 changes: 2 additions & 0 deletions plugins/DNS/requirements.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
dnspython>=2.0.0,<3.0.0
dnslink>=1.0.1,<2.0.0