2017-08-07 01:10:12 +02:00
|
|
|
#!/usr/bin/env python3
|
2017-08-13 02:29:57 +02:00
|
|
|
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
|
2017-08-07 01:10:12 +02:00
|
|
|
|
|
|
|
# Copyright 2017 Michal Siedlaczek <michal.siedlaczek@gmail.com>
|
|
|
|
|
|
|
|
# This file is part of qutebrowser.
|
|
|
|
#
|
|
|
|
# qutebrowser is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# qutebrowser is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with qutebrowser. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
"""A script installing Hunspell dictionaries.
|
|
|
|
|
|
|
|
Use: python -m scripts.install_dict [--list] [lang [lang [...]]]
|
|
|
|
"""
|
|
|
|
|
|
|
|
import argparse
|
2017-10-04 01:54:54 +02:00
|
|
|
import base64
|
|
|
|
import json
|
|
|
|
import os
|
2017-08-07 01:10:12 +02:00
|
|
|
import sys
|
2017-10-04 01:54:54 +02:00
|
|
|
import re
|
|
|
|
import urllib.parse
|
|
|
|
import urllib.request
|
2017-10-05 15:56:13 +02:00
|
|
|
|
2017-10-04 01:54:54 +02:00
|
|
|
import attr
|
2017-08-07 01:10:12 +02:00
|
|
|
|
2017-10-05 15:56:13 +02:00
|
|
|
sys.path.insert(0, os.path.join(os.path.dirname(__file__), os.pardir))
|
2017-08-07 01:10:12 +02:00
|
|
|
from qutebrowser.browser.webengine import spell
|
2017-10-04 01:54:54 +02:00
|
|
|
from qutebrowser.config import configdata
|
|
|
|
|
|
|
|
|
|
|
|
API_URL = 'https://chromium.googlesource.com/chromium/deps/hunspell_dictionaries.git/+/master/'
|
|
|
|
|
|
|
|
|
2017-10-04 15:22:35 +02:00
|
|
|
class InvalidLanguageError(Exception):
|
2017-10-05 15:56:13 +02:00
|
|
|
|
|
|
|
"""Raised when requesting invalid languages."""
|
2017-10-04 15:22:35 +02:00
|
|
|
|
|
|
|
def __init__(self, invalid_langs):
|
|
|
|
msg = 'invalid languages: {}'.format(', '.join(invalid_langs))
|
2017-10-05 15:56:13 +02:00
|
|
|
super().__init__(msg)
|
2017-10-04 15:22:35 +02:00
|
|
|
|
|
|
|
|
2017-10-04 01:54:54 +02:00
|
|
|
@attr.s
|
|
|
|
class Language:
|
2017-10-05 15:56:13 +02:00
|
|
|
|
2017-10-04 01:54:54 +02:00
|
|
|
"""Dictionary language specs."""
|
|
|
|
|
|
|
|
code = attr.ib(None)
|
|
|
|
name = attr.ib(None)
|
|
|
|
file_basename = attr.ib(None)
|
|
|
|
file_extension = attr.ib('bdic')
|
|
|
|
|
|
|
|
@property
|
|
|
|
def file_path(self):
|
|
|
|
return '.'.join([self.file_basename, self.file_extension])
|
2017-08-07 01:10:12 +02:00
|
|
|
|
|
|
|
|
|
|
|
def get_argparser():
|
|
|
|
"""Get the argparse parser."""
|
|
|
|
desc = 'Install Hunspell dictionaries for QtWebEngine.'
|
|
|
|
parser = argparse.ArgumentParser(prog='install_dict',
|
|
|
|
description=desc)
|
|
|
|
parser.add_argument('-l', '--list', action='store_true',
|
|
|
|
help="Display the list of available languages.")
|
|
|
|
parser.add_argument('languages', nargs='*',
|
|
|
|
help="A list of languages to install.")
|
|
|
|
return parser
|
|
|
|
|
|
|
|
|
|
|
|
def print_list(languages):
|
|
|
|
for lang in languages:
|
2017-10-04 15:22:35 +02:00
|
|
|
print(lang.code, lang.name, sep='\t')
|
2017-08-07 01:10:12 +02:00
|
|
|
|
|
|
|
|
2017-10-04 01:54:54 +02:00
|
|
|
def valid_languages():
|
|
|
|
"""Return a mapping from valid language codes to their names."""
|
|
|
|
option = configdata.DATA['spellcheck.languages']
|
|
|
|
return option.typ.valtype.valid_values.descriptions
|
|
|
|
|
|
|
|
|
|
|
|
def language_list_from_api():
|
|
|
|
"""Return a JSON with a list of available languages from Google API."""
|
|
|
|
listurl = urllib.parse.urljoin(API_URL, '?format=JSON')
|
|
|
|
response = urllib.request.urlopen(listurl)
|
2017-10-04 16:06:14 +02:00
|
|
|
# A special 5-byte prefix must be stripped from the response content
|
|
|
|
# See: https://github.com/google/gitiles/issues/22
|
|
|
|
# https://github.com/google/gitiles/issues/82
|
|
|
|
json_content = response.read()[5:]
|
|
|
|
entries = json.loads(json_content.decode('utf-8'))['entries']
|
2017-10-04 01:54:54 +02:00
|
|
|
return entries
|
|
|
|
|
|
|
|
|
|
|
|
def available_languages():
|
|
|
|
"""Return a list of Language objects of all available languages."""
|
|
|
|
lang_map = valid_languages()
|
|
|
|
api_list = language_list_from_api()
|
|
|
|
dict_re = re.compile(r"""
|
|
|
|
(?P<filename>(?P<dict>[a-z]{2}(-[A-Z]{2})?).*)\.bdic
|
|
|
|
""", re.VERBOSE)
|
|
|
|
code2file = {}
|
|
|
|
for lang in api_list:
|
|
|
|
match = dict_re.match(lang['name'])
|
|
|
|
if match is not None:
|
|
|
|
code2file[match.group('dict')] = match.group('filename')
|
|
|
|
return [
|
|
|
|
Language(code, name, code2file[code])
|
|
|
|
for code, name in lang_map.items()
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
def download_dictionary(url, dest):
|
|
|
|
"""Download a decoded dictionary file."""
|
|
|
|
response = urllib.request.urlopen(url)
|
|
|
|
decoded = base64.decodebytes(response.read())
|
|
|
|
with open(dest, 'bw') as dict_file:
|
|
|
|
dict_file.write(decoded)
|
|
|
|
|
|
|
|
|
|
|
|
def filter_languages(languages, selected):
|
|
|
|
"""Filter a list of languages based on an inclusion list.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
languages: a list of languages to filter
|
|
|
|
selected: a list of keys to select
|
|
|
|
"""
|
|
|
|
filtered_languages = []
|
|
|
|
for language in languages:
|
|
|
|
if language.code in selected:
|
|
|
|
filtered_languages.append(language)
|
|
|
|
selected.remove(language.code)
|
|
|
|
if selected:
|
2017-10-04 15:22:35 +02:00
|
|
|
raise InvalidLanguageError(selected)
|
2017-10-04 01:54:54 +02:00
|
|
|
return filtered_languages
|
|
|
|
|
|
|
|
|
|
|
|
def install_lang(lang):
|
|
|
|
"""Install a single lang given by the argument."""
|
|
|
|
print('Installing {}: {}'.format(lang.code, lang.name))
|
|
|
|
lang_url = urllib.parse.urljoin(API_URL, lang.file_path, '?format=TEXT')
|
|
|
|
if not os.path.isdir(spell.dictionary_dir()):
|
|
|
|
warn_msg = 'WARN: {} does not exist, creating the directory'
|
|
|
|
print(warn_msg.format(spell.dictionary_dir()))
|
|
|
|
os.makedirs(spell.dictionary_dir())
|
|
|
|
print('Downloading {}'.format(lang_url))
|
|
|
|
dest = os.path.join(spell.dictionary_dir(), lang.file_path)
|
|
|
|
download_dictionary(lang_url, dest)
|
|
|
|
print('Done.')
|
|
|
|
|
|
|
|
|
|
|
|
def install(languages):
|
|
|
|
"""Install languages."""
|
|
|
|
for lang in languages:
|
|
|
|
try:
|
|
|
|
install_lang(lang)
|
|
|
|
except PermissionError as e:
|
|
|
|
print(e)
|
2017-10-05 15:56:13 +02:00
|
|
|
sys.exit(1)
|
2017-10-04 01:54:54 +02:00
|
|
|
|
|
|
|
|
2017-08-07 01:10:12 +02:00
|
|
|
def main():
|
2017-10-04 01:54:54 +02:00
|
|
|
if configdata.DATA is None:
|
|
|
|
configdata.init()
|
2017-08-07 01:10:12 +02:00
|
|
|
parser = get_argparser()
|
|
|
|
argv = sys.argv[1:]
|
|
|
|
args = parser.parse_args(argv)
|
2017-10-04 01:54:54 +02:00
|
|
|
languages = available_languages()
|
2017-08-07 01:10:12 +02:00
|
|
|
if args.list:
|
|
|
|
print_list(languages)
|
|
|
|
elif not args.languages:
|
|
|
|
parser.print_usage()
|
|
|
|
else:
|
|
|
|
try:
|
2017-10-04 01:54:54 +02:00
|
|
|
install(filter_languages(languages, args.languages))
|
2017-10-04 15:22:35 +02:00
|
|
|
except InvalidLanguageError as e:
|
2017-08-07 01:10:12 +02:00
|
|
|
print(e)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|