Skip to content

Commit

Permalink
Initial commit using urllib.
Browse files Browse the repository at this point in the history
  • Loading branch information
amh-mw committed Dec 16, 2024
1 parent e2b3f24 commit e9b26a5
Show file tree
Hide file tree
Showing 5 changed files with 180 additions and 0 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
__pycache__
17 changes: 17 additions & 0 deletions test.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
#!/bin/sh -e
usage() {
echo "Usage: $0 [-h]" 1>&2;
echo " -h Display this help message." 1>&2;
exit 1;
}

while getopts "hi" opt; do
case $opt in
*)
usage
;;
esac
done

PYTHON=python3.9
$PYTHON -m unittest
1 change: 1 addition & 0 deletions tests/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
from . import test_vlamb
30 changes: 30 additions & 0 deletions tests/test_vlamb.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
import os
import unittest

from vlamb import Vtapi, VtapiError


class TestVtapi(unittest.TestCase):
@classmethod
def setUpClass(cls):
if not all(key in os.environ for key in ['VTIGER_HOST', 'VTIGER_USER', 'VTIGER_PASS']):
cls.skipTest("environment variables not configured")

def test_login(self):
with Vtapi(os.environ['VTIGER_HOST']) as api:
api.login(os.environ['VTIGER_USER'], os.environ['VTIGER_PASS'])

def test_login_fail(self):
with Vtapi(os.environ['VTIGER_HOST']) as api, self.assertRaises(VtapiError) as fail:
api.login(os.environ['VTIGER_USER'], 'bogus')
self.assertEqual("INVALID_USER_CREDENTIALS: Invalid username or password", str(fail.exception))

def test_count(self):
with Vtapi(os.environ['VTIGER_HOST']) as api:
api.login(os.environ['VTIGER_USER'], os.environ['VTIGER_PASS'])
self.assertGreater(api.count('Quotes'), 0)

def test_retrieve(self):
with Vtapi(os.environ['VTIGER_HOST']) as api:
api.login(os.environ['VTIGER_USER'], os.environ['VTIGER_PASS'])
self.assertIn('website', api.retrieve('CompanyDetails')[0])
131 changes: 131 additions & 0 deletions vlamb.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,131 @@
import hashlib
import json
import logging
import urllib.parse
import urllib.request

_logger = logging.getLogger(__name__)


class Vtapi:
def __init__(self, url):
self.session_name = None
self.url = url + '/webservice.php'

def __enter__(self):
return self

def __exit__(self, type, value, traceback):
try:
self.logout()
finally:
pass

def count(self, module):
query = f"select count(*) from {module};"
result = self.query(query)
return int(result[0]['count'])

def create(self, module, values):
data = {
'operation': 'create',
'sessionName': self.session_name,
'elementType': module,
'element': json.dumps(values),
}
with self._urlopen(self.url, data=data) as response:
return self._result(response)

def download(self, id):
params = {
'operation': 'download',
'sessionName': self.session_name,
'id': id,
}
with self._urlopen(self.url, params=params) as response:
return self._result(response)

def listtypes(self):
params = {
'operation': 'listtypes',
'sessionName': self.session_name,
}
with self._urlopen(self.url, params=params) as response:
return self._result(response)

def login(self, username, accesskey):
token = self._getchallenge(username)
self._login(username, token, accesskey)

def logout(self):
if self.session_name:
data = {
'operation': 'logout',
'sessionName': self.session_name,
}
self._urlopen(self.url, data=data)
self.session_name = None
self.user_id = None

def query(self, query):
params = {
'operation': 'query',
'sessionName': self.session_name,
'query': query,
}
try:
with self._urlopen(self.url, params=params) as response:
return self._result(response)
except:
_logger.error("failed to query '%s'", query)
raise

def retrieve(self, module, limit=0, offset=0):
query = f"select * from {module};"
if limit or offset:
query = query[:-1] + f" limit {offset}, {limit};"
return self.query(query)

def _getchallenge(self, username):
params = {
'operation': 'getchallenge',
'username': username,
}
with self._urlopen(self.url, params=params) as response:
result = self._result(response)
token = result['token']
return token

def _login(self, username, token, accesskey):
hasher = hashlib.md5()
hasher.update(token.encode('utf-8'))
hasher.update(accesskey.encode('utf-8'))
data = {
'operation': 'login',
'username': username,
'accessKey': hasher.hexdigest(),
}
with self._urlopen(self.url, data=data) as response:
body = self._result(response)
self.session_name, self.user_id = result['sessionName'], result['userId']

def _result(self, response):
body = json.loads(response.read().decode('utf-8'))
if not body['success']:
raise VtapiError(**body['error'], status=response.status)
return body['result']

def _urlopen(self, url, data=None, params=None):
if params:
url = url + '?' + urllib.parse.urlencode(params)
if data:
data = urllib.parse.urlencode(data).encode('utf-8')
return urllib.request.urlopen(url, data=data)


class VtapiError(Exception):
def __init__(self, code, message, status=None):
self.code = code
self.message = message
self.status = status
super().__init__(f"{self.code}: {self.message}")

0 comments on commit e9b26a5

Please sign in to comment.