Skip to content

Commit bd4c999

Browse files
authored
feat: add the code (#2)
* feat: Convert go-sdk to python-sdk * feat: Build CI scripts and refine test cases
1 parent 37f1c60 commit bd4c999

14 files changed

+834
-0
lines changed

.github/workflows/build.yml

+69
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
name: build
2+
3+
on:
4+
push:
5+
branches: [ master ]
6+
pull_request:
7+
branches: [ master ]
8+
9+
jobs:
10+
test:
11+
name: Test
12+
runs-on: ubuntu-latest
13+
strategy:
14+
matrix:
15+
python-version: ['3.8', '3.9', '3.10', '3.11']
16+
17+
steps:
18+
- name: Checkout
19+
uses: actions/checkout@v2
20+
21+
- name: Set up Python ${{ matrix.python-version }}
22+
uses: actions/setup-python@v2
23+
with:
24+
python-version: ${{ matrix.python-version }}
25+
26+
- name: Install dependencies
27+
run: |
28+
python -m pip install --upgrade pip
29+
pip install black ruff pre-commit
30+
pip install -r requirements.txt
31+
pip install .
32+
33+
- name: Run linter
34+
run: git diff --name-only HEAD~10 HEAD | xargs pre-commit run --files
35+
36+
- name: Run tests
37+
run: python -m unittest discover src/tests -v
38+
39+
release:
40+
name: Release
41+
needs: [test]
42+
runs-on: ubuntu-latest
43+
steps:
44+
- name: Checkout
45+
uses: actions/checkout@v2
46+
with:
47+
fetch-depth: 0
48+
49+
- name: Setup Node.js
50+
uses: actions/setup-node@v2
51+
with:
52+
node-version: '20'
53+
54+
- name: Setup semantic-release
55+
run: npm install -g semantic-release @semantic-release/github @semantic-release/changelog @semantic-release/commit-analyzer @semantic-release/git @semantic-release/release-notes-generator semantic-release-pypi
56+
57+
- name: Set up Python
58+
uses: actions/setup-python@v2
59+
with:
60+
python-version: '3.8'
61+
62+
- name: Install build tools
63+
run: python -m pip install --upgrade setuptools wheel twine build
64+
65+
- name: Release
66+
env:
67+
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
68+
PYPI_TOKEN: ${{ secrets.PYPI_TOKEN }}
69+
run: npx semantic-release

pyproject.toml

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
[project]
2+
name = "casvisor-python-sdk"
3+
version = "0.1.0"
4+
description = "Python SDK for Casvisor"
5+
readme = "README.md"
6+
requires-python = ">=3.8"
7+
authors = [
8+
{ name = "Casvisor", email = "[email protected]" }
9+
]
10+
license = { file = "LICENSE" }
11+
classifiers = [
12+
"Intended Audience :: Developers",
13+
"License :: OSI Approved :: Apache Software License",
14+
"Operating System :: OS Independent",
15+
"Programming Language :: Python :: 3",
16+
"Programming Language :: Python :: 3.8",
17+
"Programming Language :: Python :: 3.9",
18+
"Programming Language :: Python :: 3.10",
19+
"Programming Language :: Python :: 3.11",
20+
]
21+
22+
dynamic = ["dependencies"]
23+
[tool.setuptools.dynamic]
24+
dependencies = {file = ["requirements.txt"]}
25+
26+
[project.urls]
27+
"Home" = "https://github.com/casvisor/casvisor-python-sdk"
28+
29+
[build-system]
30+
requires = ["setuptools", "wheel"]
31+
build-backend = "setuptools.build_meta"

requirements.txt

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
requests~=2.32.0
2+
requests_toolbelt
3+
requests-mock

setup.cfg

+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
[metadata]
2+
name = casvisor
3+
version = 0.1.0
4+
author = Casvisor
5+
author_email = [email protected]
6+
url = https://github.com/casvisor/casvisor-python-sdk
7+
description = Python SDK built for Casvisor
8+
long_description = file: README.md
9+
long_description_content_type = text/markdown
10+
license = Apache 2.0
11+
platform = any
12+
keywords = Casvisor
13+
classifiers =
14+
Intended Audience :: Developers
15+
License :: OSI Approved :: Apache Software License
16+
Operating System :: OS Independent
17+
Programming Language :: Python
18+
Programming Language :: Python :: 3.8
19+
Programming Language :: Python :: 3.9
20+
Programming Language :: Python :: 3.10
21+
Programming Language :: Python :: 3.11
22+
23+
[options]
24+
package_dir =
25+
=src
26+
packages =
27+
casvisor
28+
setup_requires =
29+
setuptools
30+
python_requires = >=3.8
31+
test_suite = tests
32+
33+
[bdist_wheel]
34+
universal = true
35+
36+
[sdist]
37+
formats = gztar

setup.py

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
# Copyright 2025 The casbin Authors. All Rights Reserved.
2+
#
3+
# Licensed under the Apache License, Version 2.0 (the "License");
4+
# you may not use this file except in compliance with the License.
5+
# You may obtain a copy of the License at
6+
#
7+
# http://www.apache.org/licenses/LICENSE-2.0
8+
#
9+
# Unless required by applicable law or agreed to in writing, software
10+
# distributed under the License is distributed on an "AS IS" BASIS,
11+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
# See the License for the specific language governing permissions and
13+
# limitations under the License.
14+
15+
from setuptools import setup
16+
17+
setup()
18+

src/casvisor/__init__.py

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
# Copyright 2025 The casbin Authors. All Rights Reserved.
2+
#
3+
# Licensed under the Apache License, Version 2.0 (the "License");
4+
# you may not use this file except in compliance with the License.
5+
# You may obtain a copy of the License at
6+
#
7+
# http://www.apache.org/licenses/LICENSE-2.0
8+
#
9+
# Unless required by applicable law or agreed to in writing, software
10+
# distributed under the License is distributed on an "AS IS" BASIS,
11+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
# See the License for the specific language governing permissions and
13+
# limitations under the License.
14+
#
15+
from .main import CasvisorSDK
16+
from .base import BaseClient,Response
17+
from .record import Record, _RecordSDK
18+
19+
__all__ = ["CasvisorSDK","BaseClient", "Response", "Record", "_RecordSDK"]

src/casvisor/base.py

+99
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,99 @@
1+
# Copyright 2025 The casbin Authors. All Rights Reserved.
2+
#
3+
# Licensed under the Apache License, Version 2.0 (the "License");
4+
# you may not use this file except in compliance with the License.
5+
# You may obtain a copy of the License at
6+
#
7+
# http://www.apache.org/licenses/LICENSE-2.0
8+
#
9+
# Unless required by applicable law or agreed to in writing, software
10+
# distributed under the License is distributed on an "AS IS" BASIS,
11+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
# See the License for the specific language governing permissions and
13+
# limitations under the License.
14+
15+
import json
16+
from typing import Dict, Union, List, Tuple
17+
import requests
18+
from . import util
19+
20+
21+
class HttpClient:
22+
def do(self, request):
23+
pass
24+
25+
class Response:
26+
def __init__(self, status: str, msg: str, data: Union[Dict, List], data2: Union[Dict, List]):
27+
self.status = status
28+
self.msg = msg
29+
self.data = data
30+
self.data2 = data2
31+
32+
# Global HTTP client
33+
client = requests.Session()
34+
35+
36+
def set_http_client(http_client: HttpClient):
37+
global client
38+
client = http_client
39+
40+
41+
class BaseClient:
42+
def __init__(self, client_id: str, client_secret: str, endpoint: str):
43+
self.client_id = client_id
44+
self.client_secret = client_secret
45+
self.endpoint = endpoint
46+
47+
def do_get_response(self, url: str) -> Response:
48+
resp_bytes = self.do_get_bytes_raw_without_check(url)
49+
response = json.loads(resp_bytes)
50+
if response["status"] != "ok":
51+
raise Exception(response["msg"])
52+
return Response(response["status"], response["msg"], response["data"], response["data2"])
53+
54+
def do_get_bytes(self, url: str) -> bytes:
55+
response = self.do_get_response(url)
56+
return json.dumps(response.data).encode("utf-8")
57+
58+
def do_get_bytes_raw(self, url: str) -> bytes:
59+
resp_bytes = self.do_get_bytes_raw_without_check(url)
60+
response = json.loads(resp_bytes)
61+
if response["status"] == "error":
62+
raise Exception(response["msg"])
63+
return resp_bytes
64+
65+
def do_post(self, action: str, query_map: Dict[str, str], post_bytes: bytes, is_form: bool, is_file: bool) -> Response:
66+
url = util.get_url(self.endpoint, action, query_map)
67+
content_type, body = self.prepare_body(post_bytes, is_form, is_file)
68+
resp_bytes = self.do_post_bytes_raw(url, content_type, body)
69+
response = json.loads(resp_bytes)
70+
if response["status"] != "ok":
71+
raise Exception(response["msg"])
72+
return Response(response["status"], response["msg"], response["data"], response["data2"])
73+
74+
def do_post_bytes_raw(self, url: str, content_type: str, body: bytes) -> bytes:
75+
if not content_type:
76+
content_type = "text/plain;charset=UTF-8"
77+
headers = {
78+
"Content-Type": content_type,
79+
"Authorization": f"Basic {self.client_id}:{self.client_secret}"
80+
}
81+
resp = client.post(url, headers=headers, data=body)
82+
return resp.content
83+
84+
def do_get_bytes_raw_without_check(self, url: str) -> bytes:
85+
headers = {
86+
"Authorization": f"Basic {self.client_id}:{self.client_secret}"
87+
}
88+
resp = client.get(url, headers=headers)
89+
return resp.content
90+
91+
def prepare_body(self, post_bytes: bytes, is_form: bool, is_file: bool) -> Tuple[str, bytes]:
92+
if is_form:
93+
if is_file:
94+
return util.create_form_file({"file": post_bytes})
95+
else:
96+
params = json.loads(post_bytes)
97+
return util.create_form(params)
98+
else:
99+
return "text/plain;charset=UTF-8", post_bytes

src/casvisor/main.py

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
# Copyright 2025 The casbin Authors. All Rights Reserved.
2+
#
3+
# Licensed under the Apache License, Version 2.0 (the "License");
4+
# you may not use this file except in compliance with the License.
5+
# You may obtain a copy of the License at
6+
#
7+
# http://www.apache.org/licenses/LICENSE-2.0
8+
#
9+
# Unless required by applicable law or agreed to in writing, software
10+
# distributed under the License is distributed on an "AS IS" BASIS,
11+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
# See the License for the specific language governing permissions and
13+
# limitations under the License.
14+
15+
from .base import BaseClient
16+
from .record import _RecordSDK
17+
18+
19+
class CasvisorSDK(_RecordSDK):
20+
def __init__(
21+
self,
22+
endpoint: str,
23+
client_id: str,
24+
client_secret: str,
25+
organization_name: str,
26+
application_name: str,
27+
):
28+
self.endpoint = endpoint
29+
self.client_id = client_id
30+
self.client_secret = client_secret
31+
self.organization_name = organization_name
32+
self.application_name = application_name
33+
34+
# Initialize the base client
35+
self.base_client = BaseClient(client_id, client_secret, endpoint)

0 commit comments

Comments
 (0)