Skip to main content

Python for CPM

Project description

Python4CPM

A simple and secure way of using python scripts with CyberArk CPM/SRS password rotations.

How it works

This module leverages the Credential Management .NET SDK from CyberArk to securely offload a password rotation logic into Python.

All objects are collected from the SDK and sent as environment context to be picked up by the python4cpm module during the subprocess execution of python. All secrets of such environment are protected and encrypted by Data Protection API (DPAPI), until they are explicitely retrieved in your python script runtime, invoking the Secret.get() method. Finally, python controls the termination signal sent back to the SDK, which is consequently used as the return code to CPM/SRS. Such as a successful or failed (recoverable or not) result of the requested action.

This platform allows you to duplicate it multiple times, simply changing its settings (with regular day two operations from Privilege Cloud/PVWA) to point to different venvs and/or python scripts.

Installation

Preparing Python

  1. Install Python along CPM or the SRS Connector Management Agent.
    • Python must be installed for all users. Follow the custom install steps from the installation wizard to check the checkbox.
  2. Create a venv in the server, by running py -m venv c:\venv. If desired, use a custom location and adjust any future references.
  3. Install python4cpm in your venv:
    • If your CPM can connect to the internet, install with c:\venv\Scripts\pip.exe install python4cpm.
    • If your CPM cannot connect to the internet:
      • Download the latest python4cpm-*.whl file from the pypi project files.
      • Copy the file to the server into a temporary directory called python4cpm-wheel.
      • From the parent directory of python4cpm-wheel run c:\venv\Scripts\pip.exe install --no-index --find-links=.\python4cpm-wheel python4cpm.

Importing the platform

If you are using CPM (SaaS or Self-Hosted):

  1. Download the latest Credential Management .NET SDK and place its content in the bin folder of CPM (C:\Program Files (x86)\CyberArk\Password Manager\bin). The files for this may already be present.
  2. Download the python4cpm-platform-*.zip asset from the latest release.
  3. Import the platform zip file into Privilege Cloud/PVWA (Administration -> Platform Management -> Import platform).
  4. Craft your python script and place it within a folder in CPM (e.g., C:\python4cpm-scripts).
  5. Duplicate the imported platform in Privilege Cloud/PVWA (Administration -> Platform Management -> Application -> Python for CPM) and name it after your application (e.g., My App).
  6. Edit the duplicated platform and specify the path of your script, under Target Account Platform -> Automatic Platform Management -> Additional Policy Settings -> Parameters -> PythonScriptPath -> Value (e.g., C:\python4cpm-scripts\myapp.py).
  7. Also update Target Account Platform -> Automatic Platform Management -> Additional Policy Settings -> Parameters -> PythonExePath -> Value with the custom path for the venv's python.exe file (e.g., c:\venv\Scripts\python.exe).
  8. If you want to disable logging, update Target Account Platform -> Automatic Platform Management -> Additional Policy Settings -> Parameters -> PythonLogging -> Value to no.
  9. If you want to change the logging level to debug, update Target Account Platform -> Automatic Platform Management -> Additional Policy Settings -> Parameters -> PythonLoggingLevel -> Value to debug.
  10. For new applications repeat steps from 4 to 9.

If you are using SRS (SaaS only):

  1. Download the python4cpm-platform-*.zip asset from the latest release.
  2. Import the platform zip file into Privilege Cloud (Administration -> Platform Management -> Import platform).
  3. Craft your python script and place it within a folder in the Cloud Connector (where the SRS Management Agent runs) (e.g., C:\python4cpm-scripts).
  4. Duplicate the imported platform in Privilege Cloud/PVWA (Administration -> Platform Management -> Application -> Python for CPM) and name it after your application (e.g., My App).
  5. Edit the duplicated platform and specify the path of your script, under Plugin Settings -> Additional Parameters -> PythonScriptPath (e.g., C:\python4cpm-scripts\myapp.py).
  6. Also update Plugin Settings -> Additional Parameters -> PythonExePath with the custom path for the venv's python.exe file (e.g., c:\venv\Scripts\python.exe).
  7. If you want to disable logging, update Plugin Settings -> Additional Parameters -> PythonLogging to no.
  8. If you want to change the logging level to debug, update Plugin Settings -> Additional Parameters -> PythonLoggingLevel -> Value to debug.
  9. For new applications repeat steps from 3 to 8.

Python Script

from python4cpm import Python4CPMHandler


class MyRotator(Python4CPMHandler): # create a subclass for the Handler
    """
    These are the usable properties and methods from Python4CPMHandler:

        self.args.action
        # action requested from CPM/SRS

        ## Target Account

        self.target_account.username
        # address from account

        self.target_account.address
        # address from account

        self.target_account.port
        # port from account

        self.target_account.password.get()
        # get plaintext str from password object

        ## Logon Account
        self.logon_account.username
        self.logon_account.password.get()

        ## Reconcile Account
        self.reconcile_account.username
        self.reconcile_account.password.get()

        ## Logging

        self.logger.critical("this is critical message")
        self.logger.error("this is an error message")
        self.logger.warning("this is a warning message")
        self.logger.info("this is an info message")
        self.logger.debug("this is a debug message")

        # The logging level comes from PythonLoggingLevel (platform parameters) (default is error)

    =============================
    REQUIRED TERMINATION SIGNALS
    =============================
    Terminate signals -> MUST use one of the following three signals to terminate the script:

        self.close_success()
        # terminate and provide CPM/SRS with a success state

        self.close_fail()
        # terminate and provide CPM/SRS with a failed recoverable state
        
        self.close_fail(unrecoverable=True)
        # terminate and provide CPM/SRS with a failed unrecoverable state

    When calling a signal sys.exit is invoked and the script is terminated.
    If no signal is called, and the script finishes without any exception,
    it will behave like p4cpm.close_fail(unrecoverable=True) and log an error message.
    =============================
    =============================
    """

    # =============================
    # REQUIRED METHODS (MUST DEFINE)
    # =============================
    # verify(), logon(), change(), prereconcile(), reconcile()

    def verify(self):
        self._verify()
        self.log_info("verification successful")
        self.close_success()

    def logon(self):
        self.close_success()

    def change(self):
        self._change()
        self.log_error("something went wrong")
        self.close_fail()

    def prereconcile(self):
        self._verify(from_reconcile=True)
        self.close_success()

    def reconcile(self):
        self._change(from_reconcile=True)
        self.close_success()

    def _verify(self, from_reconcile=False):
        if from_reconcile is False:
            pass
            # TODO: use account objects for your logic
        else:
            pass
            # TODO: use account objects for your logic
        result = True
        if result is True:
            self.log_info("verification successful")
        else:
            self.log_error("something went wrong")
            self.close_fail()

    def _change(self, from_reconcile=False):
        if from_reconcile is False:
            pass
            # TODO: use account objects for your logic
        else:
            pass
            # TODO: use account objects for your logic
        result = True
        if result is True:
            self.log_info("rotation successful")
        else:
            self.log_error("something went wrong")
            self.close_fail()


if __name__ == "__main__":
    MyRotator().run() # initializes the class and calls the action that was requested from CPM/SRS.

(*) More realistic examples can be found here.

When doing verify, change or reconcile from Privilege Cloud/PVWA:

  1. Verify -> the sciprt will be executed once running the MyRotator.verify() method.
  2. Change -> the sciprt will be executed twice, running first the MyRotator.logon() method and secondly the MyRotator.change() method.
    • If both actions are not terminated with self.close_success() and the scripts terminates without any exception, CPM/SRS will see this as a self.close_fail(unrecoverable=True).
  3. Reconcile -> the sciprt will be executed twice, running first the MyRotator.prereconcile() method and secondly the MyRotator.reconcile() method.
    • If both actions are not terminated with self.close_success() and the scripts terminates without any exception, CPM/SRS will see this as a self.close_fail(unrecoverable=True).
  4. When calling MyRotator.verify(), MyRotator.logon() or MyRotator.prereconcile(): self.target_account.new_password.get() will always return an empty string.
  5. If a logon account is not linked, self.logon_account.username and self.logon_account.password.get() will return empty strings.
  6. If a reconcile account is not linked, self.reconcile_account.username and self.reconcile_account.password.get() will return empty strings.
  7. The python Logger places its logs in the Logs/ThirdParty directory. The filename will be based on the name of the subclass created (e.g., MyRotator).

Installing dependencies in python venv

As with any python venv, you can install dependencies in your venv.

  1. If your CPM can connect to the internet:
    • You can use regular pip install commands (e.g., c:\venv\Scripts\pip.exe install requests).
  2. If your CPM cannot connect to the internet:
    • You can download packages for an offline install. More info here.

Dev Helper:

For dev purposes, NETHelper is a companion helper to test your scripts before shipping to CPM/SRS. It simplifies the instantiation of the Python4CPM or Python4CPMHandler objects by simulating how the plugin creates the environment context for the python module.

Note: As CPM and the SRS management agent run in Windows, the plugin was built to encrypt secrets using DPAPI (a windows only library). For dev purposes in Linux/Mac dev workstations, those secrets put in the environment context by NETHelper will be in plaintext. In windows dev workstations, NETHelper encrypts the secrets as the .NET plugin does. This is informational only, the module will use its encryption/decryption capabilities automatically based on the platform it is running on and you do not have to do anything specific to enable it.

Example:

Set your arguments and secrets:

from python4cpm import NETHelper, Python4CPM, Python4CPMHandler
from getpass import getpass

# Get secrets for your password, logon account password, reconcile account password and new password
# You can use an empty string if it does not apply
target_password = getpass("password: ") # password from account
logon_password = getpass("logon_password: ") # password from linked logon account
reconcile_password = getpass("reconcile_password: ") # password from linked reconcile account
target_new_password = getpass("new_password: ") # new password for the rotation

NETHelper.set(
    action=Python4CPM.ACTION_CHANGE, # use actions from Python4CPM.ACTION_*
    target_username="jdoe", # -> will fall under MyRotator.target_account.username
    target_address="myapp.corp.local", # -> will fall under MyRotator.target_account.address
    target_port="8443", # -> will fall under MyRotator.target_account.port
    logon_username="ldoe", # -> will fall under MyRotator.logon_account.username
    reconcile_username="rdoe", # -> will fall under MyRotator.reconcile_account.username
    logging_level="debug", # "critical", "error", "warning", "info" or "debug"
    target_password=target_password, # -> will fall under MyRotator.target_account.password.get()
    logon_password=logon_password, # -> will fall under MyRotator.logon_account.password.get()
    reconcile_password=reconcile_password, # -> will fall under MyRotator.reconcile_account.password.get()
    target_new_password=target_new_password # -> will fall under MyRotator.target_account.new_password.get()
)

class MyRotator(Python4CPMHandler):
    def verify(self):
        # TODO: Add your logic here
        self.close_success()

    def logon(self):
        # TODO: Add your logic here
        self.close_success()

    def change(self):
        # TODO: Add your logic here
        self.close_success()

    def prereconcile(self):
        # TODO: Add your logic here
        self.close_success()

    def reconcile(self):
        # TODO: Add your logic here
        self.close_success()

MyRotator().run()

Remember for your final script:

  • Remove the import of NETHelper.
  • Remove the NETHelper.set() call.
  • If applicable, change the definition of p4cpm from p4cpm = NETHelper.get() to p4cpm = Python4CPM("MyApp").
  • Remove any secrets prompting or interactive interruptions.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

python4cpm-1.1.2.tar.gz (13.7 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

python4cpm-1.1.2-py3-none-any.whl (11.9 kB view details)

Uploaded Python 3

File details

Details for the file python4cpm-1.1.2.tar.gz.

File metadata

  • Download URL: python4cpm-1.1.2.tar.gz
  • Upload date:
  • Size: 13.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for python4cpm-1.1.2.tar.gz
Algorithm Hash digest
SHA256 9b1b3ba29af0b000cb75ef15637528e2020bb0ed54ccc33e2a92637894b7b2eb
MD5 fa48e08ed993e6feba6556190d6c75fb
BLAKE2b-256 7ced348a07702d8c9b0f31c5aca035a130b6dbe308c271e23a38d25e260c6894

See more details on using hashes here.

Provenance

The following attestation bundles were made for python4cpm-1.1.2.tar.gz:

Publisher: release-pypi.yml on gonatienza/python4cpm

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file python4cpm-1.1.2-py3-none-any.whl.

File metadata

  • Download URL: python4cpm-1.1.2-py3-none-any.whl
  • Upload date:
  • Size: 11.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for python4cpm-1.1.2-py3-none-any.whl
Algorithm Hash digest
SHA256 6d3ca669352f5f63bdf0bfa6df781d598cc74aba253f703684e96c23d9e3ccf2
MD5 b267adfbcdc2ec7d16060688c7c70447
BLAKE2b-256 238ced930987201804e57f87389187d6f75009523ef912bbe54798bb99a0ac81

See more details on using hashes here.

Provenance

The following attestation bundles were made for python4cpm-1.1.2-py3-none-any.whl:

Publisher: release-pypi.yml on gonatienza/python4cpm

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page