Create a field and plot results

Member Posts: 5
Photogenic First Comment
**
edited June 2023 in Structures

Hey,

I want to investigate the fatigue of given loads steps with DPF in Python. Therefore, I extract all data from the *.rst file and compute my damage value for each node (e.g., for a named selection). Finally, this results in a list of node ids (resulting from the named selection) and associated damage values (from my computation).

Can you please help me to create a "new" field (e.g., damage)? How can I plot this results/field in a 3D contour plot?

Thank you,

Thomas

Tagged:

Welcome!

It looks like you're new here. Sign in or register to get started.

Best Answers

  • Member, Moderator, Employee Posts: 888
    100 Answers 500 Comments 250 Likes Second Anniversary
    ✭✭✭✭
    Answer ✓

    Glad you like those posts! And nice to see that you've made great progress! You can plot the result on custom space domains, see this page for some examples: https://dpf.docs.pyansys.com/examples/00-basic/09-results_over_space_subset.html#sphx-glr-examples-00-basic-09-results-over-space-subset-py

  • Member Posts: 5
    Photogenic First Comment
    **
    edited January 2023 Answer ✓

    Below you will find a summary of the source code.

    1. # -*- coding: utf-8 -*-
    2. """
    3. Created on Wed Dec 28 12:37:46 2022
    4. @author: t.lauss
    5. """
    6.  
    7. from ansys.dpf import core as dpf
    8. from ansys.dpf.core import operators as ops
    9. import functions # contains all functions for fatigue computation
    10. import os
    11.  
    12. # load result files
    13. file_name = r"file.rst"
    14. link_file = r"...path_to_file..."
    15. your_result_file = os.path.join(link_file,file_name)
    16.  
    17. # load model and print model infos
    18. model = dpf.Model(your_result_file)
    19.  
    20. # Print available named selections
    21. print('Named selections in model:', model.metadata.available_named_selections)
    22.  
    23. # Restrict data to a specific named selection
    24. ns_operator = ops.scoping.on_named_selection()
    25. ns_operator.inputs.data_sources.connect(model)
    26. ns_operator.inputs.named_selection_name.connect('ROD_CYL_FACE')
    27. # Get node numbers for nodes in named selection
    28. mesh_data = ns_operator.outputs.mesh_scoping.get_data()
    29. node_ids = mesh_data.ids
    30.  
    31. # Extract principal streses on the named selection
    32. principal_stress = ops.result.stress_principal_1(data_sources=model)
    33.  
    34. # Create list for result sets (every time id is associated to a load step for fatigue analysis)
    35. time_ids = list(range(1, model.metadata.time_freq_support.n_sets+1))
    36.  
    37. # Restrict scoping to named selection
    38. principal_stress.inputs.mesh_scoping.connect(ns_operator.outputs.mesh_scoping)
    39. principal_stress.inputs.time_scoping.connect(time_ids)
    40. # extract fields for named selection and load steps
    41. fields = principal_stress.outputs.fields_container.get_data()
    42.  
    43. # number of nodes
    44. N_nodes = len(mesh_data.ids)
    45. # number of time steps
    46. N_time = len(time_ids)
    47.  
    48. # Parameter for fatigue calculation
    49. ...
    50.  
    51. # Damage computation for all nodes of named selection
    52. damage = N_nodes*[0.0]
    53. for i in range(0,N_nodes):# loop over all nodes
    54.     node_id = node_ids[i] # node id
    55.     s1 = N_time*[0.0]     # temporary list of maximum principla stress for one node and all time steps
    56.     idRed = N_time*[0]    # load step ID for stress calculation
    57.     for j in range(0,N_time): # loop obrt all time steps
    58.         # extract maximum principal stress for time step and node
    59.         s1[j] = fields[j].get_entity_data_by_id(node_ids[i])[0]
    60.         # get reduced pivoted load step IDs
    61.         idRed[j] = N_time-j
    62.     
    63.     # Initialize damage value with zero
    64.     Di = 0.0
    65.     
    66.     # associate stress data with fatigue spectrum (with reduced ID and load IDs)
    67.     # get list of stresses to be classified by rainflow counting
    68.     data = functions.nodeHashListLeewardLC1(idRed,s1) 
    69.     # rainflow clasification
    70.     res = functions.rainflowCount(data)
    71.     N_Count = len(res)
    72.     # Damage calculation for each rainflow cycle
    73.     for k in range(0,N_Count):
    74.         smax = res[k][0][0]
    75.         smin = res[k][0][1]
    76.         count = res[k][1]
    77.         Di = Di + functions.getDamage(...)
    78.     
    79.     # ...Do this for all load cases...
    80.     
    81.     # Store accumulated damage value in list
    82.     damage[i] = Di
    83.     # Print infos to console
    84.     print(node_id, Di, i/N_nodes*100, "%")
    85.  
    86.  
    87. # get the location and the max. Damage value
    88. maxDamage = max(damage)
    89. maxIndex  = damage.index(maxDamage)
    90. print("max. Damage: ", max(damage), "at Node: ", node_ids[maxIndex])
    91.  
    92.  
    93. ################################################################
    94. ############# Plot the damage in a 3d contour plot #############
    95. ################################################################
    96.  
    97. # create a field which contains the damage value at particular nodes
    98. field_damage = dpf.fields_factory.create_scalar_field(N_nodes, dpf.locations.nodal)
    99. # store data in field at nodes
    100. for i in range(0,N_nodes):
    101.     field_damage.append(damage[i], node_ids[i])
    102.  
    103. # extract mesh for body with named selection name "RODEND"
    104. mesh_scoping = model.metadata.named_selection("RODEND")
    105. mesh = model.metadata.meshed_region
    106. op = dpf.operators.mesh.from_scoping(scoping=mesh_scoping, inclusive=1, mesh=mesh)
    107. result_mesh = op.outputs.mesh()
    108. # assign mesh to field that contains the damage values
    109. field_damage.meshed_region = result_mesh
    110. # plot the damage values
    111. sargs = dict(title="damage", fmt="%.2e")
    112. field_damage.plot(scalar_bar_args=sargs, meshed_region=result_mesh, screenshot='image.png',
    113.                   cpos=[(0.0, -80.0, 20.0), (-0.1, -0.1, -1.0), (0.0, 0.8, -0.1)])
    114. # You can set the camera positions using the ``cpos`` argument.
    115. # The three tuples in the list for the ``cpos`` argument represent the camera
    116. # position, focal point, and view respectively.


Answers

  • Member, Moderator, Employee Posts: 888
    100 Answers 500 Comments 250 Likes Second Anniversary
    ✭✭✭✭

    Hi @t.lauss , to create a new field, use dpf.fields_factory. An example is available here: https://dpf.docs.pyansys.com/examples/00-basic/03-create_entities.html

    Plotting options are explained there: https://dpf.docs.pyansys.com/examples/05-plotting/00-basic_plotting.html#sphx-glr-examples-05-plotting-00-basic-plotting-py

    If you're looking into plotting the results in Workbench Mechanical, you will have to use a Python Result. This post could be helpful: https://www.linkedin.com/pulse/script-tip-friday-how-sum-damage-results-structural-analysis/

  • Member Posts: 5
    Photogenic First Comment
    **

    Thank you for your help. The examples and the skript tips on friday are excellent, i absolutely like this.

    I created a scalar field with the dpf.fields_factory and store all damage values and the corresponding node ids in the field. I think this is working now.

    The problem now is that the results at the named selcetion (see Fig. 1) is hidden by other bodies (see Fig. 2). Is it possible to plot only the mesh and fileds of the named selection and/or is it possible to hide bodies during plotting?


  • Member Posts: 5
    Photogenic First Comment
    **

    I had already tried to create the plot using this example, but unfortunately I did not succeed.

    Here is my code (damage is a list with the damage values and node_ids is a list of same size with the associated node numbers)

    1. # create a field which contains the damage value at particular nodes
    2. # this is a “reserve” mechanism, not a resize one. This means that you need to append data to grow the size of your field.
    3. field_damage = dpf.fields_factory.create_scalar_field(N_nodes, dpf.locations.nodal)
    4. # store data in field at nodes
    5. for i in range(0,N_nodes):
    6.     field_damage.append(Damage[i], node_ids[i])
    7.  
    8.  
    9. # create a mesh scoping for the named selection "ROD_CYL_FACE"
    10. mesh_scoping = model.metadata.named_selection("ROD_CYL_FACE")
    11. print(mesh_scoping)
    12.  
    13. #???????????????? NOT WORKING ????????????????
    14. field_damage.meshed_region = ops.mesh.from_scoping(mesh_scoping)
    15.  
    16. # plot 
    17. sargs = dict(title="damage", fmt="%.2e")
    18. field_damage.plot(scalar_bar_args=sargs)

    Output of print(mesh_scoping): DPF Scoping:  with Nodal location and 2175 entities

    After the mesh_scoping I don't really know how to proceed. How can I associate the mesh_scoping and the field_damage?

    Moreover, in the mentioned DPF examples I do not found how to plot only the results of one body and the other bodys are not visible. My problem is that the interesting damage region is hidden by other bodys and therefore not visible for me.

  • Member Posts: 5
    Photogenic First Comment
    **

    Thank you for your help! My code works now 😀


  • Member, Moderator, Employee Posts: 888
    100 Answers 500 Comments 250 Likes Second Anniversary
    ✭✭✭✭

    Thanks @t.lauss , happy to help! If you are willing to share, feel free to post your code as an example here for others to benefit from.

  • Member, Moderator, Employee Posts: 888
    100 Answers 500 Comments 250 Likes Second Anniversary
    ✭✭✭✭

    Thanks a lot for sharing @t.lauss ! I'm sure this will be of great help to other users!

Welcome!

It looks like you're new here. Sign in or register to get started.