This code uses Okada's (1992) DC3D function to compute elastic displacements, strains, and stresses in an elastic half-space due to fault slip. It performs a similar type of calculation as Coulomb, and in fact reads Coulomb input files. I wrote this tool to iterate faster than I could using the Coulomb GUI. On test cases, it reproduces the Coulomb outputs (see Examples).
This code uses Python3, numpy, matplotlib, and Pygmt (https://www.pygmt.org/dev/). This code also requires you to have Ben Thompson's Okada Python wrapper on your pythonpath (https://github.com/tbenthompson/okada_wrapper). It requires a few utility functions in a separate utilities repository (https://github.com/kmaterna/Tectonic_Utils). To get the utilities library, please pip install Tectonic-Utils
.
To install, you can clone this library onto your computer into a location that is on your $PYTHONPATH (in other words, the parent directory holding Elastic_stresses_py/ must be on your $PYTHONPATH). For convenience, it is nice (but not required) to put the full path to Elastic_stresses_py/PyCoulomb/bin on your regular system $PATH so the main executable elastic_stresses_driver.py
can be found.
Most of the behavior of the program is controlled by a config text file. From an experiment directory on your system, you can generate a default config file in the current working directory (.) with:
elastic_stresses_config_writer.py .
You should change the parameters to your own experiment needs. An example config file is provided in examples/. The elastic parameters mu and lamda, as well as your input/output options, are set in config file.
Then, you call the program by passing the config file into the main executable:
elastic_stresses_driver.py my_config.txt
- Reads source and receiver faults from .inp formatted Coulomb input files.
- Reads source and receiver faults from .intxt files, a more convenient input format wheere slip is specified by length/width or by Wells and Coppersmith (1994) scaling rules
- Reads point sources and receiver faults from .inzero, a focal mechanism / moment tensor input format
- Takes a single receiver fault and splits it into subfaults in the dip- and strike- directions.
- Computes elastic displacements and stresses due to slip on source faults.
- Writes .inp formatted Coulomb files with split subfaults.
- Plots displacements at the surface.
- Plots shear stress, normal stress, and Coulomb stress on receiver faults.
- Maps the faults and stress values using PyGMT.
- Produces output tables of stresses and displacements.
- Output computations at depths other than the surface
- Read in full moment tensor (not just double couple focal mechanisms)
- Read .inr files (like Coulomb)
Source Faults (or faults that have slip on them) and Receiver Faults (or faults that receive stress from slip on source faults) can be specified in several human-readable formats beyond the .inp file that Coulomb uses. Each fault is specified by a row in an input text file of extention .intxt or .inzero. Valid rows of the input file include:
- General Format: Describes coordinate system and domain setup. Required.
- "General: poissons_ratio friction_coef lon_min lon_max lon_zero lat_min lat_max lat_zero"
- Receiver Format: Describes receiver faults
- "Receiver: strike rake dip length_km width_km lon lat depth_km"
- Slip Format: For slip distributions and fault patches.
- "Source_Patch: strike rake dip length_km width_km lon lat depth_km slip_m (opt: tensile_m)"
- WC Format: For catalogs using Wells and Coppersmith (1994)
- "Source_WC: strike rake dip magnitude faulting_type lon lat depth_km"
- FM Format: For focal mechanisms
- "Source_FM: strike rake dip lon lat depth_km magnitude mu lambda"
- MT Format: * For full moment tensors, although only double-couple part works right now
- "Source_MT: Mrr Mtt Mpp Mrt Mrp Mtp strike rake dip lon lat depth_km mu lambda"
- Horizontal Profile Format: Specify an orientation and compute stresses on that plane/orientation over an area. Like a horizontal cross-section.
- "Receiver_Horizontal_Profile: depth_km strike dip rake centerlon centerlat length_km width_km inc_km"
For all finite sources (i.e., Patch, WC), lon/lat/depth refer to the back updip corner of the fault plane, the corner where one looks along the strike direction to see the fault's upper edge (start_x and start_y in the Aki and Richards (1980) diagram in Coulomb's documentation).
For WC Source Format, faulting_type = ["SS","R","N","ALL"] from the Wells and Coppersmith indications of Strike Slip, Reverse, Normal, and All.
An example .intxt file with a "Slip Format" source might look like:
# General: poissons_ratio friction_coef lon_min lon_max lon_zero lat_min lat_max lat_zero
# Source_Patch: strike rake dip length_km width_km lon lat depth_km slip_m
# Receiver: strike rake dip length_km width_km lon lat depth_km
General: 0.25 0.40 -125.80 -122.60 -124.50 39.30 41.70 40.30
Source_Patch: 228.0 -2.0 79.0 44.25 11.91 -125.126 40.834 16.40 1.124
Receiver: 355.0 90.0 12.0 140.00 120.00 -124.560 40.300 10.80
By convention, right lateral strike slip is positive, and reverse dip slip is positive. Strike is defined from 0 to 360 degrees, clockwise from north; dip is defined from 0 to 90 degrees by the right hand rule. As in Coulomb, positive shear stress is towards failure, and positive normal stress is unclamping. The original Okada documentation can be found at http://www.bosai.go.jp/study/application/dc3d/DC3Dhtml_E.html.
For input files, strike/rake/dip have units of degrees. Length/width/depth have units of km. Slip has units of meters.
PR1 in the Coulomb input format is never used. Specifying Poisson's ratio should be done at the computation level through lame1 and mu in the config file, thus altering alpha in the Okada formulation.
Mu and Lambda in the Source_FM and Source_MT format are only ever used for computing potency, not for the rest of the computation. Set the domain elastic parameters at the computation level through lame1 and mu1 in the config file.
Elastic_stresses_py reproduces the Coulomb outputs in the simple case of two vertical strike-slip faults, one source (green) and one receiver (blue):
Python-produced Coulomb Stress Changes:
Coulomb-produced Coulomb Stress Changes:
In a real research application, a computation would look more like this: