import json
from abc import ABC, abstractmethod
from urllib import parse as urllib
from lib.common import rest_client
from lib.services.redis.base_client import BaseClient


class UserClient(BaseClient):
    """Service client for User API"""

    def __init__(self, auth_provider, client, **kwargs):
        super().__init__(auth_provider=auth_provider, **kwargs)
        self.client = client

    def _parse_response(self, resp, body, schema):
        """Parse and validate response."""
        # Handle empty response bodies (e.g., validation endpoints)
        if not body or body.strip() == '':
            body_data = {}
        else:
            body_data = json.loads(body)

        self.validate_response(schema, resp, body_data)

        # Return format expected by tests: {'status': int, 'body': dict}
        return {
            'status': resp.status,
            'body': body_data
        }

    def list_users(self, service_name, **params):
        """A list of users in the service."""
        url = self.client.get_users_path(self.project_name, service_name)
        if params:
            url += f"?{urllib.urlencode(params)}"
        resp, body = self.get(url)
        return self._parse_response(
            resp, body, self.client.get_schema().list_users
        )

    def create_user(self, service_name, **kwargs):
        """Create user."""
        url = self.client.get_users_path(self.project_name, service_name)
        post_body = json.dumps(kwargs)
        resp, body = self.post(url, post_body)
        return self._parse_response(
            resp, body, self.client.get_schema().create_user
        )

    def delete_user(self, service_name, user_name):
        """Delete user."""
        url = self.client.get_user_path(self.project_name, service_name, user_name)
        resp, body = self.delete(url)
        return self._parse_response(
            resp, body, self.client.get_schema().delete_user
        )

    def update_user(self, service_name, user_name, **kwargs):
        """Edit user."""
        url = self.client.get_user_path(self.project_name, service_name, user_name)
        patch_body = json.dumps(kwargs)
        resp, body = self.patch(url, patch_body)
        return self._parse_response(
            resp, body, self.client.get_schema().update_user
        )

    def update_user_status(self, service_name, user_name, **kwargs):
        """Enable or disable a user in the service."""
        url = self.client.get_user_status_path(self.project_name, service_name, user_name)
        put_body = json.dumps(kwargs)
        resp, body = self.put(url, put_body)
        return self._parse_response(
            resp, body, self.client.get_schema().update_user_status
        )

    def validate_user_name(self, service_name, user_name):
        """Check the username is duplicated in the service."""
        url = self.client.get_user_validate_path(self.project_name, service_name, user_name)
        resp, body = self.get(url)
        return self._parse_response(
            resp, body, self.client.get_schema().validate_user_name
        )


class UserClientInterface(ABC):
    """Interface for cloud-specific implementations."""

    @abstractmethod
    def get_schema(self):
        """Return response schemas module."""
        pass

    @abstractmethod
    def get_users_path(self, project_name, service_name):
        """Return API endpoint path for users."""
        pass

    @abstractmethod
    def get_user_path(self, project_name, service_name, user_name):
        """Return API endpoint path for a specific user."""
        pass

    @abstractmethod
    def get_user_status_path(self, project_name, service_name, user_name):
        """Return API endpoint path for user status."""
        pass

    @abstractmethod
    def get_user_validate_path(self, project_name, service_name, user_name):
        """Return API endpoint path for user validation."""
        pass
