Skip to content

Commit 06ea861

Browse files
committed
v0.1.0
0 parents  commit 06ea861

File tree

11 files changed

+301
-0
lines changed

11 files changed

+301
-0
lines changed

.github/workflows/main.yml

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
name: Unit test
2+
3+
on: [push, pull_request]
4+
5+
jobs:
6+
build:
7+
runs-on: ${{ matrix.os }}
8+
strategy:
9+
matrix:
10+
os: [ubuntu-latest, macos-latest, windows-latest]
11+
python-version: [3.6, 3.7, 3.8, 3.9]
12+
13+
env:
14+
PHONE_VAL_API_KEY: ${{ secrets.API_KEY }}
15+
16+
steps:
17+
- uses: actions/checkout@v2
18+
19+
- name: Install dependencies, and python_core module
20+
run: |
21+
pip3 install -r requirements.txt
22+
python setup.py install --user
23+
24+
- name: Run test
25+
run: |
26+
python3 -m unittest tests/test_python_core.py
27+

.gitignore

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
build
2+
dist
3+
**/__pycache__
4+
python_core.egg-info
5+
dev_test.py
6+
.vscode
7+
*.history

LICENSE

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
MIT License
2+
3+
Copyright (c) 2021 Abstract
4+
5+
Permission is hereby granted, free of charge, to any person obtaining a copy
6+
of this software and associated documentation files (the "Software"), to deal
7+
in the Software without restriction, including without limitation the rights
8+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
copies of the Software, and to permit persons to whom the Software is
10+
furnished to do so, subject to the following conditions:
11+
12+
The above copyright notice and this permission notice shall be included in all
13+
copies or substantial portions of the Software.
14+
15+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21+
SOFTWARE.

README.md

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
# AbstractAPI python-core library
2+
3+
Pure Python library for using [Abstract API](https://www.abstractapi.com/). This is core library that has some shared logic and functions used by other libraries.
4+
5+
# See other
6+
7+
To use and maintain this library, please see the following:
8+
9+
- [Python Email Validation](https://github.com/abstractapi/python-email-validation)
10+
- [Python IP Geolocation](https://github.com/abstractapi/python-ip-geolocation)
11+
- [Python Phone Validation](https://github.com/abstractapi/python-phone-validation)

python_core/__init__.py

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
from .python_core import HttpEndpoint

python_core/exceptions/__init__.py

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
from .exceptions import HttpResponseException, APIException

python_core/exceptions/exceptions.py

+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
class HttpResponseException(Exception):
2+
"""
3+
An Exception caused by bad Http response code
4+
Http codes >= 400
5+
"""
6+
7+
def __init__(self, http_code, msg):
8+
"""
9+
An Exception caused by bad Http response code
10+
Http codes >= 400
11+
12+
Args:
13+
http_code (int): http response status code that caused the error,
14+
will determine the type of exception message
15+
msg (str): the system exec info, traceback is kept and we add msg to it
16+
"""
17+
self.http_code = http_code
18+
self.msg = msg
19+
20+
def __str__(self):
21+
22+
23+
output = f"""
24+
[ HttpResponseException ]
25+
{self.http_code} -- {self.msg}\n
26+
Make sure that : \n
27+
- subdomain is valid\n
28+
- good connection to internet\n
29+
- avoid using VPN
30+
31+
"""
32+
33+
return output
34+
35+
36+
class APIException(Exception):
37+
"""
38+
An Exception caused by an API response with
39+
an error, the recieved error message/description
40+
will be raised
41+
"""
42+
43+
def __init__(self, http_code, msg):
44+
"""
45+
An Exception caused by an API response with
46+
an error, the recieved error message/description
47+
will be raised
48+
49+
Args:
50+
http_code (int): the response http code
51+
msg (str): the system exec info, traceback is kept and we add msg to it
52+
"""
53+
self.http_code = http_code
54+
self.msg = msg
55+
56+
def __str__(self):
57+
58+
if self.http_code == 401:
59+
output = f"""
60+
[ HttpResponseException ]
61+
{self.http_code} -- {self.msg}\n
62+
[[ Unauthorized ]] Make sure that : \n
63+
- API key is valid\n
64+
65+
"""
66+
elif self.http_code == 400:
67+
output = f"""
68+
[ HttpResponseException ]
69+
{self.http_code} -- {self.msg}\n
70+
[[ Bad Request ]] Make sure that : \n
71+
- Request params are valid (valid phone number / valid IP address ...)\n
72+
73+
"""
74+
else:
75+
output = f"""
76+
[ HttpResponseException ]
77+
{self.http_code} -- {self.msg}\n
78+
Make sure that : \n
79+
- subdomain is valid\n
80+
- connection to internet\n
81+
- avoid using VPN
82+
83+
"""
84+
85+
return output
86+

python_core/python_core.py

+57
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
from python_core.exceptions.exceptions import HttpResponseException, APIException
2+
import requests
3+
import sys
4+
5+
6+
class HttpEndpoint:
7+
"""
8+
HttpEndpoint class, one instance per endpoint, offers get/post HTTP methods
9+
and handles both HTTP errors and API errors
10+
"""
11+
12+
13+
def __init__(self, endpoint_subdomain, global_req_params):
14+
"""
15+
HttpEndpoint class, one instance per endpoint, offers get/post HTTP methods
16+
and handles both HTTP errors and API errors
17+
18+
Args:
19+
endpoint (string): endpoint URL
20+
global_req_params (dict): dict of global params included in each request
21+
"""
22+
23+
self.endpoint_subdomain = endpoint_subdomain
24+
self.endpoint = f"https://{endpoint_subdomain}.abstractapi.com/v1/"
25+
self.global_req_params = global_req_params
26+
27+
28+
def get(self, req_params):
29+
"""
30+
Make a get request to the endpoint
31+
32+
Args:
33+
req_params (dict): dict of params included in this request
34+
"""
35+
36+
# requests raises only connection related exceptions, unless exlicitly
37+
# configured to raise Http exceptions
38+
try:
39+
response = requests.get(
40+
url=self.endpoint,
41+
params={
42+
**self.global_req_params,
43+
**req_params
44+
}
45+
)
46+
except requests.exceptions.RequestException as req_exc:
47+
48+
# RequestExceptions are expressive: ConnectionError, ConnectTimeout, InvalidURL, InvalidHeader
49+
# Other Http exceptions are explained with a msg
50+
raise HttpResponseException(-1, sys.exc_info())
51+
52+
if response.ok:
53+
result = response.json()
54+
return result
55+
else:
56+
# Will give a hint to the cause of this exception
57+
raise APIException(response.status_code, sys.exc_info())

requirements.txt

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
certifi==2020.12.5
2+
chardet==4.0.0
3+
idna==2.10
4+
plotly==4.14.3
5+
requests==2.25.1
6+
urllib3==1.26.4
7+
wincertstore==0.2
8+
python-dotenv==0.17.0

setup.py

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
from setuptools import setup, find_packages
2+
3+
# Set the library's long description to the repo's README.md
4+
with open('README.md', 'r') as readme_file:
5+
readme = readme_file.read()
6+
7+
requirements = ['requests>=2']
8+
9+
setup(
10+
name='python_core',
11+
version='0.1.0',
12+
author='Benjamin Bouchet',
13+
author_email='<[email protected]>',
14+
description='HTTP client to call AbstractAPI endpoints.',
15+
long_description=readme,
16+
long_description_content_type='text/markdown',
17+
url='https://github.com/abstractapi/python-core',
18+
packages=find_packages(),
19+
install_requires=requirements,
20+
classifiers=[
21+
'Programming Language :: Python :: 3.6',
22+
'Intended Audience :: Developers',
23+
'License :: OSI Approved :: MIT License',
24+
],
25+
)

tests/test_python_core.py

+57
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
import unittest
2+
import os, sys
3+
from dotenv import load_dotenv
4+
5+
load_dotenv()
6+
7+
PHONE_VAL_API_KEY = os.getenv('PHONE_VAL_API_KEY')
8+
9+
from python_core import HttpEndpoint
10+
11+
class TestHttpEndpoint(unittest.TestCase):
12+
def __init__(self, *args, **kwargs):
13+
14+
super(TestHttpEndpoint, self).__init__(*args, **kwargs)
15+
self.http_endpoint = HttpEndpoint(
16+
"https://phonevalidation.abstractapi.com/v1/", {'lang' : 'python'}
17+
)
18+
19+
def test_no_config(self):
20+
21+
try:
22+
# Some tests can change this to test with a wrong endpoint
23+
# make sure it's right here
24+
self.http_endpoint.endpoint = "https://phonevalidation.abstractapi.com/v1/"
25+
26+
self.http_endpoint.get({
27+
"api_key" : PHONE_VAL_API_KEY,
28+
"phone" : "14154582468",
29+
30+
})
31+
except Exception as e:
32+
self.fail(sys.exc_info())
33+
34+
def test_wrong_endpoint(self):
35+
36+
with self.assertRaises(Exception):
37+
self.http_endpoint.endpoint = "https://www.google.com"
38+
39+
self.http_endpoint.get({
40+
"api_key" : PHONE_VAL_API_KEY,
41+
"phone" : "14154582468"
42+
})
43+
44+
45+
def test_wrong_get_param(self):
46+
47+
with self.assertRaises(Exception):
48+
self.http_endpoint.endpoint = "https://phonevalidation.abstractapi.com/v1/"
49+
50+
self.http_endpoint.get({
51+
"api_key" : PHONE_VAL_API_KEY,
52+
"not_phone" : "14154582468"
53+
})
54+
55+
56+
if __name__ == '__main__':
57+
unittest.main()

0 commit comments

Comments
 (0)