Quickstart Guide
======================================

`mom6_tools` library can be utilized via its Python API, i.e., directly within Python
scripts or within Jupyter notebooks. In this quickstart guide, we describe how
the tool can be utilized within a Jupyter Notebook, but the majority of
these instructions apply to Python scripts as well.

Step 1: Import modules
----------------------------------------------

The first step is to import the ``Grid`` and ``Topo`` classes of the 
`mom6_bathy` package. The ``Grid`` class represents
horizontal MOM6 grids, and is to be instantiated with the desired grid
configuration and resolution. After creating a grid instance, a ``Topo`` class
instance is to be created to generate an associated bathymetry.

.. code-block:: python

    from mom6_bathy.grid import Grid
    from mom6_bathy.topo import Topo

Step 2: Create the horizontal grid 
-------------------------------------------

After having imported the modules, we can now create a horizontal grid.
An example Grid instantiation:

.. code-block:: python

  grid = Grid(
      nx         = 180,         # Number of grid points in x direction
      ny         = 80,          # Number of grid points in y direction
      lenx       = 360.0,       # grid length in x direction, e.g., 360.0 (degrees)
      leny       = 160,         # grid length in y direction
      cyclic_x   = True,        # reentrant, spherical domain
      ystart     = -80.0        # start/end 10 degrees above/below poles to avoid singularity 
  )

In the above example, the ``Grid`` object, named ``grid``, is constructed by
specifying the required arguments ``nx``, ``ny``, ``config``, ``axis_units``,  ``lenx``,
and ``leny``, in addition to the optional argument ``ystart``. The full list of 
``Grid`` arguments and their descriptions may be printed by running
``Grid?`` statement on a notebook cell:


.. code-block::

    Grid?

    ...

    Parameters
    ----------
    nx : int
        Number of grid points in x direction
    ny : int
        Number of grid points in y direction
    lenx : float
        grid length in x direction, e.g., 360.0 (degrees)
    leny : float
        grid length in y direction, e.g., 160.0 (degrees)
    srefine : int, optional
        refinement factor for the supergrid. 2 by default
    xstart : float, optional
        starting x coordinate. 0.0 by default.
    ystart : float, optional
        starting y coordinate. -0.5*leny by default.
    cyclic_x : bool, optional
        flag to make the grid cyclic in x direction. False by default.
    tripolar_n : bool, optional
        flag to make the grid tripolar. False by default.
    displace_pole : bool, optional
        flag to make the grid displaced polar. False by default.

Note that tripolar and displaced pole grids cannot yet be created from scratch,
but existing tripolar and displaced pole grids can be modified via mom6_bathy.

*Avoiding singularity points*
*****************************

To avoid singularity points within the ocean grid:
  * The grid poles (which may be different than the true poles) must be left out of the grid,
    by making sure that the extent of the grid in the y-direction do not cover the poles, 
    e.g., by setting ``ystart`` to -80.0 degrees
    and ``leny`` to 160.0 degrees.
  * Alternatively, one or two singularities (typically, in the northern hemisphere) may be 
    displaced into land masses if ``displace_pole`` or ``tripolar_n`` options are to be used.
    The other singularity (typically, in the southern hemisphere) would still need to be
    left out the geographic extent of the grid.

If a singularity (a pole) is present within the ocean grid, a land component (active or data) must be added
to the pose of hiding the singularity points of spherical ocean grids within the CESM framework.

*Grid Metrics and Attributes*
*****************************

When a ``Grid`` instance gets created, several grid metrics and attributes
on all staggerings are automatically computed and populated. These metrics and attributes
are accessible via the accessor operator (``.``). For example, to access "the array
of t-grid longitutes" of ``grid``:
    
.. code-block:: python

    grid.tlon

The full list of grid metrics and attributes:

* ``tlon``: array of t-grid longitudes
* ``tlat``: array of t-grid latitudes
* ``ulon``: array of u-grid longitudes
* ``ulat``: array of u-grid latitudes
* ``vlon``: array of v-grid longitudes
* ``vlat``: array of v-grid latitudes
* ``qlon``: array of corner longitudes
* ``qlat``: array of corner latitudes
* ``dxt``: x-distance between U points, centered at t
* ``dyt``: y-distance between V points, centered at t
* ``dxCv``: x-distance between q points, centered at v
* ``dyCu``: y-distance between q points, centered at u
* ``dxCu``: x-distance between y points, centered at u
* ``dyCv``: y-distance between t points, centered at v
* ``angle``: angle T-grid makes with latitude line
* ``tarea``: T-cell area


*Supergrid*
*****************************
In addition to above grid metrics and attributes, the ``Grid`` class incorporates an
underlying :term:`supergrid` instance associated the grid instance, which is again
accessible via the (``.``) operator:

.. code-block:: python

    grid.supergrid

Any user changes to coordinates, e.g., increasing the equatorial resolution,
must be applied to the supergrid using the ``update_supergrid`` method. This is
because the supergrid is the underlying refined grid that is used to determine the
the four staggered grids (T,U,V,Q) that forms the actual computational grid.
Users can modify the supergrid by providing a new x and y coordinate arrays, e.g.,
as follows:

.. code-block:: python

  grid.update_supergrid(xdat, ydat)

where ``xdat`` and ``ydat`` are user-defined 2-dimensional numpy arrays containing
the new x and y coordinates of the supergrid. Running the ``update_supergrid``
method of a ``Grid`` instance automatically updates all other grid metrics listed
above.

Note: the supergrid implementation in `mom6_bathy` relies on `MIDAS <https://github.com/mjharriso/MIDAS>`_,
a python library developed by M. Harrison (GFDL).

Step 3: Create Bathymetry
----------------------------------------------

After having generated the horizontal grid, we can now create an associated bathymetry
using the ``Topo`` class of the `mom6_bathy` tool. We instantiate a bathymetry 
object as follows:

.. code-block:: python

    topo = Topo(grid, min_depth=10.0)

The first argument (``grid``) of ``Topo`` constructor is the horizontal grid instance for which
the bathymetry is to be created, while the second argument (``min_depth``) is the minimum ocean depth.
Any column in the ocean grid with a depth shallower than ``min_depth``  is masked out of the ocean
domain. The minimum depth attribute of a bathymetry instance may be changed afterwards using the
assignment operator. For example:

.. code-block:: python

    topo.min_depth = 5.0

*Predefined Bathymetry Configurations*
**************************************
The ``Topo`` class provides three predefined bathymetry configurations, which are also
available in MOM6 as idealized configurations. (See `TOPO_CONFIG` parameter in MOM_input)

  * `flat`: flat bottom set to MAXIMUM_DEPTH. Example:
  * `bowl`: an analytically specified bowl-shaped basin ranging between MAXIMUM_DEPTH and MINIMUM_DEPTH.
  * `spoon`: a similar shape to 'bowl', but with an vertical wall at the southern face.

Examples:

.. code-block:: python

    # flat bottom
    topo.set_flat(D=500.0)

    # bowl
    topo.set_bowl(500.0, 50.0, expdecay=1e7)

    # spoon
    topo.set_spoon(500.0, 50.0, expdecay=1e7)
    
The first and the second arguments of ``set_bowl`` and ``set_spoon`` methods are maximum depth
and minimum depth, respectively.

Check out the following notebook to see examples of above predefined bathymetry options: `1_spherical_grid.ipynb 
<https://github.com/NCAR/mom6_bathy/blob/master/notebooks/1_spherical_grid.ipynb>`_

*Custom Bathymetry*
*************************************
In addition to the above predefined configurations, users may provide their own depth arrays. For
example:
  
.. code-block:: python

    import numpy as np

    # define a custom depth
    i = grid.tlat.nx.data                # array of x-indices
    j = grid.tlat.ny.data[:,np.newaxis]  # array of y-indices 
    custom_depth = 400.0 + 80.0 * np.sin(i*np.pi/6.) * np.cos(j*np.pi/6.)

    # update the bathymetry:
    topo.depth = custom_depth


*Adding ridges*
*************************************
Simpler model bathymetry configurations typically include ridges to represent straits and
continents in an idealized manner. The ``Topo`` class provides ``apply_ridge`` method
to add ridges to the bathymetry. Example usage:

.. code-block:: python

  topo.apply_ridge(height=200, width=8, lon=240, ilat=(10,80) )

Example notebook: `3_custom_bathy.ipynb 
<https://github.com/NCAR/mom6_bathy/blob/master/notebooks/3_custom_bathy.ipynb>`_


Step 4: Write Model Input Files
----------------------------------------------

The final step of `mom6_bathy` workflow is to write out the netcdf files containing grid
and bathymetry data. These files are to be read in by CESM and MOM6 during runtime.

*Supergrid File*
****************

The ``write_supergrid`` method of a ``Grid`` instance writes out the MOM6 supergrid file
in netcdf format. The ``GRID_FILE`` parameter in ``MOM_input`` file can then be set to
the path of the supergrid file written by the ``Grid`` instance.

.. code-block:: python

  grid.write_supergrid("my_ocean_hgrid.nc")

The supergrid file is the only input file that is written by the `Grid` class. All other
input files require either topography (depth) or mask information. Hence, they are to be
written by the `Topo` class.

*Topography (Bathymetry) File*
******************************

The ``write_topo`` method of the ``Topo`` class writes out the MOM6 bathymetry file in netcdf format.
``TOPO_FILE`` parameter in ``MOM_input`` file can then be set to the path of the topography file
written by the ``Topo`` instance.

.. code-block:: python

  topo.write_topo("my_ocean_topog.nc")

*CICE grid file*
******************************

If the model is to be run with the CICE component, the ``write_cice_grid`` method of the 
``Topo`` class writes out the CICE grid file in netcdf format. The relevant CICE namelist
parameters can then be updated to read in the CICE grid file written by the ``Topo`` instance.

.. code-block:: python

  topo.write_cice_grid("my_cice_grid.nc")

*ESMF Mesh file*
*******************************

In addition to the MOM6 supergrid file, MOM6 topography file and CICE grid file, an
ESMF mesh file is required when running CESM. The ESMF mesh file is used
by the NUOPC coupler to acquire grid and mask information. The ``write_esmf_mesh`` method
of the ``Topo`` class writes out the ESMF mesh file in netcdf format.

.. code-block:: python

  topo.write_esmf_mesh("my_esmf_mesh.nc")


Further steps
----------------------------------------------

The remaining steps of configuring the model, which include specifying initial conditions,
forcings, and runtime parameters, are beyond the scope of the `mom6_bathy` tool. Note that a 
complementary tool called `visualCaseGen`, which includes `mom6_bathy` as a submodule, can be used
to generate a complete model configuration. `visualCaseGen` provides a graphical user interface
to set up the model grid, bathymetry, initial conditions, forcing, and runtime parameters for MOM6
and other CESM components. Hence, new users are encouraged to use `visualCaseGen` for a complete
model configuration. See: `visualCaseGen <https://github.com/ESMCI/visualCaseGen>`_