.. _pycart-ex-arrow:

Demo 2: Closer Analysis of Simple Arrow Shape
=============================================

The second example is similar to the first pyCart demo except that four fins
have been added and more details of the input files are explained. To get
started, clone the following repo and enter the folder and run a couple of
commands:

    .. code-block:: console

        $ git clone https://github.com/nasa-ddalle/pycart02-arrow.git
        $ cd pycart02-arrow
        $ ./copy-files.py
        $ cd work/

This will copy all of the files into a newly created ``work/`` folder. Follow
the instructions below by entering that ``work/`` folder; the purpose is that
you can easily delete the ``work/`` folder and restart the tutorial at any
time.

The geometry used for this shape is a capped cylinder with four fins and 9216
faces and seven components.  The surface triangulation, ``arrow.tri``, is
shown below.

    .. figure:: arrow01.png
        :width: 4in
        
        Simple bullet shape triangulation with four fins
        
This example is set up with a larger run matrix in order to demonstrate more of
the features of pyCart.

    .. code-block:: none
    
        $ pycart -c
        Case Config/Run Directory    Status  Iterations  Que CPU Time
        ---- ----------------------- ------- ----------- --- --------
        0    poweroff/m1.25a0.0r0.0  ---     /           .   
        1    poweroff/m1.25a1.0r0.0  ---     /           .   
        2    poweroff/m1.25a1.0r15.0 ---     /           .   
        3    poweroff/m1.25a1.0r30.0 ---     /           .   
        4    poweroff/m1.25a1.0r45.0 ---     /           .   
        5    poweroff/m1.5a1.0r0.0   ---     /           .   
        6    poweroff/m1.5a1.0r15.0  ---     /           .   
        7    poweroff/m1.5a1.0r30.0  ---     /           .   
        8    poweroff/m1.5a1.0r45.0  ---     /           .   
        9    poweroff/m1.75a1.0r0.0  ---     /           .   
        10   poweroff/m1.75a1.0r15.0 ---     /           .   
        11   poweroff/m1.75a1.0r30.0 ---     /           .   
        12   poweroff/m1.75a1.0r45.0 ---     /           .   
        13   poweroff/m2.0a1.0r0.0   ---     /           .   
        14   poweroff/m2.0a1.0r15.0  ---     /           .   
        15   poweroff/m2.0a1.0r30.0  ---     /           .   
        16   poweroff/m2.0a1.0r45.0  ---     /           .   
        17   poweroff/m2.5a1.0r0.0   ---     /           .   
        18   poweroff/m2.5a1.0r15.0  ---     /           .   
        19   poweroff/m2.5a1.0r30.0  ---     /           .   
        20   poweroff/m2.5a1.0r45.0  ---     /           .   
        
        ---=21, 
        
Input Files
-----------
Let's look at the files in this folder.

    * ``pyCart.json``: Master settings for running pyCart, input to ``pycart``
    * ``arrow.tri``: Surface triangulation (ASCII)
    * ``arrow.xml``: Names for components of the surface
    * ``matrix.csv``: Run matrix
    
JSON Settings
^^^^^^^^^^^^^
The ``pyCart.json`` file contains the master settings divided into several
sections, which we will discuss in more detail.  The overall contents of the
file look something like the following, with the ``...`` replaced by more
content.

    .. code-block:: javascript
    
        {
            // Iteration control and command-line inputs
            "RunControl": {
                // Run sequence
                "InputSeq": [0],
                "IterSeq": [200],
                ...
            },
            
            ...
            
            // RunMatrix (i.e. run matrix) description
            "RunMatrix": {
                "Keys": ["Mach", "alpha_t", "phi"],
                "File": "matrix.csv",
                "GroupMesh": true,
                "GroupPrefix": "poweroff"
            }
        }

The first section (actually, the order does not matter, but it's the first
section in the file provided) is the ``"RunControl"`` section, which has
settings for the overall run procedure (such as number of iterations, whether
or not to submit the job to a queue, etc.) and command-line inputs to the
various Cart3D programs.

    .. code-block:: javascript
    
        "RunControl": {
            // Run sequence
            "PhaseSequece": [0],
            "PhaseIters": [200],
            // Verbosity
            "Verbose": true,
            // System configuration
            "nProc": 4,
            // Options for ``flowCart``
            "flowCart": {
                "it_fc": 200,
                "mpi_fc": 0,
                "use_aero_csh": 0,
                "cfl": 1.1,
                "mg_fc": 3,
                "y_is_spanwise": true
            },
            // Defines the flow domain automatically
            "autoInputs": {"r": 8},
            // Volume mesh options
            "cubes": {
                "maxR": 10,
                "pre": "preSpec.c3d.cntl",
                "cubes_a": 10,
                "cubes_b": 2,
                "reorder": true
            }
        },
        
The ``"flowCart"`` section contains command-line inputs for running
``flowCart``, which is the main flow solver of Cart3D, or ``mpix_flowCart``,
which is the MPI version of the same. Many of the variable names, such as
*it_fc*, are copied from Cart3D's template ``aero.csh`` scripts or
command-line inputs to Cart3D's ``flowCart``. The three main options (which are
required for any pyCart project) are *PhaseSequence*, *PhaseIters*, and
*it_fc*.

    +-----------------+-------------------------------------------------------+
    | Variable        | Description                                           |
    +=================+=======================================================+
    | *it_fc*         | Number of iterations for each call to ``flowCart``,   |
    |                 | short for ``iterations_flowCart``; command-line input |
    |                 | is ``flowCart -N $it_fc``                             |
    +-----------------+-------------------------------------------------------+
    | *PhaseSequence* | Input sequence, tells pyCart to run phase 0; in more  |
    |                 | complex projects, this will be a list like ``[0,1,3]``|
    +-----------------+-------------------------------------------------------+
    | *PhaseIters*    | Min iterations for each phase; this tells pyCart to   |
    |                 | continue calling ``flowCart`` until 200 iterations    |
    |                 | have been run.  If this was ``400``, pyCart would     |
    |                 | automatically run ``flowCart`` twice using the first  |
    |                 | run's results as inputs to the second                 |
    +-----------------+-------------------------------------------------------+
    
The *Verbose* option is relatively self-explanatory in that more information is
printed to either the terminal or the PBS output file.  In particular, each
command that is issued to the top-level terminal also prints the name of the
directory in which it is run and the name of the file storing its STDOUT
output.
    
For a simple case, these parameters seem unnecessarily confusing. Why not just
tell ``flowCart`` how many iterations to run and be done with it? For one
thing, *IterSeq* specifies a required number of iterations whereas *it_fc* just
suggests to ``flowCart`` or ``mpix_flowCart`` how many iterations to run. If
``flowCart`` exits early due to some kind of failure, this convention means
that pyCart will clearly alert us.

Secondly, some applications require more sophisticated approach. A common
example is a hypersonic case that needs to be run in first-order mode for a few
iterations first. It might have something like ``"PhaseIters": [0, 400]`` and
``"PhaseSequence": [0, 1]``. This tells pyCart to run input set ``0`` until it
has run at least ``0`` iterations and then phase ``1`` until it has run at
least ``400`` iterations.

The remaining inputs are quite a bit simpler. For example *nProc* sets the
total number of cores or threads to use. The next section allows pyCart to use
the Cart3D binary ``autoInputs`` to create the flow domain and basic volume
mesh parameters with the command ``autoInputs -r 8``, which sets the farfield
boundary at roughly 8 times the size of your surface triangulation.

Running ``autoInputs`` creates files ``input.c3d`` and ``preSpec.c3d.cntl``,
which are given as inputs to the volume generator ``cubes``.

    .. code-block:: javascript
    
        "Mesh": {
            // Surface triangulation
            "TriFile": "arrow.tri"
        },
        
The *Mesh* section controls inputs to the Cart3D commands that produce the
volume mesh. The *TriFile* setting is relatively obvious and points to the name
of the surface triangulation.

    .. code-block:: javascript
    
        "Config": {
            // Defer to a file for most things.
            "File": "arrow.xml",
            // Declare forces and moments
            "Force": ["cap", "body", "fins", "bullet_no_base", "bullet_total"],
            "RefPoint": {"bullet_no_base": [0.0, 0.0, 0.0]}
            // Reference quantities
            "RefArea": 3.14159,
            "RefLength": 1.0,
        },
        
The *Config* section gives instructions about which components to track, what
moment reference points to use, and similar definitions. The XML file allows
Cart3D and pyCart to refer to define groups of components and refer to
components by name instead of memorizing their numbers. The *Force* option
specifies a list of components on which ``flowCart`` should track the force at
each iteration. This creates files ``cap.dat``, ``body.dat``, ``fins.dat``,
etc. Then *RefPoint* specifies the list of components for which to also track
the moments, and the moment reference point to use for each such component. In
this case, the moments will be reported alongside the forces in
``bullet_no_base.dat``.

The *RefArea* and *RefLength* parameters are used here to specify global
reference values, but it is possible to use different reference lengths or
areas for different components in the same run.

    .. code-block:: javascript
    
        "RunMatrix": {
            "Keys": ["Mach", "alpha_t", "phi"],
            "File": "matrix.csv",
            "GroupMesh": true,
            "GroupPrefix": "poweroff"
        }

The final section (actually, the order is irrelevant, but it's the last section
in this file) describes the run matrix, i.e. trajectory. The *Keys* parameter
lists the names of variables that will change in the run matrix, i.e. the
independent variables. In this case, we are using Mach number, total angle of
attack, and velocity roll angle. There is a set of predefined trajectory keys,
and all three of these examples are in that set, but later examples will show
how to define customized trajectory keys in this section.

The *File* parameter points to a file in which the cases to run are listed, and
*GroupMesh* specifies whether or not each case can use the same mesh.  Setting
it to ``true`` means that ``cubes`` is only run once for the matrix (more
accurately, once for each group, but this example has only one group).  The
*GroupPrefix* gives a name for the folder in which to put all the cases, which
explains why a typical case is named ``poweroff/m1.50a2.00r0.00``, for example.

There are two more sections in the ``pyCart.json``, which describe various
products.

Triangulation File: ``arrow.tri``
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
The surface geometry is defined in an ASCII file in a straightforward Cart3D
format.  A summary of the contents is shown below.

    .. code-block:: none
    
        4610  9216
        +4.81527351e-03 +9.80171422e-02 +0.00000000e+00
        +1.92147203e-02 +1.95090326e-01 +0.00000000e+00
        ...
        +6.37716534e+00 +1.58689240e-08 +1.06574801e+00
        385 386 16
        386 387 17
        ...
        2565 4257 2530
        1
        1
        ...
        11

The first line is a summary of the contents of the file.  It states that there
are ``4610`` nodes, i.e. three-dimensional points in space, and ``9216``
triangles.  What follows is 4610 lines with three floating point numbers per
line.  Next is 9216 lines in which each line defines one triangle.  For example,
the first triangle connects node ``385`` to node ``386`` to node ``16``.  After
9216 such lines, there are 9216 more lines with a single integer on each line
that defines the component ID of each triangle.  Thus triangle 1 is part of
component 1, triangle 2 is part of component 1, and the last triangle is part of
component 11.

Component Names: ``arrow.xml``
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
Cart3D uses an optional XML file that associates names with each component.  It
uses a standard XML format with component IDs (the numbers at the end of the
``.tri`` file discussed above) with a ``Face Label`` value inside a
``<Data>`` tag.  It also allows for the definition of a "container" component
that is the combination of several other components.  This makes it possible to 
track ``fin1`` separately while also tracking all the ``fins`` as a group.  The
contents of the file are shown below.

    .. code-block:: xml
    
        <?xml version="1.0" encoding="ISO-8859-1"?>

        <Configuration Name="bullet sample" Source="bullet.tri">
        
         <!-- Containers -->
          <Component Name="bullet_no_base" Type="container" Parent="bullet_total">
          </Component>
          <Component Name="fins" Type="container" Parent="bullet_no_base">
          </Component>
         
          <Component Name="bullet_total"   Type="container">
          </Component>
         <!-- Containers -->
        
         <!-- body -->
          <Component Name="cap" Type="tri">
           <Data> Face Label=1 </Data>
          </Component>
         
          <Component Name="body" Type="tri">
           <Data> Face Label=2 </Data>
          </Component>
         
          <Component Name="base" Parent="bullet_total" Type="tri">
           <Data> Face Label=3 </Data>
          </Component>
         <!-- body -->
         
         <!-- fins -->
          <Component Name="fin1" Parent="fins" Type="tri">
           <Data> Face Label=11 </Data>
          </Component>
          
          <Component Name="fin2" Parent="fins" Type="tri">
           <Data> Face Label=12 </Data>
          </Component>
          
          <Component Name="fin3" Parent="fins" Type="tri">
           <Data> Face Label=13 </Data>
          </Component>
          
          <Component Name="fin4" Parent="fins" Type="tri">
           <Data> Face Label=14 </Data>
          </Component>
         <!-- fins -->
        
        </Configuration>

Run Matrix File: ``matrix.csv``
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
The conditions at which Cart3D are read from this file, which is a simple list
of conditions.

    .. code-block:: none
    
        # Mach, alpha, phi
        1.25,   0.00,   0.0
        1.25,   1.00,   0.0
        1.25,   1.00,   15.0
        ...
        2.50,   1.00,   45.0

The comment line at the top is not read by pyCart but is placed there for
readability.  Further, the commas are not required; pyCart and other CAPE
modules read trajectory files in a pretty general way.

Run Directives
--------------
Let's run one case, but not the first case.  We can do this by using the
``pycart -I`` command to pick out a specific index or a range of indices.

    .. code-block:: none
    
        $ pycart -I 12
        Case Config/Run Directory    Status  Iterations  Que CPU Time
        ---- ----------------------- ------- ----------- --- --------
        0    poweroff/m1.75a1.0r15.0 ---     /           .   
          Group name: 'poweroff' (index 0)
          Preparing surface triangulation...
          Reading tri file(s) from root directory.
             Writing triangulation: 'Components.i.tri'
         > autoInputs -r 8 -t Components.i.tri -maxR 10
             (PWD = 'pycart02-arrow/poweroff')
             (STDOUT = 'autoInputs.out')
         > cubes -pre preSpec.c3d.cntl -maxR 10 -reorder -a 10 -b 2
             (PWD = 'pycart02-arrow/poweroff')
             (STDOUT = 'cubes.out')
         > mgPrep -n 3
             (PWD = '/u/wk/ddalle/usr/pycart/examples/pycart/02_arrow/poweroff')
             (STDOUT = 'mgPrep.out')
        Using template for 'input.cntl' file
             Starting case 'poweroff/m1.75a1.0r15.0'.
         > flowCart -his -clic -N 200 -y_is_spanwise -limiter 2 -T -cfl 1.1 -mg 3 -binaryIO -tm 0
             (PWD = 'poweroff/m1.75a1.0r15.0')
             (STDOUT = 'flowCart.out')
        
        Submitted or ran 1 job(s).
        
        ---=1, 

We can check the status of all the cases at Mach 1.75 using the following.  Like
the previous example, the CPU time is below 0.1 hours.

    .. code-block:: none
    
        $ pycart -I 11:15 -c
        Case Config/Run Directory    Status  Iterations  Que CPU Time
        ---- ----------------------- ------- ----------- --- --------
        0    poweroff/m1.75a1.0r0.0  ---     /           .   
        1    poweroff/m1.75a1.0r15.0 DONE    200/200     .   0.0
        2    poweroff/m1.75a1.0r30.0 ---     /           .   
        3    poweroff/m1.75a1.0r45.0 ---     /           .   
        
        ---=3, DONE=1, 

We can use a more direct method to select cases with a certain Mach number using
a constraint.  Let's run the remaining Mach 1.75 cases using that capability.

    .. code-block:: none
    
        $ pycart --cons "Mach==1.75, alpha_t==1.0"
        Case Config/Run Directory    Status  Iterations  Que CPU Time
        ---- ----------------------- ------- ----------- --- --------
        0    poweroff/m1.75a1.0r0.0  ---     /           .   
        Using template for 'input.cntl' file
             Starting case 'poweroff/m1.75a1.0r0.0'.
         > flowCart -his -clic -N 200 -y_is_spanwise -limiter 2 -T -cfl 1.1 -mg 3 -binaryIO -tm 0
             (PWD = 'poweroff/m1.75a1.0r0.0')
             (STDOUT = 'flowCart.out')
        1    poweroff/m1.75a1.0r15.0 DONE    200/200     .   0.0
        2    poweroff/m1.75a1.0r30.0 ---     /           .   
        Using template for 'input.cntl' file
             Starting case 'poweroff/m1.75a1.0r30.0'.
         > flowCart -his -clic -N 200 -y_is_spanwise -limiter 2 -T -cfl 1.1 -mg 3 -binaryIO -tm 0
             (PWD = 'poweroff/m1.75a1.0r30.0')
             (STDOUT = 'flowCart.out')
        3    poweroff/m1.75a1.0r45.0 ---     /           .   
        Using template for 'input.cntl' file
             Starting case 'poweroff/m1.75a1.0r45.0'.
         > flowCart -his -clic -N 200 -y_is_spanwise -limiter 2 -T -cfl 1.1 -mg 3 -binaryIO -tm 0
             (PWD = 'poweroff/m1.75a1.0r45.0')
             (STDOUT = 'flowCart.out')
        
        Submitted or ran 3 job(s).
        
        ---=3, DONE=1,
        
It is also possible to select these cases using ``pycart --filter m1.75a1``,
``pycart --glob "*m1.75a1*"``, or ``pycart --re "m1\.75a1"``. The last of these
checks for a regular expression, which allows more complex filters to be
applied.
        
Run Folders and Output Files
----------------------------
Let's take a look at the files that pyCart created.  First, let's look at the 
files that define the mesh in the ``poweroff/`` folder.

    .. code-block:: none
    
        $ cd poweroff/
        $ ls
        autoInputs.out    input.c3d       m1.75a1.0r45.0  mgPrep.out
        Components.i.tri  m1.75a1.0r0.0   Mesh.c3d.Info   preSpec.c3d.cntl
        Config.xml        m1.75a1.0r15.0  Mesh.mg.c3d   
        cubes.out         m1.75a1.0r30.0  Mesh.R.c3d   

The ``.out`` files save STDIO printouts from the mesh-generation commands.
The ``Mesh.mg.c3d`` is the actual mesh file, including multigrid levels
(i.e., coarsened grids). Our surface triangulation, ``arrow.tri`` is copied
to ``Components.i.tri`` in this folder; and the configuration file
``arrow.xml`` is copied to ``Config.xml``. The single mesh without
multigrid levels is ``Mesh.R.c3d``, and the remaining files are created by
``autoInputs``.

The contents of ``input.c3d`` set the minimum and maximum *x*, *y*, and *z*
coordinates for the domain on which Cart3D is solved, and is a pretty unique
file.  In this case, it is created automatically by ``autoInputs`` based on the
physical size of the ``Components.i.tri`` surface.  The other auto-created
file, ``preSpec.c3d.cntl`` defines regions in which the volume mesh should
have increased resolution.  Calling ``cubes`` also generates regions of
increased resolution based on distance from the surface, but this file can be
used to request more detail.  In addition to some header lines, the contents
look something like the following.

    .. code-block:: none
    
        # BBox: level   Xmin   Xmax      Ymin   Ymax      Zmin    Zmax
        #       (int)  (float) (float) (float) (float)  (float) (float)
        
        
        $__Prespecified_Adaptation_Regions:     # <-Section head (req'd)
        BBox: 6   -0.800   8.800   -4.800   4.800   -4.800   4.800   #  Config BBox
        BBox: 7   -0.299   1.299   -0.800   0.800   -0.799   0.799   #  Comp #0
        BBox: 7    1.700   7.300   -0.800   0.800   -0.800   0.800   #  Comp #1
        BBox: 7    7.201   8.799   -0.800   0.800   -0.799   0.799   #  Comp #2
        BBox: 7    6.479   7.653   -0.401   0.401    1.099   1.900   #  Comp #10
        BBox: 7    6.479   7.653   -1.900  -1.099   -0.401   0.401   #  Comp #11
        BBox: 7    6.479   7.653   -0.401   0.400   -1.900  -1.099   #  Comp #12
        BBox: 7    6.479   7.653    1.099   1.900   -0.400   0.401   #  Comp #13

The third row of *BBox* commands define a region with *x*-coordinates between
1.7 and 7.3, *y*-coordinates between -0.8 and +0.8, and *z*-coordinates between
-0.8 and +0.8.  Within this region, ``cubes`` must make a mesh that has been
refined at least 7 times.  In other words, the mesh size must be at least 128
times smaller than the original mesh.

Now let's look at the files in a run folder.

    .. code-block:: none
    
        $ cd m1.75a1.0r0.0
        $ ls
        body.dat              Components.i.tri     history.dat    moments.dat
        bullet_no_base.dat    Components.i.triq    input.00.cntl  preSpec.c3d.cntl
        bullet_total.dat      conditions.json      input.c3d      run.00.200
        cap.dat               Config.xml           input.cntl     run_cart3d.pbs
        case.json             cutPlanes.00200.plt  loadsCC.dat     
        check.00200           entire.dat           Mesh.c3d.Info  
        checkDT.00200         forces.dat           Mesh.mg.c3d    
        Components.00200.plt  functional.dat       Mesh.R.c3d     

Obviously, there are quite a few files, although many of them are links. For
example, the files that are listed here and in the parent folder discussed
above are either links or copies. The ``input.c3d`` and
``preSpec.c3d.cntl`` files are copied because they are small.

Most of the files ending with ``.dat`` are iterative history files. Some of
these are standard results of running ``flowCart``, and others are specifically
requested. The most special of these is ``history.dat``, which contains the
residual history. In pyCart, this file is used to determine how many iterations
have been run. With the exception of some comment lines, each line reports one
iteration number and the residual at that iteration.

The files ``forces.dat`` and ``moments.dat`` report the forces and
moments on the ``entire`` component, i.e. the entire triangulation. These files
are always produced, report results before any axis changes, and are ignored by
pyCart. Four other files, ``body.dat``, ``bullet_no_base.dat``,
``bullet_total.dat``, and ``cap.dat``, are specifically requested.
Cart3D produces them because the ``input.cntl`` file contains lines ``Force
body``, ``Force cap``, etc. in the ``$__Force_Moment_Processing:`` section.
Although we did not request ``entire`` in our pyCart setup, it got produced
here because the template ``input.cntl`` file contains the line ``Force
entire``. These ``.dat`` files are used by pyCart to read the iterative history
of forces and moments on parts of the vehicle.

The volume and surfaceresults files are ``check.00200``,
``Components.00200.plt``, ``Components.i.triq``, and ``cutPlanes.00200.plt``.
The ``check.00200`` file is a binary file used and created by Cart3D, and the
``plt`` files are Tecplot files. These Tecplot files are created by Cart3D, and
pyCart changes the file names by inserting the iteration numbers to which they
correspond. Finally, the ``Components.i.triq`` file is very similar to the
surface triangulation except with extra info describing the state solution at
each vertex. Noe that the ``Components.0200.plt`` and ``Components.i.triq``
files do not contain identical information because the Tecplot file references
the Cartesian volume mesh projected onto the surface while the ``triq`` file
only has solution data at the triangulation vertices.

Also in this folder are the files ``run_cart3d.pbs``, which is a script used to
run ``flowCart``.

    .. code-block:: bash
    
        #!/bin/bash
        #PBS -S /bin/bash
        #PBS -N m1.75a1r0
        #PBS -r n
        #PBS -j oe
        #PBS -l select=1:ncpus=12:mpiprocs=12
        #PBS -l walltime=2:00:00
        #PBS -q normal
        
        # Go to the working directory.
        cd /u/wk/ddalle/usr/pycart/examples/pycart/02_arrow/poweroff/m1.75a1.0r0.0
        
        # Additional shell commands
        
        # Call the flowCart/mpix_flowCart/aero.csh interface.
        run_flowCart.py

The script includes some PBS settings (which are not used in this example), a
command to change to the correct folder using an absolute path, whatever shell
commands are specified in the JSON file, and a command to determine the correct
Cart3D command.  The file ``case.json`` contains all of the
``pyCart.json`` settings from the ``"flowCart"`` section, because they are
needed to determine the command-line inputs.

That covers the essential files for this example.  The very import
``input.cntl`` file (which in this case is just a link to
``input.00.cntl``) is worthy of far more discussion, and there are several
other files that have varying degrees of utility, but that will have to come at
a different time and place.