How-to: nml
module¶
The nml
module of glm-py provides tools for reading and writing NML
configuration files. The module is divided into two sub-modules that cater to
different use cases and levels of control:
nml.glm_nml
: High-level tools for writing GLM NML files only. Provides classes to construct each configuration block with parameter documentation and error checking. Applicable for most use cases.nml.nml
: Low-level tools for reading and writing any NML file, i.e., for GLM or AED. To write files, users will need to create a nested dictionary of parameters. Does not enforce any requirements on the supplied parameters. Applicable to users who need to do something custom or go beyond what is currently supported with the high-level tools.
The nml.glm_nml
sub-module¶
Writing a GLM NML file¶
The nml.glm_nml
sub-module provides tools for writing GLM NML files only. To
write a file, start by importing the glm_nml
sub-module from the nml
module
of glmpy
:
Classes for NML configuration blocks¶
Each configuration block in a GLM NML file (e.g., &glm_setup
,
&morphometry
, and &init_profiles
) has a respective class in the glm_nml
sub-module, e.g., glm_nml.SetupBlock
, glm_nml.MorphometryBlock
, and
glm_nml.InitProfilesBlock
. These classes are used to construct dictionaries
of model parameters which can be combined in the glm_nml.GLMNML
class to
write the NML file. For example, the &glm_setup
parameters can be constructed
with the glm_nml.SetupBlock
class as follows:
my_setup = glm_nml.SetupBlock(
sim_name='GLMSimulation',
max_layers=500,
min_layer_vol=0.5,
min_layer_thick=0.15,
max_layer_thick=0.5,
density_model=1,
non_avg=True
)
Alternatively, the class attributes can be set/updated with the
set_attrs()
method:
my_setup = glm_nml.SetupBlock()
setup_params = {
'sim_name': 'GLMSimulation',
'max_layers': 500,
'min_layer_vol': 0.5,
'min_layer_thick': 0.15,
'max_layer_thick': 0.5,
'density_model': 1,
'non_avg': True
}
my_setup.set_attrs(setup_params)
Refer to the API Reference for detailed information about the attributes (i.e., the GLM model parameters) for each block.
Returning a dictionary consolidated model parameters¶
When you call the get_params()
method of the SetupBlock
class (or any
configuration block class), a dictionary of consolidated model parameters will
be returned.
{'sim_name': 'GLMSimulation', 'max_layers': 500, 'min_layer_vol': 0.5, 'min_layer_thick': 0.15, 'max_layer_thick': 0.5, 'density_model': 1, 'non_avg': True}
Note
get_params()
has an optional check_params
parameter for the purpose of
validating your model parameters and raising errors if they are not
compliant with what GLM expects.
As of glm-py 0.2.0
, check_params
is accessible but no error checking
is currently implemented.
Writing the NML file¶
At a minimum, the GLM NML file requires model parameters set for the following blocks:
&glm_setup
with theSetupBlock
class&morphometry
with theMorphometryBlock
class&time
with theTimeBlock
class&init_profiles
with theInitProfilesBlock
class
The parameter dictionaries returned by these classes can then be combined into
a NML file with the GLMNML
class:
my_nml = glm_nml.GLMNML(
setup=my_setup.get_params(),
morphometry=my_morphometry.get_params(),
time=my_time.get_params(),
init_profiles=my_init_profiles.get_params()
)
Note
The GLMNML
class also provides an optional check_params
attribute to
validate model parameters that have dependencies between blocks. As of
0.2.0
this error checking is not currently implemented
To write the NML file to disk, call the write_nml()
method and provide the
output file path:
The nml.nml
sub-module¶
The nml.nml
sub-module provides tools for reading and writing any NML file.
It provides greater control on how to read/write each parameter but comes at
the cost of some of the more user-friendly features of the nml.glm_nml
sub-module, e.g., model parameter documentation and error checking.
Writing a NML file with NMLWriter
¶
In advanced use-cases, you may wish to write a NML file that goes beyond
what is possible with the nml.glm_nml
sub-module, e.g., writing custom
configuration blocks or extending existing ones. In these situations the
NMLWriter
class in the nml.nml
sub-module provides near complete control
over the contents of a NML file.
Writing a custom block¶
Consider the example NML file below:
&custom_block1
param1 = 123
param2 = 1.23
param3 = 'foo'
param4 = .true.
/
&custom_block2
param1 = 1,2,3
param2 = 'foo','bar','baz'
param3 = .true.,.false.,.true.,
.true.,.false.,.true.,
.true.,.false.,.true.
/
It's not possible to produce an output like this with tools in the
nml.glm_nml
sub-module - you're restricted to the standard configuration
blocks and parameters.
To write this file with NMLWriter
, first import the nml.nml
sub-module:
Next, create a nested dictionary where they keys are the block names and the values are a dictionary of parameters:
custom_nml = {
"custom_block1": {
"param1": 123,
"param2": 1.23,
"param3": "foo",
"param4": True
},
"custom_block2": {
"param1": [1, 2, 3],
"param2": ["foo", "bar", "baz"],
"param3": [True, False, True]
}
}
This is your NML file in Python syntax. Be mindful of the Python data types you use for your parameter values. They map to the respective syntax of data types in a NML file. Python integers will be converted to NML integers, floats to floats, etc. Notice too the use of Python lists for NML comma-separated lists.
Note
For lists, don't mix and match your Python data types - especially integers and floats.
Now create an instance of NMLWriter
and provide your nested dictionary:
To write the file, simply call the write_nml
method and provide and path for
the NML output:
Setting the parameter converters¶
In even more advanced use-cases, you may wish to explicitly control how
NMLWriter
converts a Python data type to the syntax of a NML data type. By
default, NMLWriter
is initialised with the detect_types
attribute set to
True
. This will trigger NMLWriter
to automatically convert parameters to
the most appropriate NML data types (ints to ints, floats to floats, etc).
By setting detect_types
to False
you can override the automatic conversion
and provide your own syntax conversion functions.
Consider again the example from above:
custom_nml = {
"custom_block1": {
"param1": 123,
"param2": 1.23,
"param3": "foo",
"param4": True
},
"custom_block2": {
"param1": [1, 2, 3],
"param2": ["foo", "bar", "baz"],
"param3": [True, False, True]
}
}
When NMLWriter
is initialised with custom_nml
and detect_types
set to
True
, it will construct the following dictionary of syntax conversion
methods:
{
"custom_block1": {
"param1": None,
"param2": None,
"param3": nml.NMLWriter.write_nml_str,
"param4": nml.NMLWriter.write_nml_bool
},
"custom_block2": {
"param1": nml.NMLWriter.write_nml_list,
"param2": lambda x: nml.NMLWriter.write_nml_list(
x, nml.NMLWriter.write_nml_str
),
"param3": lambda x: nml.NMLWriter.write_nml_list(
x, nml.NMLWriter.write_nml_bool
)
}
}
write_nml_str
, write_nml_bool
, and write_nml_list
? These are
static methods that you can call independently of initialising NMLWriter
to
convert Python syntax to NML syntax:
Note
No static methods are required for converting integers and floats.
When writing lists with string or boolean elements, lambda functions can used create new combinations of these methods.
By creating a nested dictionary in the same fashion as shown above, you can
tell NMLWriter
how to write each parameter. Simply initialise the class with
detect_types
set to False
and then call the set_converters
method
with your nested dictionary of conversion functions:
my_nml = nml.NMLWriter(custom_nml, detect_types=False)
my_nml.set_converters(my_conversion_methods_dict)
my_nml.write_nml(nml_file="custom_nml.nml")
To see this in action, let's consider an example where we wish to write the
complex number 0.5 + 1.2j
to a NML file. The desired output would be:
Here, the first component within the parentheses is a real number and the second component is an imaginary number. We can easily write a Python function that returns a string of the NML complex number:
import cmath
def write_nml_complex_num(py_complex_num: complex) -> str:
return f"({py_complex_num.real}, {py_complex_num.imag})"
z = 0.5 + 1.2j
print(write_nml_complex_num(z))
Now we can write the NML file by creating a dictionary with the parameter value and another with our custom syntax conversion function:
custom_nml = {
"custom_block": {
"param1": 0.5 + 1.2j
}
}
custom_converters = {
"custom_block": {
"param1": write_nml_complex_num
}
}
my_nml = nml.NMLWriter(nml_dict=custom_nml, detect_types=False)
my_nml.set_converters(custom_converters)
my_nml.write_nml(nml_file="custom_nml.nml")
Reading a NML file with NMLReader
¶
glm-py makes it easy to run simulations from pre-existing NML files with the
NMLReader
class.
Reading blocks¶
Consider the following exert from example_glm_nml.nml
:
!-------------------------------------------------------------------------------
! general model setup
!-------------------------------------------------------------------------------
!
! sim_name [string] title of simulation
! max_layers [integer] maximum number of layers
! min_layer_vol [real] minimum layer volume (m3 * 1000)
! min_layer_thick [real] minimum layer thickness (m)
! max_layer_thick [real] maximum layer thickness (m)
! Kw [real] background light attenuation (m**-1)
! coef_mix_conv [real] mixing efficiency - convective overturn
! coef_wind_stir [real] mixing efficiency - wind stirring
! coef_mix_turb [real] mixing efficiency - unsteady turbulence effects
! coef_mix_shear [real] mixing efficiency - shear production
! coef_mix_KH [real] mixing efficiency - hypolimnetic Kelvin-Helmholtz turbulent billows
! coef_mix_hyp [real] mixing efficiency - hypolimnetic turbulence
! deep_mixing [bool] flag to disable deep-mixing
!-------------------------------------------------------------------------------
&glm_setup
sim_name = 'GLM Simulation'
max_layers = 60
min_layer_vol = 0.0005
min_layer_thick = 0.05
max_layer_thick = 0.1
non_avg = .true.
! mobility_off = .true.
/
&mixing
!-- Mixing Parameters
coef_mix_conv = 0.125
coef_wind_stir = 0.23
coef_mix_shear = 0.00
coef_mix_turb = 0.51
coef_mix_KH = 0.30
coef_mix_hyp = 0.5
deep_mixing = 0
surface_mixing = 1
/
&light
!-- Light Parameters
light_mode = 0
Kw = 3.5
/
With NMLReader
you can convert this to a nested python dictionary that's
ready for use with the rest of tools in the nml
and glm_nml
sub-modules.
To read the file, import the nml
sub-module and initialise an instance of
NMLReader
by providing the path to example_nml.nml
:
Now call the get_nml
method to convert and return the entire NML file as
a nested dictionary:
{'glm_setup': {'sim_name': 'GLM Simulation', 'max_layers': 60, 'min_layer_vol': 0.0005, 'min_layer_thick': 0.05, 'max_layer_thick': 0.1, 'non_avg': True}, 'mixing': {'coef_mix_conv': 0.125, 'coef_wind_stir': 0.23, 'coef_mix_shear': 0.0, 'coef_mix_turb': 0.51, 'coef_mix_KH': 0.3, 'coef_mix_hyp': 0.5, 'deep_mixing': 0, 'surface_mixing': 1}, 'light': {'light_mode': 0, 'Kw': 3.5}}
To return a specific block, call the get_block
method and provide the block
name:
Bug
NMLReader
will return erroneous results if the NML file has the
following:
-
Exclamation marks (
!
) within a string parameters, e.g.,sim_name = 'A very important sim!'
. Exclamation marks are used to declare comments in NML files. -
Comma-separated lists terminating with commas, e.g.,
A = 100, 3600, 5600,
. Remove the final comma:A = 100, 3600, 5600
.
Setting the parameter converters¶
As of glm-py 0.2.0
, NMLReader
lacks a similar implementation of the
automatic syntax conversion that NMLWriter
performs when detect_types
is
set to True
, i.e., it does not go build a dictionary of the appropriate
syntax conversion methods to use from the data types it sees. As a result,
NMLReader
relies on an internal dictionary of syntax conversion methods that
currently only covers GLM configuration blocks. To read AED NML files or
custom GLM NML files, you'll need to provide your own nested dictionary syntax
conversion methods.
Note
Support for automatic syntax conversion in NMLReader
is planned.
Consider the following AED NML file saved as aed.nml
:
Attempting to read this file with NMLReader
will raise the following warning:
UserWarning: Unexpected block 'aed_tracer' in the NML file. If parsing this block is desired, update the conversion methods with `set_converters()`. Provide a dictionary containing the block name as the key and a nested dictionary of parameter conversion methods as the value. For example:
>>> converters = {"aed_tracer": {"param1": NMLReader.read_nml_str}}
As prompted we need to create a nested dictionary that contains the conversion
methods to convert NML data types to Python data types. Just like NMLWriter
,
NMLReader
provides a suite of static methods for reading various string
representations of NML data types:
nml_string = 'GLM'
python_string = nml.NMLReader.read_nml_str(nml_string)
print(python_string)
print(type(python_string))
nml_int = '123'
python_int = nml.NMLReader.read_nml_int(nml_int)
print(python_int)
print(type(python_int))
nml_float = '1.23'
python_float = nml.NMLReader.read_nml_float(nml_float)
print(python_float)
print(type(python_float))
nml_bool = '.true.'
python_bool = nml.NMLReader.read_nml_bool(nml_bool)
print(python_bool)
print(type(python_bool))
nml_list = '1,2,3'
python_list = nml.NMLReader.read_nml_list(
nml_list,
nml.NMLReader.read_nml_int
)
print(python_list)
print(type(python_list))
To read the tracer block, we need to create a nested dictionary that tells
NMLReader
which static methods to use:
nml_conversion_dict = {
"aed_tracer": {
"retention_time": nml.NMLReader.read_nml_bool,
"num_tracers": nml.NMLReader.read_nml_int,
"decay": lambda x: nml.NMLReader.read_nml_list(
x,
nml.NMLReader.read_nml_int
),
"Fsed": lambda x: nml.NMLReader.read_nml_list(
x,
nml.NMLReader.read_nml_int
)
}
}
This can be passed to NMLReader
by calling the set_converters
method.
Calling get_nml
will now return a dictionary of the NML file with Python
data types: