fix: Update user/groups API to OCS v2

This commit is contained in:
Chris Coutinho
2025-10-15 00:05:22 +02:00
parent 898c2e72ae
commit 7a4a31b52d
3 changed files with 35 additions and 55 deletions
+24 -46
View File
@@ -10,7 +10,7 @@ class UsersClient(BaseNextcloudClient):
self, additional_headers: Optional[Dict[str, str]] = None
) -> Dict[str, str]:
"""Get standard headers required for User API calls."""
headers = {"OCS-APIRequest": "true"}
headers = {"OCS-APIRequest": "true", "Accept": "application/json"}
if additional_headers:
headers.update(additional_headers)
return headers
@@ -49,7 +49,7 @@ class UsersClient(BaseNextcloudClient):
headers = self._get_user_headers()
await self._make_request(
"POST", "/ocs/v1.php/cloud/users", data=data, headers=headers
"POST", "/ocs/v2.php/cloud/users", data=data, headers=headers
)
async def search_users(
@@ -71,18 +71,11 @@ class UsersClient(BaseNextcloudClient):
headers = self._get_user_headers()
response = await self._make_request(
"GET", "/ocs/v1.php/cloud/users", params=params, headers=headers
"GET", "/ocs/v2.php/cloud/users", params=params, headers=headers
)
# The API returns XML, which is parsed into a dict.
# The user IDs are under ocs.data.users.element (can be a list or a single string)
# The v2 API returns JSON with users as a direct list under data.users
data = response.json()["ocs"]["data"]
if "users" in data and "element" in data["users"]:
elements = data["users"]["element"]
if isinstance(elements, list):
return elements
elif isinstance(elements, str):
return [elements]
return []
return data.get("users", [])
async def get_user_details(self, userid: str) -> UserDetails:
"""
@@ -90,7 +83,7 @@ class UsersClient(BaseNextcloudClient):
"""
headers = self._get_user_headers()
response = await self._make_request(
"GET", f"/ocs/v1.php/cloud/users/{userid}", headers=headers
"GET", f"/ocs/v2.php/cloud/users/{userid}", headers=headers
)
return UserDetails(**response.json()["ocs"]["data"])
@@ -101,7 +94,7 @@ class UsersClient(BaseNextcloudClient):
data = {"key": key, "value": value}
headers = self._get_user_headers()
await self._make_request(
"PUT", f"/ocs/v1.php/cloud/users/{userid}", data=data, headers=headers
"PUT", f"/ocs/v2.php/cloud/users/{userid}", data=data, headers=headers
)
async def get_editable_user_fields(self) -> List[str]:
@@ -110,16 +103,11 @@ class UsersClient(BaseNextcloudClient):
"""
headers = self._get_user_headers()
response = await self._make_request(
"GET", "/ocs/v1.php/cloud/user/fields", headers=headers
"GET", "/ocs/v2.php/cloud/user/fields", headers=headers
)
# The v2 API returns data as a direct list
data = response.json()["ocs"]["data"]
if "element" in data:
elements = data["element"]
if isinstance(elements, list):
return elements
elif isinstance(elements, str):
return [elements]
return []
return data if isinstance(data, list) else []
async def disable_user(self, userid: str) -> None:
"""
@@ -127,7 +115,7 @@ class UsersClient(BaseNextcloudClient):
"""
headers = self._get_user_headers()
await self._make_request(
"PUT", f"/ocs/v1.php/cloud/users/{userid}/disable", headers=headers
"PUT", f"/ocs/v2.php/cloud/users/{userid}/disable", headers=headers
)
async def enable_user(self, userid: str) -> None:
@@ -136,7 +124,7 @@ class UsersClient(BaseNextcloudClient):
"""
headers = self._get_user_headers()
await self._make_request(
"PUT", f"/ocs/v1.php/cloud/users/{userid}/enable", headers=headers
"PUT", f"/ocs/v2.php/cloud/users/{userid}/enable", headers=headers
)
async def delete_user(self, userid: str) -> None:
@@ -145,7 +133,7 @@ class UsersClient(BaseNextcloudClient):
"""
headers = self._get_user_headers()
await self._make_request(
"DELETE", f"/ocs/v1.php/cloud/users/{userid}", headers=headers
"DELETE", f"/ocs/v2.php/cloud/users/{userid}", headers=headers
)
async def get_user_groups(self, userid: str) -> List[str]:
@@ -154,16 +142,11 @@ class UsersClient(BaseNextcloudClient):
"""
headers = self._get_user_headers()
response = await self._make_request(
"GET", f"/ocs/v1.php/cloud/users/{userid}/groups", headers=headers
"GET", f"/ocs/v2.php/cloud/users/{userid}/groups", headers=headers
)
# The v2 API returns groups as a direct list under data.groups
data = response.json()["ocs"]["data"]
if "groups" in data and "element" in data["groups"]:
elements = data["groups"]["element"]
if isinstance(elements, list):
return elements
elif isinstance(elements, str):
return [elements]
return []
return data.get("groups", [])
async def add_user_to_group(self, userid: str, groupid: str) -> None:
"""
@@ -173,7 +156,7 @@ class UsersClient(BaseNextcloudClient):
headers = self._get_user_headers()
await self._make_request(
"POST",
f"/ocs/v1.php/cloud/users/{userid}/groups",
f"/ocs/v2.php/cloud/users/{userid}/groups",
data=data,
headers=headers,
)
@@ -186,7 +169,7 @@ class UsersClient(BaseNextcloudClient):
headers = self._get_user_headers()
await self._make_request(
"DELETE",
f"/ocs/v1.php/cloud/users/{userid}/groups",
f"/ocs/v2.php/cloud/users/{userid}/groups",
data=data,
headers=headers,
)
@@ -199,7 +182,7 @@ class UsersClient(BaseNextcloudClient):
headers = self._get_user_headers()
await self._make_request(
"POST",
f"/ocs/v1.php/cloud/users/{userid}/subadmins",
f"/ocs/v2.php/cloud/users/{userid}/subadmins",
data=data,
headers=headers,
)
@@ -212,7 +195,7 @@ class UsersClient(BaseNextcloudClient):
headers = self._get_user_headers()
await self._make_request(
"DELETE",
f"/ocs/v1.php/cloud/users/{userid}/subadmins",
f"/ocs/v2.php/cloud/users/{userid}/subadmins",
data=data,
headers=headers,
)
@@ -223,16 +206,11 @@ class UsersClient(BaseNextcloudClient):
"""
headers = self._get_user_headers()
response = await self._make_request(
"GET", f"/ocs/v1.php/cloud/users/{userid}/subadmins", headers=headers
"GET", f"/ocs/v2.php/cloud/users/{userid}/subadmins", headers=headers
)
# The v2 API returns data as a direct list
data = response.json()["ocs"]["data"]
if "element" in data:
elements = data["element"]
if isinstance(elements, list):
return elements
elif isinstance(elements, str):
return [elements]
return []
return data if isinstance(data, list) else []
async def resend_welcome_email(self, userid: str) -> None:
"""
@@ -240,5 +218,5 @@ class UsersClient(BaseNextcloudClient):
"""
headers = self._get_user_headers()
await self._make_request(
"POST", f"/ocs/v1.php/cloud/users/{userid}/welcome", headers=headers
"POST", f"/ocs/v2.php/cloud/users/{userid}/welcome", headers=headers
)
+6 -4
View File
@@ -1,5 +1,5 @@
from typing import List, Optional
from pydantic import BaseModel, Field
from typing import Any, Dict, List, Optional, Union
from pydantic import BaseModel, ConfigDict, Field
class User(BaseModel):
@@ -18,10 +18,12 @@ class User(BaseModel):
class UserDetails(BaseModel):
"""Model for retrieving detailed user information."""
model_config = ConfigDict(populate_by_name=True)
enabled: bool
id: str
quota: str
email: str
quota: Union[str, Dict[str, Any]] # Can be string or quota object
email: Optional[str] = None # Can be null
displayname: str = Field(
alias="display-name"
) # Handle both displayname and display-name
+5 -5
View File
@@ -5,7 +5,7 @@ from nextcloud_mcp_server.client import NextcloudClient
@pytest.mark.asyncio
async def test_create_and_delete_user(nc_client: NextcloudClient):
userid = "testuser1"
password = "testpassword1"
password = "SecureTestPassword123!"
display_name = "Test User One"
email = "test1@example.com"
@@ -37,7 +37,7 @@ async def test_create_and_delete_user(nc_client: NextcloudClient):
@pytest.mark.asyncio
async def test_update_user_field(nc_client: NextcloudClient):
userid = "testuser2"
password = "testpassword2"
password = "SecureTestPassword123!"
display_name = "Test User Two"
email = "test2@example.com"
@@ -60,7 +60,7 @@ async def test_update_user_field(nc_client: NextcloudClient):
@pytest.mark.asyncio
async def test_user_groups(nc_client: NextcloudClient):
userid = "testuser3"
password = "testpassword3"
password = "SecureTestPassword123!"
groupid = "testgroup"
await nc_client.users.create_user(userid=userid, password=password)
@@ -81,7 +81,7 @@ async def test_user_groups(nc_client: NextcloudClient):
@pytest.mark.asyncio
async def test_user_subadmins(nc_client: NextcloudClient):
userid = "testuser4"
password = "testpassword4"
password = "SecureTestPassword123!"
groupid = "subadmingroup"
await nc_client.users.create_user(userid=userid, password=password)
@@ -102,7 +102,7 @@ async def test_user_subadmins(nc_client: NextcloudClient):
@pytest.mark.asyncio
async def test_disable_enable_user(nc_client: NextcloudClient):
userid = "testuser5"
password = "testpassword5"
password = "SecureTestPassword123!"
await nc_client.users.create_user(userid=userid, password=password)