1 | import numpy as np
|
---|
2 |
|
---|
3 | from checkfield import checkfield
|
---|
4 | from fielddisplay import fielddisplay
|
---|
5 | from project3d import project3d
|
---|
6 | from structtoobj import structtoobj
|
---|
7 | from WriteData import WriteData
|
---|
8 |
|
---|
9 |
|
---|
10 | class spatiallinearbasalforcings(object):
|
---|
11 | """SPATIAL LINEAR BASAL FORCINGS class definition
|
---|
12 |
|
---|
13 | Usage:
|
---|
14 | spatiallinearbasalforcings = spatiallinearbasalforcings()
|
---|
15 | """
|
---|
16 |
|
---|
17 | def __init__(self, *args): #{{{
|
---|
18 | nargs = len(args)
|
---|
19 | if nargs == 0:
|
---|
20 | self.groundedice_melting_rate = np.nan
|
---|
21 | self.deepwater_melting_rate = np.nan
|
---|
22 | self.deepwater_elevation = np.nan
|
---|
23 | self.upperwater_elevation = np.nan
|
---|
24 | self.geothermalflux = np.nan
|
---|
25 |
|
---|
26 | self.setdefaultparameters()
|
---|
27 | elif nargs == 1:
|
---|
28 | lb = args[0]
|
---|
29 | if lb.__module__ == 'linearbasalforcings':
|
---|
30 | nvertices = len(lb.groundedice_melting_rate)
|
---|
31 | self.groundedice_melting_rate = lb.groundedice_melting_rate
|
---|
32 | self.deepwater_melting_rate = lb.geothermalflux
|
---|
33 | self.deepwater_elevation = lb.deepwater_elevation * np.ones((nvertices, ))
|
---|
34 | self.upperwater_elevation = lb.deepwater_melting_rate * np.ones((nvertices, ))
|
---|
35 | self.geothermalflux = lb.upperwater_elevation * np.ones((nvertices, ))
|
---|
36 | else:
|
---|
37 | # TODO: This has not been tested
|
---|
38 | self = structtoobj(self, lb)
|
---|
39 | else:
|
---|
40 | raise Exception('constructor not supported')
|
---|
41 | #}}}
|
---|
42 |
|
---|
43 | def __repr__(self): #{{{
|
---|
44 | s = ' spatial linear basal forcings parameters:\n'
|
---|
45 | s += '{}\n'.format(fielddisplay(self, 'groundedice_melting_rate', 'basal melting rate (positive if melting) [m/yr]'))
|
---|
46 | s += '{}\n'.format(fielddisplay(self, 'deepwater_melting_rate', 'basal melting rate (positive if melting applied for floating ice whith base < deepwater_elevation) [m/yr]'))
|
---|
47 | s += '{}\n'.format(fielddisplay(self, 'deepwater_elevation', 'elevation of ocean deepwater [m]'))
|
---|
48 | s += '{}\n'.format(fielddisplay(self, 'upperwater_elevation', 'elevation of ocean upperwater [m]'))
|
---|
49 | s += '{}\n'.format(fielddisplay(self, 'geothermalflux', 'geothermal heat flux [W/m^2]'))
|
---|
50 | return s
|
---|
51 | #}}}
|
---|
52 |
|
---|
53 | def extrude(self, md): #{{{
|
---|
54 | self.groundedice_melting_rate = project3d(md, 'vector', self.groundedice_melting_rate, 'type', 'node', 'layer', 1)
|
---|
55 | self.deepwater_melting_rate = project3d(md, 'vector', self.deepwater_melting_rate, 'type', 'node', 'layer', 1)
|
---|
56 | self.deepwater_elevation = project3d(md, 'vector', self.deepwater_elevation, 'type', 'node', 'layer', 1)
|
---|
57 | self.upperwater_elevation = project3d(md, 'vector', self.upperwater_elevation, 'type', 'node', 'layer', 1)
|
---|
58 | self.geothermalflux = project3d(md, 'vector', self.geothermalflux, 'type', 'node', 'layer', 1) # Bedrock only gets geothermal flux
|
---|
59 | return self
|
---|
60 | #}}}
|
---|
61 |
|
---|
62 | def initialize(self, md): #{{{
|
---|
63 | if np.all(np.isnan(self.groundedice_melting_rate)):
|
---|
64 | self.groundedice_melting_rate = np.zeros((md.mesh.numberofvertices))
|
---|
65 | print(' no basalforcings.groundedice_melting_rate specified: values set as zero')
|
---|
66 | return self
|
---|
67 | #}}}
|
---|
68 |
|
---|
69 | def setdefaultparameters(self): #{{{
|
---|
70 | return self
|
---|
71 | #}}}
|
---|
72 |
|
---|
73 | def checkconsistency(self, md, solution, analyses): #{{{
|
---|
74 | if 'MasstransportAnalysis' in analyses and not solution == 'TransientSolution' and not md.transient.ismasstransport:
|
---|
75 | md = checkfield(md, 'fieldname', 'basalforcings.groundedice_melting_rate', 'NaN', 1, 'Inf', 1, 'timeseries', 1)
|
---|
76 | md = checkfield(md, 'fieldname', 'basalforcings.deepwater_melting_rate', 'NaN', 1, 'Inf', 1, 'timeseries', 1, '>=', 0)
|
---|
77 | md = checkfield(md, 'fieldname', 'basalforcings.deepwater_elevation', 'NaN', 1, 'Inf', 1, 'timeseries', 1)
|
---|
78 | md = checkfield(md, 'fieldname', 'basalforcings.upperwater_elevation', 'NaN', 1, 'Inf', 1, 'timeseries', 1, '<', 0)
|
---|
79 | if 'BalancethicknessAnalysis' in analyses:
|
---|
80 | raise Exception('not implemented yet!')
|
---|
81 | md = checkfield(md, 'fieldname', 'basalforcings.groundedice_melting_rate', 'NaN', 1, 'Inf', 1, 'size', [md.mesh.numberofvertices])
|
---|
82 | md = checkfield(md, 'fieldname', 'basalforcings.deepwater_melting_rate', '>=', 0, 'numel', 1)
|
---|
83 | md = checkfield(md, 'fieldname', 'basalforcings.deepwater_elevation', '<', 'basalforcings.upperwater_elevation', 'numel', 1)
|
---|
84 | md = checkfield(md, 'fieldname', 'basalforcings.upperwater_elevation', '<=', 0, 'numel', 1)
|
---|
85 | if 'ThermalAnalysis' in analyses and not solution == 'TransientSolution' and not md.transient.isthermal:
|
---|
86 | raise Exception('not implemented yet!')
|
---|
87 | md = checkfield(md, 'fieldname', 'basalforcings.groundedice_melting_rate', 'NaN', 1, 'Inf', 1, 'timeseries', 1)
|
---|
88 | md = checkfield(md, 'fieldname', 'basalforcings.deepwater_melting_rate', '>=', 0, 'numel', 1)
|
---|
89 | md = checkfield(md, 'fieldname', 'basalforcings.deepwater_elevation', '<', 'basalforcings.upperwater_elevation', 'numel', 1)
|
---|
90 | md = checkfield(md, 'fieldname', 'basalforcings.upperwater_elevation', '<', 0, 'numel', 1)
|
---|
91 | md = checkfield(md, 'fieldname', 'basalforcings.geothermalflux', 'NaN', 1, 'Inf', 1, 'timeseries', 1, '>=', 0)
|
---|
92 | return md
|
---|
93 | # }}}
|
---|
94 |
|
---|
95 | def marshall(self, prefix, md, fid): #{{{
|
---|
96 | yts = md.constants.yts
|
---|
97 |
|
---|
98 | floatingice_melting_rate = np.zeros((md.mesh.numberofvertices, ))
|
---|
99 | pos = np.where(md.geometry.base <= md.basalforcings.deepwater_elevation)[0]
|
---|
100 | floatingice_melting_rate[pos] = md.basalforcings.deepwater_melting_rate[pos]
|
---|
101 | pos = np.where(np.logical_and.reduce(md.geometry.base > md.basalforcings.deepwater_elevation, md.geometry.base < md.basalforcings.upperwater_elevation))
|
---|
102 | floatingice_melting_rate[pos] = md.basalforcings.deepwater_melting_rate[pos] * (md.geometry.base[pos] - md.basalforcings.upperwater_elevation[pos]) / (md.basalforcings.deepwater_elevation[pos] - md.basalforcings.upperwater_elevation[pos])
|
---|
103 | WriteData(fid, prefix, 'name', 'md.basalforcings.model', 'data', 6, 'format', 'Integer')
|
---|
104 | WriteData(fid, prefix, 'data', floatingice_melting_rate, 'format', 'DoubleMat', 'name', 'md.basalforcings.floatingice_melting_rate', 'mattype', 1, 'scale', 1. / yts, 'timeserieslength', md.mesh.numberofvertices + 1, 'yts', md.constants.yts)
|
---|
105 | WriteData(fid, prefix, 'object', self, 'fieldname', 'groundedice_melting_rate', 'format', 'DoubleMat', 'name', 'md.basalforcings.groundedice_melting_rate', 'mattype', 1, 'scale', 1. / yts, 'timeserieslength', md.mesh.numberofvertices + 1,'yts', md.constants.yts)
|
---|
106 | WriteData(fid, prefix,'object', self, 'fieldname', 'geothermalflux', 'name', 'md.basalforcings.geothermalflux', 'format', 'DoubleMat', 'mattype', 1,'timeserieslength', md.mesh.numberofvertices + 1, 'yts', md.constants.yts)
|
---|
107 | WriteData(fid, prefix, 'object', self, 'fieldname', 'deepwater_melting_rate', 'format', 'DoubleMat', 'name', 'md.basalforcings.deepwater_melting_rate', 'scale', 1. / yts, 'mattype', 1)
|
---|
108 | WriteData(fid, prefix, 'object', self, 'fieldname', 'deepwater_elevation', 'format', 'DoubleMat', 'name', 'md.basalforcings.deepwater_elevation', 'mattype', 1)
|
---|
109 | WriteData(fid, prefix, 'object', self, 'fieldname', 'upperwater_elevation', 'format', 'DoubleMat', 'name', 'md.basalforcings.upperwater_elevation', 'mattype', 1)
|
---|
110 | #}}}
|
---|