Plotting Examples#

There are a variety of methods that can be used to visualize a network.

import pynucastro as pyna

Standard view#

Let’s build a quick network that represents He, C, and O burning

library = pyna.ReacLibLibrary()
sub = library.linking_nuclei(["p", "n", "he4", "c12", "n13", "n14", "o16",
                              "ne20", "ne21", "na23", "mg23", "mg24", "al27",
                              "si27", "si28"])
rc = pyna.RateCollection(libraries=sub)
/opt/hostedtoolcache/Python/3.11.11/x64/lib/python3.11/site-packages/pynucastro/networks/rate_collection.py:576: UserWarning: ReacLib neutron decay rate (<n_to_p_weak_wc12>) does not account for degeneracy at high densities. Consider using tabular rate from Langanke.
  warnings.warn(msg)

The standard network plot of this is:

fig = rc.plot()
Ignoring fixed x limits to fulfill fixed data aspect with adjustable data limits.
_images/f9932b745b6379aef8ef278937730623df9913abe45564c1119e3c0083f53032.png

Rotated view#

Since everything lies close to the diagonal, it is hard to see the structure. We can instead look at a rotated view of the network.

While we’re at it, we’ll also color the rates by reaction rate strength

rho = 1.e4
T = 1.e8
comp = pyna.Composition(rc.get_nuclei())
comp.set_solar_like()
fig = rc.plot(rho, T, comp, rotated=True)
_images/5e1db9737dc884602fd2755c8eccb9be62c87e20a60de8f526e7b268e19ec71e.png

This looks a bit crowded, since it is showing all of the links to \({}^4\mathrm{He}\) from reactions of the form \(A(x, \alpha)B\). We can hide those links with hide_xalpha=True

fig = rc.plot(rho, T, comp, rotated=True, hide_xalpha=True)
_images/f84c0091c67c8ce27282aec13d1be70c57f03d6bb2e2d70e95d448433744f984.png

We can also plot the edges curved, to help distinguish the forward and reverse rates

fig = rc.plot(rho, T, comp, rotated=True, hide_xalpha=True,
              curved_edges=True)
_images/e0fe1b122a4344f6061efab90160d5de142cf3fb542035b3f5765c61ce3d93fc.png

We can also label edges if desired

edge_labels = {(pyna.Nucleus("he4"), pyna.Nucleus("c12")):
               r"$\alpha(\alpha\alpha,\gamma){}^{12}\mathrm{C}$"}
fig = rc.plot(rho, T, comp, rotated=True, hide_xalpha=True,
              edge_labels=edge_labels, size=(1200, 700),
              curved_edges=True)
_images/127657ce176c87453494107709a72e81fe658e10bb17e411eb76d060acb0252d.png

Filters#

We can use a rate filter to show only certain links. For example, to show only links involving proton capture, we can do:

fig = rc.plot(rho, T, comp, rotated=True, hide_xalpha=True,
              rate_filter_function=lambda r: pyna.Nucleus("p") in r.reactants)
_images/78f011852a43e9996cd39149f2b69a7f9fba3837ba88363d713a31cd5ac3914e.png

We can also highlight rates using the highlight_filter_function mechanism. This takes a function that operates on a rate and returns True if we wish to highlight the rate (i.e., as if we used a highlighter marker over the line).

Here’s an example where we highlight the rates that have \(Q > 0\)

fig = rc.plot(rotated=True, hide_xalpha=True, size=(1500, 600),
              highlight_filter_function=lambda r: r.Q > 0,
              curved_edges=True)
_images/8ddd0e341753864ebfd7408537f8e50395bc1d2fe1ff96916922d06ca9463fbe.png

Plotting the Jacobian#

For implicit integration, we solve:

\[\dot{\bf Y} = {\bf f}({\bf Y})\]

and the Jacobian, \({\bf J} = \partial {\bf f} / \partial {\bf Y}\) plays an important role in solving for the update.

We can visualize the Jacobian of our network easily. By default, we will show all elements where

$$\frac{\mathrm{max}{i,j} (|J{ij}|)}{|J_{ij}|} > 10^{10}$$

but this ratio can be set via the rate_scaling keyword argument.

rho = 1.e4
T = 2.e9
comp.set_equal()
fig = rc.plot_jacobian(rho, T, comp, rate_scaling=1.e10)
_images/0cd447408cdd1bc4a8168fb0053665087c69688011c19f9b4494769d050d4875.png

Note

If you set rate_scaling too large, then roundoff error in mapping the Jacobian elements to colors can cause elements that are zero to appear colored. It is best to stay below 1.e15 to minimize roundoff.

Plot nuclides on a grid#

Nuclides in a network may also be visualized as cells on a grid of Z vs. N, colored by some quantity. This can be more interpretable for large networks.

Calling gridplot without any arguments will just plot the grid – to see anything interesting we need to supply some conditions. Here is a plot of nuclide mass fraction on a log scale, with a 36 square inch figure:

rate_names = ["c12(p,g)n13", 
              "c13(p,g)n14",
              "n13(,)c13",
              "n13(p,g)o14",
              "n14(p,g)o15",
              "n15(p,a)c12",
              "o14(,)n14",
              "o15(,)n15"]
rates = library.get_rate_by_name(rate_names)
rc = pyna.RateCollection(rates=rates)
rho = 1.e4
T = 1.e8
comp = pyna.Composition(rc.get_nuclei())
comp.set_solar_like()
fig = rc.gridplot(comp=comp, color_field="X", scale="log", area=36)
_images/81aba4d3a0218931262b58d245d834ff1f927844738574aaacec8b6f692de965.png

The plot is configurable through a large number of keyword arguments. Here we want to look at the rates at which nuclides are being created or destroyed, so we color by \(\dot{Y}\), the rate of change of molar abundance. Density and temperature need to be supplied to evaluate the rates. A full list of valid keyword arguments can be found in the API documentation.

fig = rc.gridplot(comp=comp, rho=rho, T=T, color_field="ydot", area=36,
                  cmap="RdGy", cbar_bounds=(-0.1, 0.1))
_images/a2ef888bd5af98be129e6e2d0503123877ffe1bffb911e34bc749361cc757c38.png

Unlike the network plot, this won’t omit hydrogen and helium by default. To just look at the heavier nuclides, we can define a function to filter by proton number. Here we also plot activity, which is the sum of all rates contributions to Ydot ignoring the sign.

ff = lambda nuc: nuc.Z > 2
fig = rc.gridplot(comp=comp, rho=rho, T=T, color_field="activity", scale="log",
                  filter_function=ff, area=20, cmap="Blues")
_images/9371c0ff15d48a0ffc86b82c19cb1190bd209b15e71044759110fb59c6fc42e0.png

Network chart#

A network chart shows how each rate affects each nucleus

fig = rc.plot_network_chart(rho, T, comp)
_images/e23f67c991b397848f1241cdef67ca236e8a18a05ea952a122ad0566d631114b.png