Python Result fatigue calculation

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

Is there an example of fatigue calculation with DPF?

Tagged:

Answers

  • Member, Moderator, Employee Posts: 884
    100 Answers 500 Comments 250 Likes Second Anniversary
    ✭✭✭✭
    edited December 2023

    The following example has no physical meaning and is just an illustration of DPF capabilies.
    It uses the dummy custom stress computed in this example, and reads the S-N curve of the material of the first body in the tree (adapt to your use case!).

    Note - a value of -1 or -2 in the result plot will indicate that the interpolation of the number of cycles to failure could not be performed (usually due to lack of material data to interpolate).

    Complete code:

    1. def LinearInterpolation(x,x1,y1,x2,y2):
    2. # Simple linear interpolation
    3. # Returns -1 if x is outside of [x1,x2]
    4. if x<x1 or x>x2:
    5. return -1 # out of range
    6. if y1==y2:
    7. return y1
    8. else:
    9. a=(y2-y1)/(x2-x1)
    10. b=y1-a*x1
    11. return a*x+b
    12.  
    13. def GetLocalCyclesToFailure(sn_curve, local_stress_si):
    14. cycles_list = sn_curve["Cycles"][1:]
    15. alternating_stress_list = sn_curve["Alternating Stress"][1:]
    16. index = 0
    17. # Find closest cycle number in stress_data
    18. for i in range(len(cycles_list)):
    19. alt_stress = alternating_stress_list[i]
    20. if alt_stress < local_stress_si:
    21. index=i
    22. break
    23. # Get (cycles, alternating stress) points between which to interpolate
    24. if index!= len(cycles_list):
    25. y1=cycles_list[index]
    26. y2=cycles_list[index-1]
    27. x1=alternating_stress_list[index]
    28. x2=alternating_stress_list[index-1]
    29. else:
    30. return -2 # value is out of bounds
    31. return LinearInterpolation(local_stress_si,x1,y1,x2,y2) # return number of cycles
    32.  
    33. def GetFieldCyclesToFailure(stress_op, sn_curve, analysis):
    34. import mech_dpf
    35. import Ans.DataProcessing as dpf
    36. node_ids = stress_op.outputs.getfields_container()[0].ScopingIds
    37. limit_field = dpf.FieldsFactory.CreateScalarField(1,'Nodal')
    38. stress_data = stress_op.outputs.getfields_container()[0]
    39. for nid in node_ids:
    40. local_stress =stress_data.GetEntityDataById(nid)[0]
    41. # Convert local stress to SI units
    42. local_stress_quantity = Quantity(str(local_stress) + '[' + analysis.CurrentConsistentUnitFromQuantityName("Stress") + ']')
    43. local_stress_si = local_stress_quantity.ConvertToUnitSystem("SI", "Stress").Value
    44. limit_field.Add(nid,[float(GetLocalCyclesToFailure(sn_curve, local_stress_si))])
    45. return limit_field
    46.  
    47.  
    48. def define_dpf_workflow(analysis):
    49. import mech_dpf
    50. import Ans.DataProcessing as dpf
    51. mech_dpf.setExtAPI(ExtAPI)
    52. data_source = dpf.DataSources(analysis.ResultFileName)
    53. model = dpf.Model(data_source)
    54. mesh = model.Mesh
    55.  
    56. # Get SN curves for material defined on first body in the Mechanical tree
    57. import materials
    58. first_body = Model.GetChildren(DataModelObjectCategory.Body, True)[0]
    59. material = ExtAPI.DataModel.GetObjectsByName(first_body.Material)[0]
    60. mat_enginerring_data = material.GetEngineeringDataMaterial()
    61. sn_curve = materials.GetMaterialPropertyByName(mat_enginerring_data,"S-N Curve")
    62.  
    63. # Time sets
    64. timePointsOp =dpf.operators.metadata.time_freq_provider()
    65. timePointsOp.inputs.data_sources.Connect(data_source)
    66. timepoints = dpf.operators.metadata.time_freq_support_get_attribute(time_freq_support=timePointsOp,property_name="time_freqs",)
    67.  
    68. # S1
    69. prin1 = dpf.operators.result.stress_principal_1()
    70. prin1.inputs.data_sources.Connect(data_source)
    71. prin1.inputs.time_scoping.Connect(timepoints)
    72. # S3
    73. prin3 = dpf.operators.result.stress_principal_3()
    74. prin3.inputs.data_sources.Connect(data_source)
    75. prin3.inputs.time_scoping.Connect(timepoints)
    76. # S1-S3
    77. minus_fc = dpf.operators.math.minus_fc()
    78. minus_fc.inputs.field_or_fields_container_A.Connect(prin1.outputs.fields_container)
    79. minus_fc.inputs.field_or_fields_container_B.Connect(prin3.outputs.fields_container)
    80. # Scale factor
    81. scale = dpf.operators.math.scale_fc()
    82. scale.inputs.fields_container.Connect(minus_fc)
    83. scale.inputs.ponderation.Connect(sqrt(2)/2)
    84.  
    85. # Get alternate stress
    86. min_max = dpf.operators.min_max.min_max_over_time_by_entity()
    87. min_max.inputs.fields_container.Connect(scale)
    88. min_field_by_time = min_max.outputs.getmin()
    89. max_field_by_time = min_max.outputs.getmax()
    90. alternate_stress = dpf.operators.math.minus_fc()
    91. alternate_stress.inputs.field_or_fields_container_A.Connect(min_max.outputs.getmax())
    92. alternate_stress.inputs.field_or_fields_container_B.Connect(min_max.outputs.getmin())
    93.  
    94. # Get cycles to failure
    95. new_field = GetFieldCyclesToFailure(alternate_stress, sn_curve, analysis)
    96. output = dpf.operators.utility.forward()
    97. output.inputs.any.Connect(new_field)
    98.  
    99. # Plot
    100. dpf_workflow = dpf.Workflow()
    101. dpf_workflow.Add(output)
    102. dpf_workflow.SetOutputContour(output)
    103. dpf_workflow.Record('wf_id', False)
    104. this.WorkflowId = dpf_workflow.GetRecordedId()
    105.  
    106.  
    107.  
    108.  

Welcome!

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