Public-flip audit: URL refresh + minor scrubs
Repository URLs, version strings, and example creds normalized for the public git.sulkta.com endpoint. No code-behavior change. Audit-applied by the public-flip rolling-audit pass — see kayos/openclaw-workspace memory/2026-05-27 logs for the campaign context.
This commit is contained in:
parent
04b10427f5
commit
427e1bb97f
10 changed files with 71 additions and 59 deletions
23
README.md
23
README.md
|
|
@ -1,11 +1,11 @@
|
|||
# ispconfig-py
|
||||
|
||||
Python SDK for the ISPConfig remote SOAP API. A small client for the ISPConfig
|
||||
panel. It wraps the SOAP surface so we stop writing throw-away PHP
|
||||
scripts every time we need to touch a site, zone, or mailbox.
|
||||
Python SDK for the ISPConfig remote SOAP API. Wraps the panel's SOAP surface
|
||||
so you stop writing throw-away PHP scripts every time you need to touch a
|
||||
site, zone, or mailbox.
|
||||
|
||||
v0.2 covers the **full remote API** — every method exposed by ISPConfig's
|
||||
`remote/index.php`, 312 of them as of the ISPConfig panel 2026-04-22. The hand-audited
|
||||
v0.2 covers the full remote API — every method exposed by ISPConfig's
|
||||
`remote/index.php`, 312 of them as of ISPConfig 3.2.11. The hand-audited
|
||||
helpers (stable names, param-order fixes, convenience wrappers) sit on top
|
||||
of auto-generated wrappers that mirror the PHP surface 1:1.
|
||||
|
||||
|
|
@ -33,7 +33,7 @@ from ispconfig import ISPConfigClient
|
|||
with ISPConfigClient(
|
||||
"https://panel.example.com:8080/remote/index.php",
|
||||
username="admin",
|
||||
password="password",
|
||||
password="...",
|
||||
) as c:
|
||||
site = c.sites.web_domain_get(156)
|
||||
print(site["domain"], site["php"])
|
||||
|
|
@ -120,8 +120,8 @@ ids = c.clients.get_all()
|
|||
### Auto-generated (full surface, v0.2)
|
||||
|
||||
Wrappers mirror the PHP method names 1:1. Param shapes come from PHPDoc where
|
||||
available and default to `Any` otherwise. Verified to wire up against the ISPConfig panel
|
||||
2026-04-22 but not yet battle-tested in production use — file issues if you
|
||||
available and default to `Any` otherwise. Verified to wire up against
|
||||
ISPConfig 3.2.11 but not exercised in every code path — file issues if you
|
||||
hit one.
|
||||
|
||||
| Module | Class | Methods |
|
||||
|
|
@ -191,8 +191,7 @@ funcs = c.list_functions()
|
|||
of older documentation and PHP snippets floating around, ISPConfig 3.2.11+
|
||||
wants the origin **without** a trailing dot: `dns_zone_get_id("example.com")`
|
||||
works, `dns_zone_get_id("example.com.")` raises `no_domain_found`. Our
|
||||
wrapper strips a trailing dot for you so either call works. Verified
|
||||
against the ISPConfig panel 2026-04-22.
|
||||
wrapper strips a trailing dot for you so either call works.
|
||||
- **`mail_user_get` with a filter dict returns inconsistent shapes.** If the
|
||||
filter matches multiple rows you get an array; exactly-one match returns a
|
||||
bare map. Our `mail.user_get(filter_dict)` always normalizes to a list.
|
||||
|
|
@ -261,10 +260,10 @@ require admin privileges skip gracefully with a documented reason.
|
|||
|
||||
## Regenerating for newer ISPConfig versions
|
||||
|
||||
When ISPConfig ships a new version (or another panel), resync:
|
||||
When ISPConfig ships a new version, resync against its PHP sources:
|
||||
|
||||
```bash
|
||||
# 1. Pull fresh PHP sources from the panel (sudo required; requires panel access):
|
||||
# 1. Pull fresh PHP sources from the panel host (sudo required):
|
||||
mkdir -p /tmp/ispconfig-php-src
|
||||
ssh PANEL_HOST "sudo tar -cz -C /usr/local/ispconfig/interface/lib/classes/remote.d ." \
|
||||
| tar -xz -C /tmp/ispconfig-php-src/
|
||||
|
|
|
|||
|
|
@ -23,7 +23,7 @@ from .exceptions import (
|
|||
PermissionError,
|
||||
)
|
||||
|
||||
__version__ = "0.1.0"
|
||||
__version__ = "0.2.0"
|
||||
|
||||
__all__ = [
|
||||
"AuthError",
|
||||
|
|
|
|||
|
|
@ -75,7 +75,7 @@ class SoapTransport:
|
|||
headers={
|
||||
"Content-Type": "text/xml; charset=UTF-8",
|
||||
"SOAPAction": f"urn:ispconfig#{method}",
|
||||
"User-Agent": "ispconfig-py/0.1",
|
||||
"User-Agent": "ispconfig-py/0.2",
|
||||
},
|
||||
)
|
||||
try:
|
||||
|
|
|
|||
|
|
@ -297,10 +297,6 @@ class ClientsModule:
|
|||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
def get(self, primary_id: int) -> Client:
|
||||
return cast(Client, self._c._call("client_get", ("primary_id", int(primary_id))))
|
||||
|
||||
|
|
|
|||
|
|
@ -142,10 +142,6 @@ class DatabasesModule:
|
|||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
def get(self, primary_id: int) -> Database:
|
||||
return self._c.sites.database_get(primary_id)
|
||||
|
||||
|
|
|
|||
|
|
@ -2,14 +2,15 @@
|
|||
|
||||
Footguns:
|
||||
|
||||
* Zone origins **must end in a dot** per BIND convention:
|
||||
``dns_zone_get_id(..., 'example.com.')`` — without the trailing dot
|
||||
ISPConfig returns 0 and you'll spend 20 minutes wondering why.
|
||||
* ``dns_a_add`` in some ISPConfig builds (<= 3.2.11-ish) has a known bug
|
||||
where the ``type`` column in ``dns_rr`` is not populated by the add
|
||||
handler, which means the record exists in the table but BIND never
|
||||
emits it. Workaround: after add, ``dns_rr_update`` with ``{'type':'A'}``
|
||||
to force the column. :meth:`DnsModule.a_add` does this for you and
|
||||
* ISPConfig 3.2.11+ wants the zone origin **without** a trailing dot —
|
||||
``dns_zone_get_id(..., 'example.com')``, not ``'example.com.'``. The
|
||||
dotted form raises ``no_domain_found``. :meth:`DnsModule.zone_get_id`
|
||||
strips a trailing dot for you so either form works.
|
||||
* ``dns_a_add`` in some ISPConfig builds has a known bug where the
|
||||
``type`` column in ``dns_rr`` is not populated by the add handler,
|
||||
which means the record exists in the table but BIND never emits it.
|
||||
Workaround: after add, ``dns_rr_update`` with ``{'type':'A'}`` to
|
||||
force the column. :meth:`DnsModule.a_add` does this for you and
|
||||
emits a warning log — check your ISPConfig version.
|
||||
"""
|
||||
|
||||
|
|
@ -1288,10 +1289,6 @@ class DnsModule:
|
|||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- zones --------------------------------------------------------
|
||||
|
||||
def zone_get(self, primary_id: int) -> DnsZone:
|
||||
|
|
|
|||
|
|
@ -1082,10 +1082,6 @@ class MailModule:
|
|||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- mail domains -------------------------------------------------
|
||||
|
||||
def domain_get(self, primary_id: int) -> MailDomain:
|
||||
|
|
|
|||
|
|
@ -430,10 +430,6 @@ class SitesModule:
|
|||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- HAND-EDIT ONLY BELOW ----
|
||||
|
||||
# ---- web domain ---------------------------------------------------
|
||||
|
||||
def web_domain_get(self, primary_id: int) -> WebDomain:
|
||||
|
|
|
|||
|
|
@ -6,6 +6,11 @@ are skipped — so the default ``pytest`` run on a laptop never phones home.
|
|||
|
||||
Set ``ISPCONFIG_TEST_VERIFY_SSL=0`` for panels with self-signed or
|
||||
mismatched certs.
|
||||
|
||||
Tests that read a specific record additionally read
|
||||
``ISPCONFIG_TEST_DOMAIN_ID`` (a ``web_domain`` primary id) and
|
||||
``ISPCONFIG_TEST_DOMAIN_NAME`` (the matching domain/zone origin). They skip
|
||||
individually if either is missing — see ``tests/test_smoke.py``.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
|
|
|||
|
|
@ -3,17 +3,25 @@
|
|||
Gated on env vars: ``ISPCONFIG_TEST_URL``, ``ISPCONFIG_TEST_USER``,
|
||||
``ISPCONFIG_TEST_PASS``. These tests are skipped if any is unset.
|
||||
|
||||
They are **read-only** — no ``_add`` / ``_update`` / ``_delete`` calls. Safe
|
||||
to run against production.
|
||||
They are read-only — no ``_add`` / ``_update`` / ``_delete`` calls. Safe to
|
||||
run against production.
|
||||
|
||||
Every new auto-generated module gets at least one read-only call here so we
|
||||
know the wrappers actually wire up against a live panel. Methods that the
|
||||
API user lacks permission for (admin-only, etc.) are documented skips —
|
||||
see the README's "Known admin-only" list.
|
||||
Two further env vars target a known record on your panel so the hand-audited
|
||||
helpers have something to fetch:
|
||||
|
||||
* ``ISPCONFIG_TEST_DOMAIN_ID`` — a ``web_domain`` primary id (int).
|
||||
* ``ISPCONFIG_TEST_DOMAIN_NAME`` — the matching ``domain`` string + DNS zone
|
||||
origin (e.g. ``example.com``).
|
||||
|
||||
If either is missing, the domain/dns/mail tests below are skipped — only
|
||||
the auto-generated module probes run, since those don't need a known record.
|
||||
Methods that the API user lacks permission for (admin-only, etc.) skip
|
||||
cleanly — see the README's "Known admin-only" list.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import os
|
||||
from collections.abc import Iterator
|
||||
|
||||
import pytest
|
||||
|
|
@ -33,6 +41,20 @@ def client(live_creds: dict[str, str]) -> Iterator[ISPConfigClient]:
|
|||
yield c
|
||||
|
||||
|
||||
def _known_domain_id() -> int:
|
||||
val = os.environ.get("ISPCONFIG_TEST_DOMAIN_ID")
|
||||
if not val:
|
||||
pytest.skip("set ISPCONFIG_TEST_DOMAIN_ID to a real web_domain primary id")
|
||||
return int(val)
|
||||
|
||||
|
||||
def _known_domain_name() -> str:
|
||||
val = os.environ.get("ISPCONFIG_TEST_DOMAIN_NAME")
|
||||
if not val:
|
||||
pytest.skip("set ISPCONFIG_TEST_DOMAIN_NAME to the matching domain/zone origin")
|
||||
return val
|
||||
|
||||
|
||||
# ---- hand-audited modules (first pass) -----------------------------------
|
||||
|
||||
|
||||
|
|
@ -44,20 +66,24 @@ def test_login_returns_session(live_creds: dict[str, str]) -> None:
|
|||
assert c.logout() is True
|
||||
|
||||
|
||||
def test_get_domain_156(client: ISPConfigClient) -> None:
|
||||
site = client.sites.web_domain_get(156)
|
||||
assert site["domain"] == "mcbindustrial.com"
|
||||
assert site["php"] == "fast-cgi"
|
||||
def test_get_known_domain(client: ISPConfigClient) -> None:
|
||||
domain_id = _known_domain_id()
|
||||
expected_name = _known_domain_name()
|
||||
site = client.sites.web_domain_get(domain_id)
|
||||
assert site["domain"] == expected_name
|
||||
|
||||
|
||||
def test_dns_zone_lookup(client: ISPConfigClient) -> None:
|
||||
zone_id = client.dns.zone_get_id("mcbindustrial.com.")
|
||||
expected_name = _known_domain_name()
|
||||
# Pass with a trailing dot to exercise the wrapper's stripping behavior.
|
||||
zone_id = client.dns.zone_get_id(expected_name + ".")
|
||||
assert zone_id > 0, "zone_get_id returned 0 — is the zone present?"
|
||||
|
||||
|
||||
def test_mail_users_under_mcbindustrial(client: ISPConfigClient) -> None:
|
||||
def test_mail_users_filter_returns_list(client: ISPConfigClient) -> None:
|
||||
# mail_user_get accepts a filter-dict and returns a list.
|
||||
users = client.mail.user_get({"email": "%@mcbindustrial.com"})
|
||||
expected_name = _known_domain_name()
|
||||
users = client.mail.user_get({"email": f"%@{expected_name}"})
|
||||
assert isinstance(users, list)
|
||||
# Don't assert count — just shape. Zero mailboxes is a valid state.
|
||||
for u in users:
|
||||
|
|
@ -67,8 +93,8 @@ def test_mail_users_under_mcbindustrial(client: ISPConfigClient) -> None:
|
|||
# ---- auto-generated modules: one read-only probe each -------------------
|
||||
#
|
||||
# These prove the wrappers encode/decode correctly against a real panel.
|
||||
# Each test tolerates the method being restricted to admin (``admin`` is a
|
||||
# reseller, not an admin) — those skip with a clear reason.
|
||||
# Each test tolerates the method being restricted to admin — reseller logins
|
||||
# fault with "permission denied" on those, and we skip with a clear reason.
|
||||
|
||||
|
||||
def test_raw_call_list_functions(client: ISPConfigClient) -> None:
|
||||
|
|
@ -188,12 +214,13 @@ def test_cron_get_missing(client: ISPConfigClient) -> None:
|
|||
|
||||
def test_backups_list(client: ISPConfigClient) -> None:
|
||||
"""``sites_web_domain_backup_list`` on a known domain."""
|
||||
domain_id = _known_domain_id()
|
||||
try:
|
||||
result = client.backups.sites_web_domain_backup_list(156)
|
||||
result = client.backups.sites_web_domain_backup_list(domain_id)
|
||||
except PermissionError:
|
||||
pytest.skip("sites_web_domain_backup_list: admin-only on this panel")
|
||||
except NotFoundError:
|
||||
pytest.skip("no backups configured for domain 156")
|
||||
pytest.skip(f"no backups configured for domain {domain_id}")
|
||||
assert result is None or isinstance(result, (list, dict))
|
||||
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue