Coverage for /usr/local/lib/python3.11/dist-packages/grond/problems/vlvd/problem.py: 54%
52 statements
« prev ^ index » next coverage.py v6.5.0, created at 2025-04-03 09:31 +0000
« prev ^ index » next coverage.py v6.5.0, created at 2025-04-03 09:31 +0000
1# https://pyrocko.org/grond - GPLv3
2#
3# The Grond Developers, 21st Century
4import logging
6from pyrocko import gf, util
7from pyrocko.guts import String, Float, Dict
9from grond.meta import Forbidden, expand_template, Parameter, \
10 has_get_plot_classes
12from ..base import Problem, ProblemConfig
14guts_prefix = 'grond'
15logger = logging.getLogger('grond.problems.volume_point')
16km = 1e3
17as_km = dict(scale_factor=km, scale_unit='km')
18as_km3 = dict(scale_factor=km**3, scale_unit='km^3')
21class VLVDProblemConfig(ProblemConfig):
22 ranges = Dict.T(String.T(), gf.Range.T())
23 distance_min = Float.T(default=0.0)
25 def get_problem(self, event, target_groups, targets):
26 self.check_deprecations()
27 if event.depth is None:
28 event.depth = 0.
30 base_source = gf.VLVDSource.from_pyrocko_event(event)
31 base_source.stf = gf.HalfSinusoidSTF(duration=event.duration or 0.0)
33 subs = dict(
34 event_name=event.name,
35 event_time=util.time_to_str(event.time))
37 problem = VLVDProblem(
38 name=expand_template(self.name_template, subs),
39 base_source=base_source,
40 target_groups=target_groups,
41 targets=targets,
42 ranges=self.ranges,
43 distance_min=self.distance_min,
44 norm_exponent=self.norm_exponent)
46 return problem
49@has_get_plot_classes
50class VLVDProblem(Problem):
52 problem_parameters = [
53 Parameter('north_shift', 'm', label='Northing', **as_km),
54 Parameter('east_shift', 'm', label='Easting', **as_km),
55 Parameter('depth', 'm', label='Depth', **as_km),
57 Parameter('dip', 'deg', label='Dip'),
58 Parameter('azimuth', 'deg', label='Azimuth'),
59 Parameter('clvd_moment', 'Nm', label='CLVD Moment'),
60 Parameter('volume_change', 'm^3', label='Volume Change', **as_km3),
62 ]
64 problem_waveform_parameters = [
65 Parameter('time', 's', label='Time'),
66 Parameter('duration', 's', label='Duration')
67 ]
69 distance_min = Float.T(default=0.0)
71 def pack(self, source):
72 arr = self.get_parameter_array(source)
73 for ip, p in enumerate(self.parameters):
74 if p.name == 'time':
75 arr[ip] -= self.base_source.time
76 return arr
78 def get_source(self, x):
79 d = self.get_parameter_dict(x)
81 p = {k: float(self.ranges[k].make_relative(self.base_source[k], d[k]))
82 for k in self.base_source.keys() if k in d}
84 stf = None
85 if self.has_waveforms:
86 stf = gf.HalfSinusoidSTF(duration=float(d.duration))
88 source = self.base_source.clone(stf=stf, **p)
89 return source
91 def preconstrain(self, x):
92 source = self.get_source(x)
93 if any(self.distance_min > source.distance_to(t)
94 for t in self.waveform_targets + self.phase_pick_targets):
95 raise Forbidden()
97 return x
99 @classmethod
100 def get_plot_classes(cls):
101 from . import plot
102 plots = super(VLVDProblem, cls).get_plot_classes()
103 plots.extend([plot.VLVDLocationPlot])
104 return plots