chore: v3.0.0 (#181)
Co-authored-by: Kirill Bobrov <miaplanedo@gmail.com> Co-authored-by: Zhuopsticks <36221214+terryyz@users.noreply.github.com>
This commit is contained in:
parent
2f83668b2f
commit
e10c5e8fa7
4
.gitignore
vendored
4
.gitignore
vendored
@ -60,3 +60,7 @@ target/
|
|||||||
|
|
||||||
.DS_Store
|
.DS_Store
|
||||||
.python-version
|
.python-version
|
||||||
|
|
||||||
|
# IDE
|
||||||
|
.idea
|
||||||
|
.vscode
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
language: python
|
language: python
|
||||||
python:
|
python:
|
||||||
- 3.5
|
|
||||||
- 3.6
|
- 3.6
|
||||||
- 3.7
|
- 3.7
|
||||||
- 3.8
|
- 3.8
|
||||||
|
13
Pipfile
13
Pipfile
@ -4,20 +4,9 @@ verify_ssl = true
|
|||||||
name = "pypi"
|
name = "pypi"
|
||||||
|
|
||||||
[packages]
|
[packages]
|
||||||
requests = "==2.13.0"
|
httpx = "==0.13.3"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
[dev-packages]
|
[dev-packages]
|
||||||
|
|
||||||
coveralls = "*"
|
coveralls = "*"
|
||||||
"pytest-watch" = "*"
|
"pytest-watch" = "*"
|
||||||
"pytest-testmon" = "*"
|
"pytest-testmon" = "*"
|
||||||
|
|
||||||
|
|
||||||
[requires]
|
|
||||||
python_version = "3.7"
|
|
98
Pipfile.lock
generated
98
Pipfile.lock
generated
@ -1,12 +1,10 @@
|
|||||||
{
|
{
|
||||||
"_meta": {
|
"_meta": {
|
||||||
"hash": {
|
"hash": {
|
||||||
"sha256": "bde08e25225ef9645715fdf4240e5d14e20533f2ba2522fe85bb9a1156eaa882"
|
"sha256": "e3f9ee3c7d118b3905fb3797a8b610e4f0485055c0809c93aba126596529673b"
|
||||||
},
|
},
|
||||||
"pipfile-spec": 6,
|
"pipfile-spec": 6,
|
||||||
"requires": {
|
"requires": {},
|
||||||
"python_version": "3.7"
|
|
||||||
},
|
|
||||||
"sources": [
|
"sources": [
|
||||||
{
|
{
|
||||||
"name": "pypi",
|
"name": "pypi",
|
||||||
@ -16,13 +14,90 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"default": {
|
"default": {
|
||||||
"requests": {
|
"certifi": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:1a720e8862a41aa22e339373b526f508ef0c8988baf48b84d3fc891a8e237efb",
|
"sha256:5ad7e9a056d25ffa5082862e36f119f7f7cec6457fa07ee2f8c339814b80c9b1",
|
||||||
"sha256:5722cd09762faa01276230270ff16af7acf7c5c45d623868d9ba116f15791ce8"
|
"sha256:9cd41137dc19af6a5e03b630eefe7d1f458d964d406342dd3edf625839b944cc"
|
||||||
|
],
|
||||||
|
"version": "==2020.4.5.2"
|
||||||
|
},
|
||||||
|
"chardet": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:84ab92ed1c4d4f16916e05906b6b75a6c0fb5db821cc65e70cbd64a3e2a5eaae",
|
||||||
|
"sha256:fc323ffcaeaed0e0a02bf4d117757b98aed530d9ed4531e3e15460124c106691"
|
||||||
|
],
|
||||||
|
"version": "==3.0.4"
|
||||||
|
},
|
||||||
|
"h11": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:33d4bca7be0fa039f4e84d50ab00531047e53d6ee8ffbc83501ea602c169cae1",
|
||||||
|
"sha256:4bc6d6a1238b7615b266ada57e0618568066f57dd6fa967d1290ec9309b2f2f1"
|
||||||
|
],
|
||||||
|
"version": "==0.9.0"
|
||||||
|
},
|
||||||
|
"h2": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:61e0f6601fa709f35cdb730863b4e5ec7ad449792add80d1410d4174ed139af5",
|
||||||
|
"sha256:875f41ebd6f2c44781259005b157faed1a5031df3ae5aa7bcb4628a6c0782f14"
|
||||||
|
],
|
||||||
|
"version": "==3.2.0"
|
||||||
|
},
|
||||||
|
"hpack": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:0edd79eda27a53ba5be2dfabf3b15780928a0dff6eb0c60a3d6767720e970c89",
|
||||||
|
"sha256:8eec9c1f4bfae3408a3f30500261f7e6a65912dc138526ea054f9ad98892e9d2"
|
||||||
|
],
|
||||||
|
"version": "==3.0.0"
|
||||||
|
},
|
||||||
|
"hstspreload": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:2858151b4f77322c6a61312abccead20217b1169ae0855753c0da45da2049329",
|
||||||
|
"sha256:9850f199a4678bbc6a392255948d2435b706bc8a7e40df0f84ac2f7348cf4195"
|
||||||
|
],
|
||||||
|
"version": "==2020.6.5"
|
||||||
|
},
|
||||||
|
"httpcore": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:9850fe97a166a794d7e920590d5ec49a05488884c9fc8b5dba8561effab0c2a0",
|
||||||
|
"sha256:ecc5949310d9dae4de64648a4ce529f86df1f232ce23dcfefe737c24d21dfbe9"
|
||||||
|
],
|
||||||
|
"version": "==0.9.1"
|
||||||
|
},
|
||||||
|
"httpx": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:32d930858eab677bc29a742aaa4f096de259f1c78c68a90ad11f5c3c04f08335",
|
||||||
|
"sha256:3642bd13e90b80ba8a243a730275eb10a4c26ec96f5fc16b87e458d4ab21efae"
|
||||||
],
|
],
|
||||||
"index": "pypi",
|
"index": "pypi",
|
||||||
"version": "==2.13.0"
|
"version": "==0.13.3"
|
||||||
|
},
|
||||||
|
"hyperframe": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:5187962cb16dcc078f23cb5a4b110098d546c3f41ff2d4038a9896893bbd0b40",
|
||||||
|
"sha256:a9f5c17f2cc3c719b917c4f33ed1c61bd1f8dfac4b1bd23b7c80b3400971b41f"
|
||||||
|
],
|
||||||
|
"version": "==5.2.0"
|
||||||
|
},
|
||||||
|
"idna": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:7588d1c14ae4c77d74036e8c22ff447b26d0fde8f007354fd48a7814db15b7cb",
|
||||||
|
"sha256:a068a21ceac8a4d63dbfd964670474107f541babbd2250d61922f029858365fa"
|
||||||
|
],
|
||||||
|
"version": "==2.9"
|
||||||
|
},
|
||||||
|
"rfc3986": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:112398da31a3344dc25dbf477d8df6cb34f9278a94fee2625d89e4514be8bb9d",
|
||||||
|
"sha256:af9147e9aceda37c91a05f4deb128d4b4b49d6b199775fd2d2927768abdc8f50"
|
||||||
|
],
|
||||||
|
"version": "==1.4.0"
|
||||||
|
},
|
||||||
|
"sniffio": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:20ed6d5b46f8ae136d00b9dcb807615d83ed82ceea6b2058cecb696765246da5",
|
||||||
|
"sha256:8e3810100f69fe0edd463d02ad407112542a11ffdc29f67db2bf3771afb87a21"
|
||||||
|
],
|
||||||
|
"version": "==1.1.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"develop": {
|
"develop": {
|
||||||
@ -184,11 +259,10 @@
|
|||||||
},
|
},
|
||||||
"requests": {
|
"requests": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:1a720e8862a41aa22e339373b526f508ef0c8988baf48b84d3fc891a8e237efb",
|
"sha256:43999036bfa82904b6af1d99e4882b560e5e2c68e5c4b0aa03b655f3d7d73fee",
|
||||||
"sha256:5722cd09762faa01276230270ff16af7acf7c5c45d623868d9ba116f15791ce8"
|
"sha256:b3f43d496c6daba4493e7c431722aeb7dbc6288f52a6e04e7b6023b0247817e6"
|
||||||
],
|
],
|
||||||
"index": "pypi",
|
"version": "==2.23.0"
|
||||||
"version": "==2.13.0"
|
|
||||||
},
|
},
|
||||||
"six": {
|
"six": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
|
12
README.rst
12
README.rst
@ -9,7 +9,7 @@ implemented Google Translate API. This uses the `Google Translate Ajax
|
|||||||
API <https://translate.google.com>`__ to make calls to such methods as
|
API <https://translate.google.com>`__ to make calls to such methods as
|
||||||
detect and translate.
|
detect and translate.
|
||||||
|
|
||||||
Compatible with Python 3.5+.
|
Compatible with Python 3.6+.
|
||||||
|
|
||||||
For details refer to the `API
|
For details refer to the `API
|
||||||
Documentation <https://py-googletrans.readthedocs.io/en/latest>`__.
|
Documentation <https://py-googletrans.readthedocs.io/en/latest>`__.
|
||||||
@ -22,7 +22,6 @@ Features
|
|||||||
- Auto language detection
|
- Auto language detection
|
||||||
- Bulk translations
|
- Bulk translations
|
||||||
- Customizable service URL
|
- Customizable service URL
|
||||||
- Connection pooling (the advantage of using requests.Session)
|
|
||||||
- HTTP/2 support
|
- HTTP/2 support
|
||||||
|
|
||||||
TODO
|
TODO
|
||||||
@ -36,11 +35,7 @@ more features are coming soon.
|
|||||||
HTTP/2 support
|
HTTP/2 support
|
||||||
~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~
|
||||||
|
|
||||||
This is a great deal for everyone! (up to 2x times faster in my test) If
|
This library uses httpx for HTTP requests so HTTP/2 is supported by default.
|
||||||
you want to get googletrans faster you should install
|
|
||||||
`hyper <https://github.com/Lukasa/hyper>`__ package. Googletrans will
|
|
||||||
automatically detect if hyper is installed and if so, it will be used
|
|
||||||
for http networking.
|
|
||||||
|
|
||||||
How does this library work
|
How does this library work
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
@ -64,8 +59,7 @@ Installation
|
|||||||
|
|
||||||
To install, either use things like pip with the package "googletrans"
|
To install, either use things like pip with the package "googletrans"
|
||||||
or download the package and put the "googletrans" directory into your
|
or download the package and put the "googletrans" directory into your
|
||||||
python path. Anyway, it is noteworthy that, this just requires two
|
python path.
|
||||||
modules: requests and future.
|
|
||||||
|
|
||||||
.. code:: bash
|
.. code:: bash
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
"""Free Google Translate API for Python. Translates totally free of charge."""
|
"""Free Google Translate API for Python. Translates totally free of charge."""
|
||||||
__all__ = 'Translator',
|
__all__ = 'Translator',
|
||||||
__version__ = '2.4.1'
|
__version__ = '3.0.0'
|
||||||
|
|
||||||
|
|
||||||
from googletrans.client import Translator
|
from googletrans.client import Translator
|
||||||
|
@ -1,16 +0,0 @@
|
|||||||
from requests.adapters import HTTPAdapter
|
|
||||||
|
|
||||||
|
|
||||||
class TimeoutAdapter(HTTPAdapter):
|
|
||||||
"""HTTP adapter that adds timeout to each query."""
|
|
||||||
def __init__(self, timeout=None, *args, **kwargs):
|
|
||||||
"""HTTP adapter that adds timeout to each query.
|
|
||||||
|
|
||||||
:param timeout: Timeout that will be added to each query
|
|
||||||
"""
|
|
||||||
self.timeout = timeout
|
|
||||||
super(TimeoutAdapter, self).__init__(*args, **kwargs)
|
|
||||||
|
|
||||||
def send(self, *args, **kwargs):
|
|
||||||
kwargs['timeout'] = self.timeout
|
|
||||||
return super(TimeoutAdapter, self).send(*args, **kwargs)
|
|
@ -4,16 +4,21 @@ A Translation module.
|
|||||||
|
|
||||||
You can translate text using this module.
|
You can translate text using this module.
|
||||||
"""
|
"""
|
||||||
import requests
|
|
||||||
import random
|
import random
|
||||||
|
import typing
|
||||||
|
|
||||||
|
import httpcore
|
||||||
|
import httpx
|
||||||
|
from httpx import Timeout
|
||||||
|
|
||||||
from googletrans import urls, utils
|
from googletrans import urls, utils
|
||||||
from googletrans.adapters import TimeoutAdapter
|
|
||||||
from googletrans.gtoken import TokenAcquirer
|
from googletrans.gtoken import TokenAcquirer
|
||||||
from googletrans.constants import DEFAULT_USER_AGENT, LANGCODES, LANGUAGES, SPECIAL_CASES
|
from googletrans.constants import (
|
||||||
|
DEFAULT_USER_AGENT, LANGCODES, LANGUAGES, SPECIAL_CASES,
|
||||||
|
DEFAULT_RAISE_EXCEPTION, DUMMY_DATA
|
||||||
|
)
|
||||||
from googletrans.models import Translated, Detected
|
from googletrans.models import Translated, Detected
|
||||||
|
|
||||||
|
|
||||||
EXCLUDES = ('en', 'ca', 'fr')
|
EXCLUDES = ('en', 'ca', 'fr')
|
||||||
|
|
||||||
|
|
||||||
@ -34,33 +39,35 @@ class Translator:
|
|||||||
For example ``{'http': 'foo.bar:3128', 'http://host.name': 'foo.bar:4012'}``
|
For example ``{'http': 'foo.bar:3128', 'http://host.name': 'foo.bar:4012'}``
|
||||||
:type proxies: dictionary
|
:type proxies: dictionary
|
||||||
|
|
||||||
:param timeout: Definition of timeout for Requests library.
|
:param timeout: Definition of timeout for httpx library.
|
||||||
Will be used by every request.
|
Will be used for every request.
|
||||||
:type timeout: number or a double of numbers
|
:type timeout: number or a double of numbers
|
||||||
|
||||||| constructed merge base
|
||||||
|
:param proxies: proxies configuration.
|
||||||
|
Dictionary mapping protocol or protocol and host to the URL of the proxy
|
||||||
|
For example ``{'http': 'foo.bar:3128', 'http://host.name': 'foo.bar:4012'}``
|
||||||
|
:param raise_exception: if `True` then raise exception if smth will go wrong
|
||||||
|
:type raise_exception: boolean
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, service_urls=None, user_agent=DEFAULT_USER_AGENT,
|
def __init__(self, service_urls=None, user_agent=DEFAULT_USER_AGENT,
|
||||||
proxies=None, timeout=None):
|
raise_exception=DEFAULT_RAISE_EXCEPTION,
|
||||||
|
proxies: typing.Dict[str, httpcore.SyncHTTPTransport] = None, timeout: Timeout = None):
|
||||||
|
|
||||||
self.session = requests.Session()
|
self.client = httpx.Client()
|
||||||
if proxies is not None:
|
if proxies is not None: # pragma: nocover
|
||||||
self.session.proxies = proxies
|
self.client.proxies = proxies
|
||||||
self.session.headers.update({
|
|
||||||
|
self.client.headers.update({
|
||||||
'User-Agent': user_agent,
|
'User-Agent': user_agent,
|
||||||
})
|
})
|
||||||
|
|
||||||
if timeout is not None:
|
if timeout is not None:
|
||||||
self.session.mount('https://', TimeoutAdapter(timeout))
|
self.client.timeout = timeout
|
||||||
self.session.mount('http://', TimeoutAdapter(timeout))
|
|
||||||
|
|
||||||
self.service_urls = service_urls or ['translate.google.com']
|
self.service_urls = service_urls or ['translate.google.com']
|
||||||
self.token_acquirer = TokenAcquirer(session=self.session, host=self.service_urls[0])
|
self.token_acquirer = TokenAcquirer(client=self.client, host=self.service_urls[0])
|
||||||
|
self.raise_exception = raise_exception
|
||||||
# Use HTTP2 Adapter if hyper is installed
|
|
||||||
try: # pragma: nocover
|
|
||||||
from hyper.contrib import HTTP20Adapter
|
|
||||||
self.session.mount(urls.BASE, HTTP20Adapter())
|
|
||||||
except ImportError: # pragma: nocover
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _pick_service_url(self):
|
def _pick_service_url(self):
|
||||||
if len(self.service_urls) == 1:
|
if len(self.service_urls) == 1:
|
||||||
@ -73,10 +80,16 @@ class Translator:
|
|||||||
token=token, override=override)
|
token=token, override=override)
|
||||||
|
|
||||||
url = urls.TRANSLATE.format(host=self._pick_service_url())
|
url = urls.TRANSLATE.format(host=self._pick_service_url())
|
||||||
r = self.session.get(url, params=params)
|
r = self.client.get(url, params=params)
|
||||||
|
|
||||||
data = utils.format_json(r.text)
|
if r.status_code == 200:
|
||||||
return data
|
data = utils.format_json(r.text)
|
||||||
|
return data
|
||||||
|
else:
|
||||||
|
if self.raise_exception:
|
||||||
|
raise Exception('Unexpected status code "{}" from {}'.format(r.status_code, self.service_urls))
|
||||||
|
DUMMY_DATA[0][0][0] = text
|
||||||
|
return DUMMY_DATA
|
||||||
|
|
||||||
def _parse_extra_data(self, data):
|
def _parse_extra_data(self, data):
|
||||||
response_parts_name_mapping = {
|
response_parts_name_mapping = {
|
||||||
@ -189,7 +202,7 @@ class Translator:
|
|||||||
if pron is None:
|
if pron is None:
|
||||||
try:
|
try:
|
||||||
pron = data[0][1][2]
|
pron = data[0][1][2]
|
||||||
except: # pragma: nocover
|
except: # pragma: nocover
|
||||||
pass
|
pass
|
||||||
|
|
||||||
if dest in EXCLUDES and pron == origin:
|
if dest in EXCLUDES and pron == origin:
|
||||||
|
@ -182,3 +182,6 @@ LANGUAGES = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
LANGCODES = dict(map(reversed, LANGUAGES.items()))
|
LANGCODES = dict(map(reversed, LANGUAGES.items()))
|
||||||
|
DEFAULT_RAISE_EXCEPTION = False
|
||||||
|
DUMMY_DATA = [[["", None, None, 0]], None, "en", None,
|
||||||
|
None, None, 1, None, [["en"], None, [1], ["en"]]]
|
||||||
|
@ -4,7 +4,7 @@ import math
|
|||||||
import re
|
import re
|
||||||
import time
|
import time
|
||||||
|
|
||||||
import requests
|
import httpx
|
||||||
|
|
||||||
from googletrans.utils import rshift
|
from googletrans.utils import rshift
|
||||||
|
|
||||||
@ -38,8 +38,8 @@ class TokenAcquirer:
|
|||||||
RE_TKK = re.compile(r'tkk:\'(.+?)\'', re.DOTALL)
|
RE_TKK = re.compile(r'tkk:\'(.+?)\'', re.DOTALL)
|
||||||
RE_RAWTKK = re.compile(r'tkk:\'(.+?)\'', re.DOTALL)
|
RE_RAWTKK = re.compile(r'tkk:\'(.+?)\'', re.DOTALL)
|
||||||
|
|
||||||
def __init__(self, tkk='0', session=None, host='translate.google.com'):
|
def __init__(self, tkk='0', client: httpx.Client = None, host='translate.google.com'):
|
||||||
self.session = session or requests.Session()
|
self.client = client or httpx.Client()
|
||||||
self.tkk = tkk
|
self.tkk = tkk
|
||||||
self.host = host if 'http' in host else 'https://' + host
|
self.host = host if 'http' in host else 'https://' + host
|
||||||
|
|
||||||
@ -51,7 +51,7 @@ class TokenAcquirer:
|
|||||||
if self.tkk and int(self.tkk.split('.')[0]) == now:
|
if self.tkk and int(self.tkk.split('.')[0]) == now:
|
||||||
return
|
return
|
||||||
|
|
||||||
r = self.session.get(self.host)
|
r = self.client.get(self.host)
|
||||||
|
|
||||||
raw_tkk = self.RE_TKK.search(r.text)
|
raw_tkk = self.RE_TKK.search(r.text)
|
||||||
if raw_tkk:
|
if raw_tkk:
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
"""A conversion module for googletrans"""
|
"""A conversion module for googletrans"""
|
||||||
import re
|
|
||||||
import json
|
import json
|
||||||
|
import re
|
||||||
|
|
||||||
|
|
||||||
def build_params(query, src, dest, token, override):
|
def build_params(query, src, dest, token, override):
|
||||||
|
6
setup.py
6
setup.py
@ -52,18 +52,14 @@ def install():
|
|||||||
'Operating System :: MacOS :: MacOS X',
|
'Operating System :: MacOS :: MacOS X',
|
||||||
'Topic :: Education',
|
'Topic :: Education',
|
||||||
'Programming Language :: Python',
|
'Programming Language :: Python',
|
||||||
'Programming Language :: Python :: 3.5',
|
|
||||||
'Programming Language :: Python :: 3.6',
|
'Programming Language :: Python :: 3.6',
|
||||||
'Programming Language :: Python :: 3.7',
|
'Programming Language :: Python :: 3.7',
|
||||||
'Programming Language :: Python :: 3.8'],
|
'Programming Language :: Python :: 3.8'],
|
||||||
packages=find_packages(exclude=['docs', 'tests']),
|
packages=find_packages(exclude=['docs', 'tests']),
|
||||||
keywords='google translate translator',
|
keywords='google translate translator',
|
||||||
install_requires=[
|
install_requires=[
|
||||||
'requests',
|
'httpx==0.13.3',
|
||||||
],
|
],
|
||||||
extras_require={
|
|
||||||
'h2': ['hyper'],
|
|
||||||
},
|
|
||||||
tests_require=[
|
tests_require=[
|
||||||
'pytest',
|
'pytest',
|
||||||
'coveralls',
|
'coveralls',
|
||||||
|
@ -1,3 +1 @@
|
|||||||
requests==2.13.0
|
|
||||||
future==0.14.3
|
|
||||||
coveralls==1.1
|
coveralls==1.1
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
# -*- coding: utf-8 -*-
|
from httpcore import TimeoutException
|
||||||
|
from httpcore._exceptions import ConnectError
|
||||||
|
from httpx import Timeout, Client
|
||||||
|
from unittest.mock import patch
|
||||||
from pytest import raises
|
from pytest import raises
|
||||||
from requests.exceptions import ConnectionError
|
|
||||||
from requests.exceptions import ReadTimeout
|
|
||||||
|
|
||||||
from googletrans import Translator
|
from googletrans import Translator
|
||||||
|
|
||||||
@ -127,20 +128,21 @@ def test_dest_not_in_supported_languages(translator):
|
|||||||
translator.translate(*args)
|
translator.translate(*args)
|
||||||
|
|
||||||
|
|
||||||
def test_connection_timeout():
|
def test_timeout():
|
||||||
# Requests library specifies two timeouts: connection and read
|
# httpx will raise ConnectError in some conditions
|
||||||
|
with raises((TimeoutException, ConnectError)):
|
||||||
with raises((ConnectionError, ReadTimeout)):
|
translator = Translator(timeout=Timeout(0.0001))
|
||||||
"""If a number is passed to timeout parameter, both connection
|
|
||||||
and read timeouts will be set to it.
|
|
||||||
Firstly, the connection timeout will fail.
|
|
||||||
"""
|
|
||||||
translator = Translator(timeout=0.00001)
|
|
||||||
translator.translate('안녕하세요.')
|
translator.translate('안녕하세요.')
|
||||||
|
|
||||||
|
|
||||||
def test_read_timeout():
|
class MockResponse:
|
||||||
|
def __init__(self, status_code):
|
||||||
|
self.status_code = status_code
|
||||||
|
self.text = 'tkk:\'translation\''
|
||||||
|
|
||||||
|
|
||||||
|
@patch.object(Client, 'get', return_value=MockResponse('403'))
|
||||||
|
def test_403_error(session_mock):
|
||||||
|
translator = Translator()
|
||||||
|
assert translator.translate('test', dest='ko')
|
||||||
|
|
||||||
with raises(ReadTimeout):
|
|
||||||
translator = Translator(timeout=(10, 0.00001))
|
|
||||||
translator.translate('안녕하세요.')
|
|
||||||
|
@ -5,16 +5,18 @@ from pytest import raises
|
|||||||
def test_format_json():
|
def test_format_json():
|
||||||
text = '[,,"en",,,,0.96954316,,[["en"],,[0.96954316]]]'
|
text = '[,,"en",,,,0.96954316,,[["en"],,[0.96954316]]]'
|
||||||
|
|
||||||
result = utils.legacy_format_json(text)
|
result = utils.format_json(text)
|
||||||
|
|
||||||
assert result == [None, None, 'en', None, None, None, 0.96954316, None,
|
assert result == [None, None, 'en', None, None, None, 0.96954316, None,
|
||||||
[['en'], None, [0.96954316]]]
|
[['en'], None, [0.96954316]]]
|
||||||
|
|
||||||
|
|
||||||
def test_format_malformed_json():
|
def test_format_malformed_json():
|
||||||
text = '[,,"en",,,,0.96954316,,[["en"],,0.96954316]]]'
|
text = '[,,"en",,,,0.96954316,,[["en"],,0.96954316]]]'
|
||||||
|
|
||||||
with raises(ValueError):
|
with raises(ValueError):
|
||||||
utils.legacy_format_json(text)
|
utils.format_json(text)
|
||||||
|
|
||||||
|
|
||||||
def test_rshift():
|
def test_rshift():
|
||||||
value, n = 1000, 3
|
value, n = 1000, 3
|
||||||
@ -22,3 +24,17 @@ def test_rshift():
|
|||||||
result = utils.rshift(value, n)
|
result = utils.rshift(value, n)
|
||||||
|
|
||||||
assert result == 125
|
assert result == 125
|
||||||
|
|
||||||
|
|
||||||
|
def test_build_params_with_override():
|
||||||
|
params = utils.build_params(
|
||||||
|
query='',
|
||||||
|
src='',
|
||||||
|
dest='',
|
||||||
|
token='',
|
||||||
|
override={
|
||||||
|
'otf': '3',
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
assert params['otf'] == '3'
|
||||||
|
3
tox.ini
3
tox.ini
@ -1,10 +1,11 @@
|
|||||||
[tox]
|
[tox]
|
||||||
envlist = py35,py36,py37,py38,pypy3
|
envlist = py36,py37,py38,pypy3
|
||||||
|
|
||||||
[testenv]
|
[testenv]
|
||||||
deps=
|
deps=
|
||||||
pytest
|
pytest
|
||||||
pytest-cov
|
pytest-cov
|
||||||
|
mock
|
||||||
commands=
|
commands=
|
||||||
py.test --cov-report= --cov={envsitepackagesdir}/googletrans {posargs:}
|
py.test --cov-report= --cov={envsitepackagesdir}/googletrans {posargs:}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user