[25688] | 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): #{{{
|
---|
[25762] | 18 | nargs = len(args)
|
---|
[25688] | 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])
|
---|
[25762] | 82 | md = checkfield(md, 'fieldname', 'basalforcings.deepwater_melting_rate', '>=', 0, 'numel', 1)
|
---|
[25688] | 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 | #}}}
|
---|