import h5py
import numpy as np
from .utils.common import init_axis, sind, cosd
from .para import ATTPara
import copy
[docs]
class Checker():
"""Create checkerboard model by adding perturbations on an exist model
"""
def __init__(self, model_fname:str, para_fname='input_params.yml') -> None:
"""Initialize Checker object
:param fname: Path to initial model file
:type fname: str
:param para_fname: Path to parameter file, defaults to 'input_params.yml'
:type para_fname: str, optional
"""
self.model_file = model_fname
self.para_fname = para_fname
with h5py.File(model_fname) as f:
self.vel = f['vel'][:]
self.eta = f['eta'][:]
self.xi = f['xi'][:]
self.zeta = f['zeta'][:]
self._init_axis()
def _init_axis(self):
"""Initialize axis
:param min_max_dep: min and max depth, ``[min_dep, max_dep]``
:type min_max_dep: list
:param min_max_lat: Min and max latitude, ``[min_lat, max_lat]``
:type min_max_lat: list
:param min_max_lon: Min and max longitude, ``[min_lon, max_lon]``
:type min_max_lon: list
:param n_rtp: number of dimensions [ndep, nlat, nlon]
:type n_rtp: list
"""
para = ATTPara(self.para_fname)
n_rtp = para.input_params['domain']['n_rtp']
min_max_dep = para.input_params['domain']['min_max_dep']
min_max_lat = para.input_params['domain']['min_max_lat']
min_max_lon = para.input_params['domain']['min_max_lon']
self.dd, self.tt, self.pp, self.dr, self.dt, self.dp, = init_axis(
min_max_dep, min_max_lat, min_max_lon, n_rtp
)
def _create_taper(self, xleft, xright, type='d'):
if type == 'd':
x = np.flip(self.dd); dx = self.dr
elif type == 't':
x = self.tt; dx = self.dt
elif type == 'p':
x = self.pp; dx = self.dp
else:
pass
if xleft < x[0] or xright > x[-1]:
raise ValueError('limitation out of range')
ntaper_left = int((xleft-x[0])/dx)
ntaper_right = int((x[-1]-xright)/dx)
return ntaper_left, ntaper_right
[docs]
def checkerboard(self, n_pert_x:int, n_pert_y:int, n_pert_z:int,
pert_vel=0.08, pert_ani=0.04, ani_dir=45,
lim_x=None, lim_y=None, lim_z=None):
"""Create checkerboard
:param n_pert_x: Multiple of period along X, e.g., set to 1 for 2 anomalies
:type n_pert_x: float
:param n_pert_y: Multiple of period along Y
:type n_pert_y: float
:param n_pert_z: Multiple of period along Z
:type n_pert_z: float
:param pert_vel: Perturbation for velocity, defaults to ``0.08``
:type pert_vel: float, optional
:param pert_ani: Perturbation for anisotropy, defaults to ``0.04``
:type pert_ani: float, optional
:param ani_dir: Direction of fast velocity direction, defaults to ``45``
:type ani_dir: float, optional
:param lim_x: Left and right bound along X, defaults to ``None``
:type lim_x: list, optional
:param lim_y: Left and right bound along Y, defaults to ``None``
:type lim_y: list, optional
:param lim_z: Left and right bound along Z, defaults to ``None``
:type lim_z: list, optional
"""
if lim_x is not None:
ntaper_left, ntaper_right = self._create_taper(*lim_x, type='p')
else:
ntaper_left = 0
ntaper_right = 0
x_pert = np.zeros_like(self.pp)
x_pert[ntaper_left:self.pp.size-ntaper_right] = \
np.sin(n_pert_x*np.pi*np.arange(self.pp.size-(ntaper_left+ntaper_right))/ \
(self.pp.size-(ntaper_left+ntaper_right)))
if lim_y is not None:
ntaper_left, ntaper_right = self._create_taper(*lim_y, type='t')
else:
ntaper_left = 0
ntaper_right = 0
y_pert = np.zeros_like(self.tt)
y_pert[ntaper_left:self.tt.size-ntaper_right] = \
np.sin(n_pert_y*np.pi*np.arange(self.tt.size-(ntaper_left+ntaper_right))/ \
(self.tt.size-(ntaper_left+ntaper_right)))
if lim_z is not None:
ntaper_left, ntaper_right = self._create_taper(*lim_z, type='d')
else:
ntaper_left = 0
ntaper_right = 0
z_pert = np.zeros_like(self.dd)
z_pert[ntaper_right:self.dd.size-ntaper_left] = \
np.sin(n_pert_z*np.pi*np.arange(self.dd.size-(ntaper_left+ntaper_right))/ \
(self.dd.size-(ntaper_left+ntaper_right)))
xx, yy, zz= np.meshgrid(z_pert, y_pert, x_pert, indexing='ij')
self.perturbation = xx*yy*zz
self.vel_pert = self.vel * (1+self.perturbation*pert_vel)
self.dlnv = self.perturbation*pert_vel
self.epsilon = np.abs(self.perturbation)*pert_ani
self.phi = np.zeros_like(self.vel)
self.phi[np.where(self.perturbation>0)] = ani_dir
self.phi[np.where(self.perturbation<0)] = ani_dir+90
self.xi = self.epsilon*cosd(2*self.phi)
self.eta = self.epsilon*sind(2*self.phi)
[docs]
def copy(self):
"""Create a deep copy of the Checker object
:return: A copy of the Checker object
:rtype: Checker
"""
return copy.deepcopy(self)
[docs]
def write(self, fname):
"""Write new model to h5 file
:param fname: Path to output file
:type fname: str
"""
if fname is None:
fname = '.'.join(self.model_file.split('.')[:-1])+'_pert.h5'
with h5py.File(fname, 'w') as f:
f.create_dataset('xi', data=self.xi)
f.create_dataset('eta', data=self.eta)
f.create_dataset('zeta', data=self.zeta)
f.create_dataset('vel', data=self.vel_pert)
f.create_dataset('epsilon', data=self.epsilon)
f.create_dataset('phi', data=self.phi)
f.create_dataset('dlnv', data=self.dlnv)