Migration of DCS Python client scripts#

Just as HPS is the successor to DCS (Distributed Compute Services), PyHPS is the successor to the DCS Python client. For more information, see the latest DCS Python Client documentation.

Key terminology and API changes#

The Design Points Service (DPS) in DCS has evolved into the Job Management Service (JMS) in HPS. Some key entities that were part of DPS are renamed in JMS:

DPS

JMS

Configuration

Job definition

Design point

Job

Process step

Task definition

Parameter location

Parameter mapping

Here are some other key changes impacting scripts that use the DCS Python Client:

  • All resource IDs are strings now (as opposed to integers).

  • In DCS, the client specifies the ID of the project when creating it. In HPS, the client specifies only the name of the project. The ID is assigned server-side.

  • Some nested endpoints have been removed. For example, the following endpoints do not have a counterpart in the JMS API:

    GET /dps/api/projects/{project_id}/configurations/{configuration_id}/design_points
    GET /dps/api/projects/{project_id}/design_points/{design_point_id}/tasks
    
  • In the DPS API, the configuration resource (now job definition) used to include process steps, parameter definitions, and parameter locations. These are now separate resources (with corresponding endpoints) in the JMS API, namely task definitions, parameter definitions, and parameter mappings.

  • Within a task definition (previously process step), you can now also specify the following information:

    • Multiple software requirements

    • Environment variables

    • Execution context

    • Custom resource requirements

    • HPC resource requirements

    The cpu_core_usage field has been renamed to num_cores. The memory and disk_space fields are expressed in bytes rather than megabytes.

Python client changes#

Installation#

The DCS Python client used to be distributed as a wheel in the Ansys installation. You can install PyHPS as indicated in Getting started.

Client and API objects#

With respect to the DCS Python client, the PyHPS structure has been streamlined. The main Client object stores only the connection, without exposing any service API. Similarly, resources like Project and Job do not expose API endpoints. Instead, objects like JmsApi, ProjectApi, and RmsApi wrap around HPS REST APIs.

This code shows how to instantiate a Client object and retrieve projects:

from ansys.dcs.client.dps import Client

client = Client(
    dcs_url="https://localhost/dcs",
    username="dcadmin",
    password="dcadmin"
)
projects = client.get_projects()

Project ID#

As mentioned earlier, in HPS the client only specifies the name of the project. The project ID is assigned server-side. This code shows how to create a project:

from ansys.dcs.client.dps import Client, Project

client = Client(...)

proj = Project(
    id="my_new_project",
    display_name="My New Project"
)
proj = client.create_project(proj)

Removed nested endpoints#

Following the changes in the API, nested endpoints are removed.

Exceptions#

Exception handling works the same. However, the DCSError class has been renamed to HPSError.

from ansys.dcs.client import DCSError
from ansys.dcs.client.dps import Client

try:
    client = Client(
        dcs_url="https://localhost/dcs/",
        username="dcadmin",
        password="wrong_psw"
    )
except DCSError as e:
    print(e)

Evaluators#

The evaluator resources and corresponding endpoints have been moved to the new Resource Management Service (RMS). This is reflected in PyHPS accordingly.

from ansys.dcs.client.dps import Client

client = Client(...)

evaluators = client.get_evaluators()

Example project#

This example shows how to migrate a script that creates a DCS project consisting of an Ansys APDL beam model of a tubular motorbike-frame. The script was originally included in the DCS Python Client documentation and is now available as a PyHPS script in the example_mapdl_motorbike_frame example.

Import modules and instantiate the client#

import os

from ansys.dcs.client.dps import Client
from ansys.dcs.client.dps.resource import (
    Configuration,
    DesignPoint,
    File,
    FitnessDefinition,
    Project,
    SuccessCriteria
)

client = Client(
    dcs_url="https://127.0.0.1/dcs",
    username="dcadmin",
    password="dcadmin"
)

Create an empty project and a job definition#

proj = Project(
    id="mapdl_motorbike_frame",
    display_name="MAPDL motorbike frame",
    priority=1,
    active=True
)
proj = client.create_project(proj, replace=True)

cfg = Configuration(name="Configuration.1", active=True)

Create file resources#

cwd = os.path.dirname(__file__)
files = []

# Input File
files.append(
    File(
        name="mac",
        evaluation_path="motorbike_frame.mac",
        type="text/plain",
        src=os.path.join(cwd, "motorbike_frame.mac")
    )
)

# Output Files
files.append( File( name="results", evaluation_path="motorbike_frame_results.txt", type="text/plain" ) )
files.append( File( name="img", evaluation_path="file000.jpg", type="image/jpeg", collect=True) )
files.append( File( name="img2", evaluation_path="file001.jpg", type="image/jpeg", collect=True) )
files.append( File( name="out", evaluation_path="file.out", type="text/plain", collect=True) )

# create file resources on the server
files = proj.create_files(files)

# For convenience, keep a reference to the input and result files.
mac_file = files[0]
result_file = files[1]

Create parameter definitions#

# Input params: Dimensions of three custom tubes
float_input_params=[]
for i in range(1,4):
    pd = cfg.add_float_parameter_definition(
        name='tube%i_radius' %i,
        lower_limit=4.0,
        upper_limit=20.0,default=12.0
    )
    cfg.add_parameter_location(
        key_string='radius(%i)' % i,
        tokenizer="=",
        parameter_definition_name=pd.name,
        file_id=mac_file.id
    )
    float_input_params.append(pd)
    pd = cfg.add_float_parameter_definition(
        name='tube%i_thickness' %i,
        lower_limit=0.5,
        upper_limit=2.5,
        default=1.0 )
    cfg.add_parameter_location(
        key_string='thickness(%i)' % i,
        tokenizer="=",
        parameter_definition_name=pd.name,
        file_id=mac_file.id
    )
    float_input_params.append(pd)

# Input params: Custom types used for all the different tubes of the frame
str_input_params=[]
for i in range(1,22):
    pd = cfg.add_string_parameter_definition(
        name="tube%s" %i,
        default="1",
        value_list=["1","2","3"]
    )
    cfg.add_parameter_location(
        key_string='tubes(%i)' % i,
        tokenizer="=",
        parameter_definition_name=pd.name,
        file_id=mac_file.id
    )
    str_input_params.append(pd)

# Output Parames
for pname in ["weight", "torsion_stiffness", "max_stress"]:
    pd = cfg.add_float_parameter_definition(name=pname)
    cfg.add_parameter_location(
        key_string=pname,
        tokenizer="=",
        parameter_definition_name=pd.name,
        file_id=result_file.id
    )

Create process steps (now tasks)#

cfg.add_process_step(
    name="MAPDL_run",
    application_name="ANSYS Mechanical APDL",
    application_version="2024 R1",
    execution_command="%executable% -b -i %file:mac% -o file.out",
    max_execution_time=20.0,
    cpu_core_usage=1,
    execution_level=0,
    memory=250,
    disk_space=5,
    input_file_ids=[f.id for f in files[:1]],
    output_file_ids=[f.id for f in files[1:]],
    success_criteria= SuccessCriteria(
        return_code=0,
        expressions= ["values['tube1_radius']>=4.0", "values['tube1_thickness']>=0.5"],
        required_output_file_ids=[ f.id for f in files[2:] ],
        require_all_output_files=False,
        required_output_parameter_names=["weight", "torsion_stiffness", "max_stress"],
        require_all_output_parameters=False
    )
)

Add a fitness definition#

fd = FitnessDefinition(error_fitness=10.0)
fd.add_fitness_term(name="weight", type="design_objective", weighting_factor=1.0,
                    expression="map_design_objective( values['weight'], 7.5, 5.5)")
fd.add_fitness_term(name="torsional_stiffness", type="target_constraint", weighting_factor=1.0,
                expression="map_target_constraint( values['torsion_stiffness'], 1313.0, 5.0, 30.0 )" )
fd.add_fitness_term(name="max_stress", type="limit_constraint", weighting_factor=1.0,
                expression="map_limit_constraint( values['max_stress'], 451.0, 50.0 )")
cfg.fitness_definition =fd

Create a configuration (now job definition)#

cfg = proj.create_configurations([cfg])[0]

Create design points (now jobs)#

import random

dps = []
for i in range(10):
    values = {
        p.name: p.lower_limit + random.random() * (p.upper_limit - p.lower_limit)
        for p in float_input_params
    }
    values.update({ p.name: random.choice(p.value_list) for p in str_input_params})
    dps.append( DesignPoint( name=f"DesignPoint.{i}", values=values, eval_status="pending") )

dps = cfg.create_design_points(dps)