Notebook - Fractopo – KB11 Fracture Network Analysis
fractopo
enables fast and scalable analysis of two-dimensional georeferenced fracture and lineament datasets. These are typically created with remote sensing using a variety of background materials: fractures can be extracted from outcrop orthomosaics and lineaments from digital elevation models (DEMs). Collectively both lineament and fracture datasets can be referred to as trace datasets or as fracture networks.
fractopo
implements suggestions for structural geological studies by Peacock and Sanderson (2018):
Basic geological descriptions should be followed by measuring their geometries and topologies, understanding their age relationships, kinematic and mechanics, and developing a realistic, data-led model for related fluid flow.
fractopo
characterizes the individual and overall geometry and topology of fractures and the fracture network. Furthermore the age relations are investigated with determined topological cross-cutting and abutting relationship between fracture sets.
Whether fractopo
evolves to implement all the steps in the quote remains to be seen! The functionality grows as more use cases require implementation.
Imports
Avoid cluttering outputs with warnings.
[1]:
import warnings
warnings.filterwarnings("ignore", message="The Shapely GEOS")
warnings.filterwarnings("ignore", message="In a future version, ")
warnings.filterwarnings("ignore", message="No data for colormapping provided via")
warnings.filterwarnings(
"ignore", message="Shapely 2.0 is installed, but because PyGEOS is also installed"
)
geopandas
is the main module which fractopo
is based on. It along with shapely
and pygeos
implement all spatial operations required for two-dimensional fracture network analysis. geopandas
further implements all input-output operations like reading and writing spatial datasets (shapefiles, GeoPackages, GeoJSON, etc.).
[2]:
import geopandas as gpd
geopandas
uses matplotlib
for visualizing spatial datasets.
[3]:
import matplotlib.pyplot as plt
Miscellaneous imports.
[4]:
from pathlib import Path
Data
Fracture network data consists of georeferenced lineament or fracture traces, manually or automatically digitized, and a target area boundary that delineates the area in which fracture digiziting has been done. The boundary is important to handle edge effects in network analysis. fractopo
only has a stub (and untested) implementation for cases where no target area is given so I strongly recommend to always delineate the traced fractures and pass the target area to Network
.
geopandas
is used to read and write spatial datasets. Here we use geopandas
to both download and load trace and area datasets that are hosted on GitHub. A more typical case is that you have local files you wish to analyze in which case you can replace the url string with a path to the local file. E.g.
# Local trace data
trace_data_url = "~/data/traces.gpkg"
The example dataset here is from an island south of Loviisa, Finland. The island only consists of outcrop quite well polished by glaciations. The dataset is in ETRS-TM35FIN
coordinate reference system.
[5]:
traces_path = Path("../../tests/sample_data/KB11/KB11_traces.geojson")
area_path = Path("../../tests/sample_data/KB11/KB11_area.geojson")
traces = gpd.read_file(traces_path)
area = gpd.read_file(area_path)
# Name the dataset
name = "KB11"
Visualizing trace map data
geopandas
has easy methods for plotting spatial data along with data coordinates. The plotting is based on matplotlib
.
[6]:
# Initialize the figure and ax in which data is plotted
fig, ax = plt.subplots(figsize=(9, 9))
# Plot the loaded trace dataset consisting of fracture traces.
traces.plot(ax=ax, color="blue")
# Plot the loaded area dataset that consists of a single polygon that delineates the traces.
area.boundary.plot(ax=ax, color="red")
# Give the figure a title
ax.set_title(f"{name}, Coordinate Reference System = {traces.crs}")
[6]:
Text(0.5, 1.0, 'KB11, Coordinate Reference System = EPSG:3067')
Network
So far we have not used any fractopo
functionality, just geopandas
. Now we use the Network
class to create Network
instances that can be thought of as abstract representations of fracture networks. The fracture network contains traces and a target area boundary delineating the traces.
To characterize the topology of a fracture network fractopo
determines the topological branches and nodes (Sanderson and Nixon 2015).
Nodes consist of trace endpoints which can be isolated or snapped to end at another trace.
Branches consist of every trace segment between the aforementioned nodes.
Automatic determination of branches and nodes is determined with the determine_branches_nodes
keyword. If given as False
, they are not determined. You can still use the Network
object to investigate geometric properties of just the traces.
Network
initialization should be supplied with information regarding the trace dataset:
truncate_traces
If you wish to only characterize the network within the target area boundary, the input traces should be cropped to the target area boundary. This is done when
truncate_traces
is given asTrue.
True
recommended.
circular_target_area
If the target area is a circle
circular_target_area
should be given asTrue
. A circular target area is recommended to avoid orientation bias in node counting.
snap_threshold
To determine topological relationships between traces the abutments between traces should be snapped to some tolerance. This tolerance can be given here, in the same unit used for the traces. It represents the smallest distance between nodes for which
fractopo
interprets them as two distinct nodes. This value should be carefully chosen depending on the size of the area of interest. As a reference, when digitizing in QGIS with snapping turned on, the tolerance is probably much lower than0.001
.The trace validation functionality of
fractopo
can be (and should be) used to check that there are no topological errors within a certain tolerance.
[7]:
# Import the Network class from fractopo
from fractopo import Network
[8]:
# Create Network and automatically determine branches and nodes
# The Network instance is saved as kb11_network variable.
kb11_network = Network(
traces,
area,
name=name,
determine_branches_nodes=True,
truncate_traces=True,
circular_target_area=True,
snap_threshold=0.001,
)
Visualizing fracture network branches and nodes
We can similarly to the traces visualize the branches and nodes with geopandas
plotting.
[9]:
# Import identifier strings of topological branches and nodes
from fractopo.general import CC_branch, CI_branch, II_branch, X_node, Y_node, I_node
# Function to determine color for each branch and node type
def assign_colors(feature_type: str):
if feature_type in (CC_branch, X_node):
return "green"
if feature_type in (CI_branch, Y_node):
return "blue"
if feature_type in (II_branch, I_node):
return "black"
return "red"
Branch or Node Type |
Color |
---|---|
C - C, X |
Green |
C - I, Y |
Blue |
I - I, I |
Black |
Other |
Red |
Branches
[10]:
fix, ax = plt.subplots(figsize=(9, 9))
kb11_network.branch_gdf.plot(
colors=[assign_colors(bt) for bt in kb11_network.branch_types], ax=ax
)
area.boundary.plot(ax=ax, color="red")
[10]:
<Axes: >
Nodes
[11]:
fix, ax = plt.subplots(figsize=(9, 9))
# Traces
kb11_network.trace_gdf.plot(ax=ax, linewidth=0.5)
# Nodes
kb11_network.node_gdf.plot(
c=[assign_colors(bt) for bt in kb11_network.node_types], ax=ax, markersize=10
)
area.boundary.plot(ax=ax, color="red")
[11]:
<Axes: >
Geometric Fracture Network Characterization
The most basic geometric properties of traces are their length and orientation.
Length is the overall travel distance along the digitized trace. The length of traces individually is usually not interesting but the value distribution of all of the lengths is (Bonnet et al. 2001). fractopo
uses another Python package, powerlaw
, for determining power-law, lognormal and exponential distribution fits. The wrapper around powerlaw
is thin and therefore I urge you to see its
documentation and associated article for more info.
Orientation of a trace (or branch, or any line) can be defined in multiple ways that approach the same result when the line is sublinear:
Draw a straight line between the start and endpoints of the trace and calculate the orientation of that line.
This is the approach used in
fractopo
. Simple, but when the trace is curvy enough the simplification might be detrimental to analysis.
Plot each coordinate point of a trace and fit a linear regression trend line. Calculate the orientation of the trend line.
Calculate the orientation of each segment between coordinate points resulting in multiple orientation values for a single trace.
Length distributions
Traces
[12]:
# Plot length distribution fits (powerlaw, exponential and lognormal) of fracture traces using powerlaw
fit, fig, ax = kb11_network.plot_trace_lengths()
[13]:
# Fit properties
print(f"Automatically determined powerlaw cut-off: {fit.xmin}")
print(f"Powerlaw exponent: {fit.alpha - 1}")
print(
f"Compare powerlaw fit to lognormal: R, p = {fit.distribution_compare('power_law', 'lognormal')}"
)
Automatically determined powerlaw cut-off: 4.321478841775524
Powerlaw exponent: 1.6051888040287916
Compare powerlaw fit to lognormal: R, p = (-0.6749942925041004, 0.4105159341196074)
Branches
[14]:
# Length distribution of branches
fit, fig, ax = kb11_network.plot_branch_lengths()
[15]:
# Fit properties
print(f"Automatically determined powerlaw cut-off: {fit.xmin}")
print(f"Powerlaw exponent: {fit.alpha - 1}")
print(
f"Compare powerlaw fit to lognormal: R, p = {fit.distribution_compare('power_law', 'lognormal')}"
)
Automatically determined powerlaw cut-off: 2.5085762870774153
Powerlaw exponent: 3.475884087746053
Compare powerlaw fit to lognormal: R, p = (-0.4850754023456776, 0.5591241023355533)
Rose plots
A rose plot is a histogram where the bars have been oriented based on pre-determined bins. fractopo
rose plots are length-weighted and equal-area. Length-weighted means that each bin contains the total length of traces or branches within the orientation range of the bin.
The method for calculating the bins and reasoning for using equal-area rose plots is from publication by Sanderson and Peacock (2020).
[16]:
# Plot azimuth rose plot of fracture traces and branches
azimuth_bin_dict, fig, ax = kb11_network.plot_trace_azimuth()
[17]:
# Plot azimuth rose plot of fracture branches
azimuth_bin_dict, fig, ax = kb11_network.plot_branch_azimuth()
Topological Fracture Network Characterization
The determination of branches and nodes are essential for characterizing the topology of a fracture network. The topology is the collection of properties of the traces that do not change when the traces are transformed continously i.e. the traces are not cut but are extended or shrinked. In geological terms the traces can go through ductile transformation without losing their topological properties but not brittle transformation. Furthermore this means the end topology of the traces is a result of brittle transformation(s).
At its simplest the proportion of different types of branches and nodes are used to characterize the topology.
Branches can be categorized into three main categories:
C–C is connected at both endpoints
C-I is connected at one endpoint
I-I is not connected at either endpoint
Nodes can be similarly categorized into three categories:
X represents intersection between two traces
Y represents abutment of one trace to another
I represents isolated termination of a trace
Furthermore E node and any E-containing branch classification (e.g. I-E) are related to the trace area boundary. Branches are always cropped to the boundary and branches that are cut then have a E node as end endpoint.
Node and branch proportions
The proportion of the different types of nodes and branches have direct implications for the overall connectivity of a fracture network (Sanderson and Nixon 2015).
The proportions are plotted on ternary plots. The plotting uses python-ternary.
[18]:
kb11_network.node_counts
[18]:
{'X': 270, 'Y': 824, 'I': 478, 'E': 114}
[19]:
# Plot ternary XYI-node proportion plot
fig, ax, tax = kb11_network.plot_xyi()
[20]:
kb11_network.branch_counts
[20]:
{'C - C': 1521,
'C - I': 410,
'I - I': 28,
'C - E': 100,
'I - E': 12,
'E - E': 1}
[21]:
# Plot ternary branch (C-C, C-I, I-I) proportion plot
fig, ax, tax = kb11_network.plot_branch()
Crosscutting and abutting relationships
If the geometry and topology of the fracture network are investigated together the cross-cutting and abutting relationships between orientation-binned fracture sets can be determined. Traces can be binned into sets based on their orientation (e.g. N-S oriented traces could belong to Set 1 and E-W oriented traces to Set 2). If the endpoints of the traces of sets are examined the abutment relationship between can be determined i.e. which abuts which (e.g. does the N-S oriented Set 1 traces abut to E-W oriented Set 2 or do they crosscut each other equal amounts.)
[22]:
# Sets are defaults
print(f"Azimuth set names: {kb11_network.azimuth_set_names}")
print(f"Azimuth set ranges: {kb11_network.azimuth_set_ranges}")
Azimuth set names: ('1', '2', '3')
Azimuth set ranges: ((0, 60), (60, 120), (120, 180))
[23]:
# Plot crosscutting and abutting relationships between azimuth sets
figs, fig_axes = kb11_network.plot_azimuth_crosscut_abutting_relationships()
Numerical Fracture Network Characterization Parameters
The quantity, total length and other geometric and topological properties of the traces, branches and nodes within the target area can be determined as numerical values. For the following parameters I refer you to the following articles:
[24]:
kb11_network.parameters
[24]:
{'Fracture Intensity B21': 1.2633728710295158,
'Fracture Intensity P21': 1.2633728710295158,
'Trace Min Length': 5.820766091346741e-10,
'Trace Max Length': 33.1085306416674,
'Trace Mean Length': 2.2058247378420526,
'Dimensionless Intensity P22': 2.786779132035443,
'Area': 1237.9003657531266,
'Number of Traces (Real)': 709,
'Number of Traces': 651.0,
'Branch Min Length': 0.006639501944314653,
'Branch Max Length': 5.721558550556387,
'Branch Mean Length': 0.7761437911315212,
'Areal Frequency B20': 1.627756203766927,
'Areal Frequency P20': 0.5258904658323919,
'Dimensionless Intensity B22': 0.9805590097335628,
'Connections per Trace': 3.3609831029185866,
'Connections per Branch': 1.7627791563275435,
'Fracture Intensity (Mauldon)': 1.4357438039436288,
'Fracture Density (Mauldon)': 0.5258904658323919,
'Trace Mean Length (Mauldon)': 2.7301194777720483,
'Connection Frequency': 0.8837544848243267,
'Number of Branches': 2015.0,
'Number of Branches (Real)': 2072}
Contour Grids
To visualize the spatial variation of geometric and topological parameter values the network can be sampled with a grid of rectangles. From the center of each rectangle a sampling circle is made which is used to do the actual sampling following Nyberg et al. 2018 (See Fig. 3F).
Sampling with a contour grid is time-consuming and therefore the following code is not executed within this notebooks by default. The end result is embedded as images. Paste the code from the below cell blocks to a Python cell to execute them.
To perform sampling with a cell width of 0.75 \(m\):
sampled_grid = kb11_network.contour_grid(
cell_width=0.75,
)
To visualize results for parameter Fracture Intensity P21:
kb11_network.plot_contour(parameter="Fracture Intensity P21", sampled_grid=sampled_grid)
Result:
To visualize results for parameter Connections per Branch:
kb11_network.plot_contour(parameter="Connections per Branch", sampled_grid=sampled_grid)
Result:
Tests to verify notebook consistency
[25]:
assert hasattr(kb11_network, "contour_grid") and hasattr(kb11_network, "plot_contour")