Files
aqtinstall/aqt/helper.py
Hiroshi Miura c2750e34f1 Update documents and copyrights
Signed-off-by: Hiroshi Miura <miurahr@linux.com>
2021-01-30 00:28:44 +09:00

156 lines
5.6 KiB
Python

#!/usr/bin/env python
#
# Copyright (C) 2019-2021 Hiroshi Miura <miurahr@linux.com>
#
# Permission is hereby granted, free of charge, to any person obtaining a copy of
# this software and associated documentation files (the "Software"), to deal in
# the Software without restriction, including without limitation the rights to
# use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
# the Software, and to permit persons to whom the Software is furnished to do so,
# subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in all
# copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
# FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
# COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
# IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
import ast
import configparser
import json
import logging
import multiprocessing
import os
import sys
import xml.etree.ElementTree as ElementTree
from typing import List, Optional
import requests
def _get_meta(url: str):
return requests.get(url + '.meta4')
def _check_content_type(ct: str) -> bool:
candidate = ['application/metalink4+xml', 'text/plain']
return any(ct.startswith(t) for t in candidate)
def altlink(url: str, alt: str, logger=None):
"""Blacklisting redirected(alt) location based on Settings.blacklist configuration.
When found black url, then try download a url + .meta4 that is a metalink version4
xml file, parse it and retrieve best alternative url."""
if logger is None:
logger = logging.getLogger(__name__)
blacklist = Settings().blacklist # type: Optional[List[str]]
if blacklist is None or not any(alt.startswith(b) for b in blacklist):
return alt
try:
m = _get_meta(url)
except requests.exceptions.ConnectionError:
logger.error("Got connection error. Fall back to recovery plan...")
return alt
else:
# Expected response->'application/metalink4+xml; charset=utf-8'
if not _check_content_type(m.headers['content-type']):
logger.error("Unexpected meta4 response;content-type: {}".format(m.headers['content-type']))
return alt
try:
mirror_xml = ElementTree.fromstring(m.text)
meta_urls = {}
for f in mirror_xml.iter("{urn:ietf:params:xml:ns:metalink}file"):
for u in f.iter("{urn:ietf:params:xml:ns:metalink}url"):
meta_urls[u.attrib['priority']] = u.text
mirrors = [meta_urls[i] for i in sorted(meta_urls.keys(), key=lambda x: int(x))]
except Exception:
exc_info = sys.exc_info()
logger.error("Unexpected meta4 file; parse error: {}".format(exc_info[1]))
return alt
else:
# Return first priority item which is not blacklist in mirrors list,
# if not found then return alt in default
return next(filter(lambda mirror: not any(mirror.startswith(b) for b in blacklist), mirrors), alt)
class Settings(object):
"""Class to hold configuration and settings.
Actual values are stored in 'settings.ini' file.
It also holds a combinations database.
"""
# this class is Borg/Singleton
_shared_state = {
'_config': None,
'_combinations': None,
'_lock': multiprocessing.Lock()
}
def __init__(self, config=None):
self.__dict__ = self._shared_state
if self._config is None:
with self._lock:
if self._config is None:
if config is None:
self.inifile = os.path.join(os.path.dirname(__file__), 'settings.ini')
else:
self.inifile = config
self._config = self.configParse(self.inifile)
with open(os.path.join(os.path.dirname(__file__), 'combinations.json'), 'r') as j:
self._combinations = json.load(j)[0]
def configParse(self, file_path):
if not os.path.exists(file_path):
raise IOError(file_path)
config = configparser.ConfigParser()
config.read(file_path)
return config
@property
def qt_combinations(self):
return self._combinations['qt']
@property
def tools_combinations(self):
return self._combinations['tools']
@property
def available_versions(self):
return self._combinations['versions']
def available_modules(self, qt_version):
"""Known module names
:returns: dictionary of qt_version and module names
:rtype: List[str]
"""
modules = self._combinations['modules']
versions = qt_version.split('.')
version = "{}.{}".format(versions[0], versions[1])
result = None
for record in modules:
if record["qt_version"] == version:
result = record["modules"]
return result
@property
def concurrency(self):
"""concurrency configuration.
:return: concurrency
:rtype: int
"""
return self._config.getint("aqt", "concurrency")
@property
def blacklist(self):
"""list of sites in a blacklist
:returns: list of site URLs(scheme and host part)
:rtype: List[str]
"""
return ast.literal_eval(self._config.get("mirrors", "blacklist"))