
Fast centrifugal deformation program. Can be applied to both stellar and planetary models.
Explore the docs
v1.1.0
View Demo
·
Report Bug
·
Request Feature
Table of Contents
RUBIS
(standing for Rotation code Using Barotropy conservation over Isopotential Surfaces) is a centrifugal deformation program that takes as input a 1D model (with spherical symmetry) and returns its deformed version by applying a conservative rotation profile specified by the user.
More specifically, the code only needs the density as a function of radial distance,
As a result, the only equation effectively solved by the program is Poisson's equation,
Finally, the code has been designed to allow both stellar and planetary models to be deformed, thereby dealing with potential discontinuities in the density profile. This is made possible by solving Poisson's equation in spheroidal rather than spherical coordinates whenever a discontinuity is present. More details regarding the Deformation Method can be found below.
Regarding the program itself, RUBIS
is currently in version v.1.1.0
, reflecting the major changes that follow:
v.0.1.0
:RUBIS
relies on an adaptive rotation rate to imporve the stability of the iterative scheme.v.0.2.0
:RUBIS
integrates diagnostics to evaluate the solution's precision, such as the precision on Poisson's equation and the Virial test. The precision of numerous interpolation routines has also been improved.v.0.3.0
:RUBIS
finds the new mappping using a reciprocal interpolation on an adaptive grid instead of Newton's method, considerably improving the stability of the spheroidal method. The performance of the spherical method is also improved by solving Poisson's equation only once.v.1.0.0
:RUBIS
integrates a proper user interface,RUBIS.py
, allowing all parameters to be tuned from the same file.v.1.1.0
:RUBIS
offers a new feature, the radiative-flux computation on the deformed model surface.
Get a local copy of RUBIS
by following the following steps.
RUBIS
has been written in such a way that it only depends on the standard Python libraries: NumPy, SciPy and Matplotlib. The setup.py
is only here to ensure that these libraries are up to date.
Note: since v1.1.0
, the program may benefit the use of ProPlot for displaying colormaps. It should not raise any error if not installed, however.
- Clone the repo
git clone https://github.com/pierrehoudayer/RUBIS.git
- Make sure that the standard libraries are up to date by running the
setup.py
python setup.py install
![]() |
---|
Flowchart illustrating how the model deformation method works. Each step shows the quantity that is obtained and in terms of which variable it is obtained. |
As described in the flowchart, RUBIS
uses an iterative approach to determine the deformation induced by the rotation profile.
The method's central assumption is the preservation of the barotropic relation
- If the model do not contain any discontinuity, the density profile is first interpolated onto spherical coordinates in order to then solve Poisson's equation in this coordinate system and obtain the gravitational potential
$\Phi_G(r, \theta)$ . Because the decomposition of Poisson's equation over spherical harmonics can be decoupled, this path is the fastest one. - If the model do contain discontinuities, Poisson's equation is directly solved in terms of
$(\zeta, \theta)$ , thus yielding$\Phi_G(\zeta, \theta)$ . The reason for this change is that discontinuities follows isopotentials (which are also isobars from the hydrostatic equilibrium), and therefore that fixed values of$r$ cross multiple domains, making unhandly to solve the equation in the (simpler) spherical coordinate system. Since the isopotential shapes$\zeta(r, \theta)$ are known from the previous iteration, the gravitational can simply be reexpressed as$\Phi_G(r, \theta)$ , leading to the same quantity as the other path.
The hydrostatic equilibrium implies that the total potential (expressed on the isopotentials, $\Phi_\mathrm{eff}(\zeta)$) only varies by an additive constant from one iteration to the next. Therefore, once the gravitational potential has been calculated, the constant can be determined by applying this relationship on the origin which allows to express the total potential on each isopotential,
In parallel the total potential, solid
, lorentzian
, have a plateau
in the central region for instance, the only constraint is that it must conservative, i.e. a function of the distance from the axis of rotation,
Before this step, however, an adaptive rotation rate,
![]() |
---|
Plot showing sub and super-critical isopotential lines. Finding an adaptive rotation rate ensures that new level surfaces remain in the sub-critical range. |
Since
The code runs until it meets a convergence criterion, typically if the polar radius of the deformed model changes less than a user-specified threshold from one iteration to the next.
On a practical level, the user interface can be found in the file RUBIS.py
while the core of the program is located in the files model_deform_radial.py
and model_deform_spheroidal.py
(depending on whether Poisson's equation is solved in radial or spheroidal coordinates).
The routine in these files largely rely on functions that can be found in the helpers.py
(functions common to both deformation methods), legendre.py
(functions involved in the spectral decomposition and evaluation), numerical.py
(purely numerical functions) and plot.py
(functions involved to display the procedure's outputs).
The file rotation_profiles.py
contains the implementation of the rotation profiles and corresponding centrifugal potentials (solid
, lorentzian
, plateau
or la_bidouille
in case of user-specified numerical profile).
Finally, polytrope.py
contains the function used to generate 1D polytropes and composite polytropes to be deformed by either the radial_method()
and spheroidal_method()
functions.
With the method described, we will now give a few examples of how the code can be used.
Let's open the file RUBIS.py
.
Most of the actions to be performed are limited to modifying the parameters in the set_params()
function.
Let's start by choosing a 1D model to deform.
If I do not have a model at hand (and I do not want to deform a model already present in the ./Models/
directory), I can choose to deform a polytrope of index indices
, specify its surface pressure with target_pressures
(in model_choice
:
#### MODEL CHOICE ####
model_choice = DotDict(indices = 3.0, target_pressures = -np.inf)
I can specify more options to build the polytrope like its mass or radius for instance. The above dictionary is equivalent in practice to the following choice:
#### MODEL CHOICE ####
model_choice = DotDict(
indices = 3.0,
target_pressures = -np.inf,
density_jumps = None,
mass = 1.0,
radius = 1.0,
resolution = 1001
)
with resolution
indicating the number of points in the radial direction.
We'll come back to the other options in a following example.
We now need to specify how we want to deform the model, i.e. the rotation profile to consider.
In this first simple example, I will just consider a solid rotation profile at rate
#### ROTATION PARAMETERS ####
rotation_profile = solid
rotation_target = 0.9
central_diff_rate = 1.0
rotation_scale = 1.0
It is not necessary to modify the other rotation parameters central_diff_rate
and rotation_scale
which are only involved in the cylindrical rotation profiles lorentzian
and plateau
.
Let's now have a look at the parameters affecting the resolution itself.
The most important parameters are max_degree
and angular_resolution
.
To avoid complications, it is recommended to set them to the same value (as is the case by default).
In our case the use of full_rate
iterations at the beginning of the deformation to gradually reach the target velocity rotation_target
.
It won't be necessary in this case, and I'll keep a value of full_rate = 1
.
There should be no convergence difficulties in our example, and we can require the procedure to be highly precise by setting mapping_precision
to 1e-12
, for instance.
However, from experience, diminishing this value is not always beneficial in terms of accuracy while necessarily having a cost in terms of computation time (by increasing the number of iterations).
I would thus recommend setting this value to 1e-10
or 1e-11
and only consider values below 1e-12
in cases where highly precise computations are required (when calculating gravitational moments, for instance) and when the model allows it (i.e when it has sufficient radial resolution).
Note that most of (legal!) deformations will converge with such criterion but really complex cases may sometimes requiere lowering this value.
The last two parameters are a slighly more technical: lagrange_order
sets the effective order for the finite-differences scheme when solving Poisson's equation and spline_order
sets the degree of interpolating B-splines in the routine.
Keeping the latter to lagrange_order
may be adapted between
#### SOLVER PARAMETERS ####
max_degree = angular_resolution = 101
full_rate = 1
mapping_precision = 1e-10
lagrange_order = 3
spline_order = 5
The parameters presented above are the main ones to modify when running the deformation procedure.
There are many more parameters controlling the RUBIS
' output that are documented in the set_params()
function.
We will come back on a few of them latter in the example.
The output parameters that I have used here look like:
#### OUTPUT PARAMETERS ####
output_params = DotDict(
show_harmonics = False,
virial_test = True,
show_model = True,
plot_resolution = 501,
plot_surfaces = True,
plot_cmap_f = get_cmap_from_proplot("Greens1_r"),
plot_cmap_surfaces = get_cmap_from_proplot("Greys"),
gravitational_moments = False,
radiative_flux = False,
plot_flux_lines = True,
flux_origin = 0.05,
flux_lines_number = 20,
show_T_eff = True,
flux_res = (200, 100),
flux_cmap = get_cmap_from_proplot("Stellar_r"),
dim_model = False,
save_model = False,
save_name = give_me_a_name(model_choice, rotation_target)
)
We can now run the code.
If all goes well, it should display the current iteration with the value of the polar radius (expressed in units of equatorial radius).
Once the variation of the polar radius drops below mapping_precision
, the program stops and displays the time required for the deformation:
In [12]: %run "/home/phoudayer/Documents/Codes/RUBIS/RUBIS.py"
+---------------------+
| Deformation started |
+---------------------+
Iteration n°01, R_pol = 0.7131439994
Iteration n°02, R_pol = 0.7092052194
Iteration n°03, R_pol = 0.7087377715
Iteration n°04, R_pol = 0.7085851812
Iteration n°05, R_pol = 0.708519354
Iteration n°06, R_pol = 0.7084951248
Iteration n°07, R_pol = 0.7084845984
Iteration n°08, R_pol = 0.7084798012
Iteration n°09, R_pol = 0.7084777463
Iteration n°10, R_pol = 0.7084768433
Iteration n°11, R_pol = 0.7084764395
Iteration n°12, R_pol = 0.7084762616
Iteration n°13, R_pol = 0.708476183
Iteration n°14, R_pol = 0.7084761481
Iteration n°15, R_pol = 0.7084761327
Iteration n°16, R_pol = 0.7084761258
Iteration n°17, R_pol = 0.7084761228
Iteration n°18, R_pol = 0.7084761215
Iteration n°19, R_pol = 0.7084761209
Iteration n°20, R_pol = 0.7084761206
Iteration n°21, R_pol = 0.7084761205
Iteration n°22, R_pol = 0.7084761204
+------------------+
| Deformation done |
+------------------+
Time taken: 4.1 secs
Kinetic energy : 0.0178210325
Internal energy : 0.6584638677
Potential energy: 4.0220673364
Surface term : 0.0000000000
Virial theorem verified at -2.8947e-12
I can see here that the deformation only took about 4 seconds (and 22 iterations).
I also have a quantification of how well the Virial theorem is verified, here about 3e-12
, since I have set the option virial_test = True
.
A plot with the mapping and the density in the deformed model should also appear, as specified by the requierement show_model = True
:
![]() |
---|
Deformation of a polytropic structure with index N=3 at 90% of the Keplerian rotation rate. Isopotentials are shown on the left and the density distribution on the right. |
The angular resolution of the plot can be modified with the plot_resolution
parameter in output_params
, as well as whether to show the isopotentials on the left, and the colormaps to use.
Finally, the output model can be saved in order to be exploited externally specifying the save_model
option in output_params
.
We only have to indicate the filename as save_name
and whether the model should be saved in dimensioned or dimensionless version with the dim_model
option.
It is also possible to use the command:
save_name = give_me_a_name(model_choice, rotation_target)
which automatically construct an output filename from the input one.
Note that there is not a "clever" way to specify which variables to save for now.
One must modify manually the content of the write_model()
function in either the radial_deform()
or spheroidal_deform()
routines, according to the documentation.
For instance, the command: if I want to save the pressure and density (in addition to the mapping which is automatically saved), I will write:
write_model(
output_params.save_name,
(N, M, mass, radius, rotation_target, G),
map_n,
additional_var,
zeta, P, rho, phi_eff, rota
)
will save the radial and angular resolutions, as well as the model mass, radius and equatorial rotation rate in addition to the gravitational constant in a header.
It will also save the mapping
Note that additional variables in the input file are automatically passed through the program without alterations and returned at the end of the output file for file-writting convenience.
However, make sure that these variables are invariant on the isopotentials during the deformation or they will no longer correspond to the 2D structure of the model!
Let's say that I want to test RUBIS
' stability, and therefore deform the very same model at a rotation rate close to the critical one.
Since the rotation_target
as follows:
rotation_target = 0.9999
As we will likely be dealing with a considerably deformed model, the number of harmonics,
#### SOLVER PARAMETERS ####
max_degree = angular_resolution = 401
full_rate = 3
The deformation procedure (for a continuous model) scaling roughly with
We can run the RUBIS.py
without any other changes:
In [2]: %run "/home/phoudayer/Documents/Codes/RUBIS/RUBIS.py"
+---------------------+
| Deformation started |
+---------------------+
Iteration n°01, R_pol = 0.9473783921
Iteration n°02, R_pol = 0.8174127887
Iteration n°03, R_pol = 0.6652743254
Iteration n°04, R_pol = 0.6640548366
Iteration n°05, R_pol = 0.6616264593
Iteration n°06, R_pol = 0.6638080392
Iteration n°07, R_pol = 0.66378137
Iteration n°08, R_pol = 0.6637704632
Iteration n°09, R_pol = 0.663765638
Iteration n°10, R_pol = 0.6637635109
Iteration n°11, R_pol = 0.6637625834
Iteration n°12, R_pol = 0.6637621725
Iteration n°13, R_pol = 0.6637619902
Iteration n°14, R_pol = 0.6637619097
Iteration n°15, R_pol = 0.663761874
Iteration n°16, R_pol = 0.6637618582
Iteration n°17, R_pol = 0.6637618512
Iteration n°18, R_pol = 0.6637618481
Iteration n°19, R_pol = 0.6637618467
Iteration n°20, R_pol = 0.6637618461
Iteration n°21, R_pol = 0.6637618459
Iteration n°22, R_pol = 0.6637618457
Iteration n°23, R_pol = 0.6637618457
+------------------+
| Deformation done |
+------------------+
Time taken: 20.58 secs
Kinetic energy : 0.0193605344
Internal energy : 0.7020113009
Potential energy: 4.2895099430
Surface term : 0.0000000000
Virial theorem verified at -3.0337e-12
The procedure took slighly longer than expected (probably because of the transcient phase): ~ show_harmonics = True
in output_params
.
This will show us a plot in log scale of the gravitational potential harmonics:
![]() |
---|
Gravitational potential harmonics. Lower and higher degrees correspond to darker and lighter shades, respectively. |
As it is clear that the gravitational potential cannot be known with more precision than the higher degree harmonic's amplitude, this plot informs us on the precision to which Poisson's equation is solved. A message should also appear in the shell:
Estimated error on Poisson's equation: 4e-16
We see here that there should not be any problems coming our Poisson's equation solving. A plot of the model should again be available:
![]() |
---|
Deformation of a polytropic structure with index N=3 at 99.99% of the Keplerian rotation rate. Isopotentials are shown on the left and the density distribution on the right. |
We can see here a clear cusp on the edges, reflecting that we are indeed rather close to the critical rotation rate on the equator.
As said above, the critical rotation rate might not correspond to the Keplerian one for models with more homogeneous mass distributions, which can allow us to exceed the Keplerian rotation rate.
We will test this assumption by considering here a polytrope of index
#### MODEL CHOICE ####
model_choice = DotDict(indices = 0.5, target_pressures = -np.inf, resolution=3001)
Since it will end up being considerably more deformed than the previous polytrope, it is recommended to use more points in the radial resolution, here
#### ROTATION PARAMETERS ####
rotation_profile = solid
rotation_target = 1.105
central_diff_rate = 1.0
rotation_scale = 1.0
#### SOLVER PARAMETERS ####
max_degree = angular_resolution = 401
full_rate = 3
mapping_precision = 1e-10
lagrange_order = 2
spline_order = 5
I reduced the order of the finite-differences scheme (as I have increased the resolution) but, more importantly, I set the equatorial rotation rate to
In [3]: %run "/home/phoudayer/Documents/Codes/RUBIS/RUBIS.py"
+---------------------+
| Deformation started |
+---------------------+
Iteration n°01, R_pol = 0.9364745134
Iteration n°02, R_pol = 0.7672390968
Iteration n°03, R_pol = 0.6175574971
Iteration n°04, R_pol = 0.5466449736
Iteration n°05, R_pol = 0.5046344772
Iteration n°06, R_pol = 0.4791713897
Iteration n°07, R_pol = 0.4633433499
Iteration n°08, R_pol = 0.4551258659
Iteration n°09, R_pol = 0.450644672
Iteration n°10, R_pol = 0.447623725
Iteration n°11, R_pol = 0.4456423532
Iteration n°12, R_pol = 0.4444527056
Iteration n°13, R_pol = 0.4435563768
Iteration n°14, R_pol = 0.4430896373
Iteration n°15, R_pol = 0.4427447297
Iteration n°16, R_pol = 0.4425369649
Iteration n°17, R_pol = 0.4423929664
Iteration n°18, R_pol = 0.4422978062
Iteration n°19, R_pol = 0.4422369117
Iteration n°20, R_pol = 0.442197209
Iteration n°21, R_pol = 0.4421713782
Iteration n°22, R_pol = 0.4421546071
Iteration n°23, R_pol = 0.4421436031
Iteration n°24, R_pol = 0.4421365426
Iteration n°25, R_pol = 0.442131964
Iteration n°26, R_pol = 0.4421289567
Iteration n°27, R_pol = 0.4421269855
Iteration n°28, R_pol = 0.4421257097
Iteration n°29, R_pol = 0.4421248793
Iteration n°30, R_pol = 0.4421243146
Iteration n°31, R_pol = 0.4421239646
Iteration n°32, R_pol = 0.442123728
Iteration n°33, R_pol = 0.4421235724
Iteration n°34, R_pol = 0.4421234701
Iteration n°35, R_pol = 0.442123404
Iteration n°36, R_pol = 0.4421233624
Iteration n°37, R_pol = 0.4421233364
Iteration n°38, R_pol = 0.4421233205
Iteration n°39, R_pol = 0.4421233108
Iteration n°40, R_pol = 0.4421233049
Iteration n°41, R_pol = 0.4421233013
Iteration n°42, R_pol = 0.4421232992
Iteration n°43, R_pol = 0.4421232979
Iteration n°44, R_pol = 0.4421232971
Iteration n°45, R_pol = 0.4421232965
Iteration n°46, R_pol = 0.4421232963
Iteration n°47, R_pol = 0.4421232961
Iteration n°48, R_pol = 0.4421232961
+------------------+
| Deformation done |
+------------------+
Time taken: 79.13 secs
Estimated error on Poisson's equation: 1.534198839e-07
Kinetic energy : 0.1627972894
Internal energy : 0.1920031872
Potential energy: 1.8032089377
Surface term : 0.0000000000
Virial theorem verified at -1.820523534e-07
We see that more iterations were required for convergence (2e-7
).
A look at the gravitational potential harmonics shows that we have a very slow convergence of the partial series
Here is the resulting model:
![]() |
---|
Deformation of a polytropic structure with index N=1/2 at 110.5% of the Keplerian rotation rate. Isopotentials are shown on the left and the density distribution on the right. |
We see that there is a considerable flattening due to the more homogeneous mass distribution, and that we are again close to the critical rotation rate, which we can test by running the program with rotation_target = 1.11
:
In [4]: %run "/home/phoudayer/Documents/Codes/RUBIS/RUBIS.py"
+---------------------+
| Deformation started |
+---------------------+
Iteration n°01, R_pol = 0.9359352379
Iteration n°02, R_pol = 0.7656043185
Iteration n°03, R_pol = 0.6185886618
Iteration n°04, R_pol = 0.5482268566
Iteration n°05, R_pol = 0.506512168
Iteration n°06, R_pol = 0.4809454073
Iteration n°07, R_pol = 0.4648374205
Iteration n°08, R_pol = 0.4545022485
Iteration n°09, R_pol = 0.4493768901
Iteration n°10, R_pol = 0.4462242424
---------------------------------------------------------------------------
ValueError: Error on input data
In the Models/
directory, there is a model of Jupiter that we can try to deform:
#### MODEL CHOICE ####
model_choice = 'Jupiter.txt'
This model contains two discontinuities (the first resulting from the presence of a solid core and the second from an hydrogen phase change), meaning that RUBIS
will call the deform_spheroidal()
method as long as ot run in auto mode (highly recommended!):
#### METHOD CHOICE ####
method_choice = 'auto'
This method being much more costly than deform_radial()
from a numerical point of view (see the article for more details), it is preferable to reduce the number of harmonics to use:
#### ROTATION PARAMETERS ####
rotation_profile = solid
rotation_target = 0.9
central_diff_rate = 1.0
rotation_scale = 1.0
#### SOLVER PARAMETERS ####
max_degree = angular_resolution = 101
full_rate = 1
mapping_precision = 1e-10
lagrange_order = 2
spline_order = 5
When running the program, we obtain:
In [5]: %run "/home/phoudayer/Documents/Codes/RUBIS/RUBIS.py"
+---------------------+
| Deformation started |
+---------------------+
Iteration n°01, R_pol = 0.7117437964
Iteration n°02, R_pol = 0.6642329043
Iteration n°03, R_pol = 0.6385151656
Iteration n°04, R_pol = 0.6265341389
Iteration n°05, R_pol = 0.6205290412
Iteration n°06, R_pol = 0.6174305796
Iteration n°07, R_pol = 0.6158408091
Iteration n°08, R_pol = 0.6150241936
Iteration n°09, R_pol = 0.6146033552
Iteration n°10, R_pol = 0.6143862351
Iteration n°11, R_pol = 0.6142741791
Iteration n°12, R_pol = 0.6142163285
Iteration n°13, R_pol = 0.6141864558
Iteration n°14, R_pol = 0.6141710286
Iteration n°15, R_pol = 0.6141630611
Iteration n°16, R_pol = 0.614158946
Iteration n°17, R_pol = 0.6141568205
Iteration n°18, R_pol = 0.6141557227
Iteration n°19, R_pol = 0.6141551556
Iteration n°20, R_pol = 0.6141548627
Iteration n°21, R_pol = 0.6141547114
Iteration n°22, R_pol = 0.6141546332
Iteration n°23, R_pol = 0.6141545929
Iteration n°24, R_pol = 0.614154572
Iteration n°25, R_pol = 0.6141545612
Iteration n°26, R_pol = 0.6141545557
Iteration n°27, R_pol = 0.6141545528
Iteration n°28, R_pol = 0.6141545513
Iteration n°29, R_pol = 0.6141545505
Iteration n°30, R_pol = 0.6141545501
Iteration n°31, R_pol = 0.6141545499
Iteration n°32, R_pol = 0.6141545498
Iteration n°33, R_pol = 0.6141545498
+------------------+
| Deformation done |
+------------------+
Time taken: 99.46 secs
Estimated error on Poisson's equation: 9.217224e-10
Kinetic energy : 0.0905246063
Internal energy : 0.2451316995
Potential energy: 1.8326119730
Surface term : 0.0000000057
Virial theorem verified at 7.54707317054e-05
As expected, the spheroidal method is slower (Jupiter.txt
, the jump in density caused by the core is about
Here is the plot with the option plot_surfaces = False
in output_params
, so that we can clearly see the discontinuities:
![]() |
---|
Deformation of a model of Jupiter at 90% of the Keplerian rotation rate with its density distribution (in log scale). The discontinuities are shown in white. |
There are multiple ways to define a differential rotation profile for a model.
RUBIS
implements two analytical profiles: lorentzian
and plateau
.
-
The first needs a single additional parameter compared to the
solid
profile (which only depends onrotation_target
) that gives the relative rate difference between the center and the equator:$\alpha = (\Omega_0 - \Omega_\mathrm{eq}) / \Omega_\mathrm{eq}$ . The latter is obtained by specifying the value ofcentral_diff_rate
. -
The
plateau
profile needs an extra parameter in addition torotation_target
andcentral_diff_rate
, which isrotation_scale
. The latter indicates the lenght scale of the plateau in the rotation profile. For instance, the combinaisoncentral_diff_rate = 1.0
androtation_scale = 0.5
results in a profile with a central plateau rotating twice as fast as the equatorial rotation rate over ~$R_\mathrm{eq}/2$ , followed by a decrease to reach the equatorial rotation rate at$R_\mathrm{eq}$ . -
RUBIS
also allows the user to specify its own function$\Omega(s)$ numerically by providing a discretisation of this profile on a grid going from$s = 0$ to$s = 1$ . The latter should be contained on a file located in theModels/
directory and can be called with the commandrotation_profile = la_bidouille(fname, smoothing)
, wherefname
is the name of the file andsmoothing
can be used to smooth the numerical profile.
In the following, we will consider the deformation of an
#### ROTATION PARAMETERS ####
rotation_profile = lorentzian
rotation_target = 0.97
central_diff_rate = 5.0
This is a quite extreme example, close to the critical rotation rate at multiple locations in the model, but the procedure converges without too much difficulty:
In [6]: %run "/home/phoudayer/Documents/Codes/RUBIS/RUBIS.py"
+---------------------+
| Deformation started |
+---------------------+
Iteration n°01, R_pol = 0.818195594
Iteration n°02, R_pol = 0.5463293458
Iteration n°03, R_pol = 0.3583846729
Iteration n°04, R_pol = 0.3001048147
Iteration n°05, R_pol = 0.2940297034
Iteration n°06, R_pol = 0.2834783666
Iteration n°07, R_pol = 0.2740794419
Iteration n°08, R_pol = 0.2689482907
Iteration n°09, R_pol = 0.2653685831
Iteration n°10, R_pol = 0.2629754631
Iteration n°11, R_pol = 0.2614954317
Iteration n°12, R_pol = 0.2605639267
Iteration n°13, R_pol = 0.2599678942
Iteration n°14, R_pol = 0.2596621136
Iteration n°15, R_pol = 0.2594861206
Iteration n°16, R_pol = 0.2591778582
Iteration n°17, R_pol = 0.2589535797
Iteration n°18, R_pol = 0.2588484694
Iteration n°19, R_pol = 0.2588099488
Iteration n°20, R_pol = 0.258799086
Iteration n°21, R_pol = 0.2587978365
Iteration n°22, R_pol = 0.2587992407
Iteration n°23, R_pol = 0.2588009542
Iteration n°24, R_pol = 0.2588023379
Iteration n°25, R_pol = 0.258803317
Iteration n°26, R_pol = 0.258803969
Iteration n°27, R_pol = 0.2588043898
Iteration n°28, R_pol = 0.2588046565
Iteration n°29, R_pol = 0.2588048238
Iteration n°30, R_pol = 0.2588049281
Iteration n°31, R_pol = 0.2588049928
Iteration n°32, R_pol = 0.2588050329
Iteration n°33, R_pol = 0.2588050578
Iteration n°34, R_pol = 0.2588050731
Iteration n°35, R_pol = 0.2588050826
Iteration n°36, R_pol = 0.2588050884
Iteration n°37, R_pol = 0.2588050921
Iteration n°38, R_pol = 0.2588050943
Iteration n°39, R_pol = 0.2588050957
Iteration n°40, R_pol = 0.2588050965
Iteration n°41, R_pol = 0.258805097
Iteration n°42, R_pol = 0.2588050974
Iteration n°43, R_pol = 0.2588050976
Iteration n°44, R_pol = 0.2588050977
Iteration n°45, R_pol = 0.2588050978
+------------------+
| Deformation done |
+------------------+
Time taken: 39.82 secs
Estimated error on Poisson's equation: 5.6e-15
Kinetic energy : 0.1023561421
Internal energy : 1.6513168494
Potential energy: 10.3173256664
Surface term : 0.0000000000
Virial theorem verified at -5.92941e-11
We can already see the considerable flattening caused by this profile;
Its shape is rather original, as it is often the case with highly differential rotation profiles. You can find below a last example using the options:
#### MODEL CHOICE ####
model_choice = DotDict(indices = 4.5, target_pressures = -np.inf)
#### ROTATION PARAMETERS ####
rotation_profile = lorentzian
rotation_target = 0.2
central_diff_rate = 200.0
rotation_scale = 1.0
#### SOLVER PARAMETERS ####
max_degree = angular_resolution = 401
full_rate = 3
mapping_precision = 1e-10
lagrange_order = 3
spline_order = 5
Since v1.1.0
, RUBIS
allows the user to visualise the outcoming flux at the surface of the model.
This is made possible using an assumption expressed in (Espinosa Lara & Rieutord (2011)), i.e considering that the energy flux propagates along effective gravity lines (which is well verified, cf. their Fig. 1).
In this article, the authors solved analytically the radiative flux transfer
In RUBIS
however, effective gravity lines correspond to lines normal to the isopotentials (since their is a total potential as long as the rotation profile is conservative) and we know exactly their shapes when deforming the model.
The radiative flux transfer can therefore be solved numerically for basically free: we only need to compute the lines normal to the isopotential and integrate the flux equation along those lines.
This procedure works whatever the model structure and the rotation profile, whether it is differential or not.
To compute the radiative flux in RUBIS
, one simply needs to set the option radiative_flux = True
in output_params
. The other options can be chosen as follows:
radiative_flux = True,
plot_flux_lines = True,
flux_origin = 0.05,
flux_lines_number = 20,
show_T_eff = True,
flux_res = (200, 100),
flux_cmap = get_cmap_from_proplot("Stellar_r"),
plot_flux_lines
decides whether to add the flux lines to the model plot, and flux_lines_number
gives the number of flux lines to be computed:
![]() |
---|
Flux lines on top of the [First Example][first-simple-example]. |
The calculation is rather rapid:
Flux lines found in 0.47 secs
We indeed see that the flux lines are normal to the isopotentials.
The flux_origin
defines the
The last three parameters impact the 3D rendering of the radiative flux at the surface.
show_T_eff
decides whether to show the the effective temperature instead of the flux norm on the surface, flux_res
gives the mesh resolution in the flux_cmap
defines the colormap:
![]() |
---|
Corresponding effective temperature over the surface. |
We can clearly see the effect of the gravity darkening on the equator.
The flux of this model can still be approximated with the
![]() |
---|
Corresponding effective temperature over the surface. |
Here the surface is so deformed that almost no flux comes from the equator while the poles are considerably hotter than the rest of the surface.
- Implement a function for computing the gravitational moments from the deformed model.
- Allow the user to choose a multiple index
$(N_1, N_2, \ldots, N_k)$ polytrope (with potential density discontinuities on the interfaces) as a 1D model to deform. - Improve the density interpolation in the spherical method.
- Implement the Viriel test. ⇨
v0.2.0
- Implement a reciprocal interpolation to determine the level surfaces instead of Newton's method (which lack stability).
- Exploit the invariance of Poisson's matrix in the spherical method ⇨
v0.3.0
- Improve the conditioning of Poisson's matrix in the spheroidal method.
- Improve the reciprocal interpolation in the spheroidal method thanks to an adaptive
$\zeta$ grid. - Add the global user interface
RUBIS.py
. - Rewrite the completely outdated
README.md
⇨v1.0.0
- Implement a radiative flux model on the deformed surface to model the gravitaty darkening caused by the rotation. ⇨
v1.1.0
- Find a way to improve the hydrostatic equilibrium in discontinuous models.
- Look at series acceleration/transformation to accelerate the gravitational potential series convergence.
- Propose 2D incomming flux map to simulate interferometric data and allow a parametric adjustment (rotation rate, inclinaison, luminosity, ...) of incoming flux maps.
See the open issues for a list of proposed features (and known issues).
RUBIS
is still in stage of development.
If you have a suggestion to improve this repository, please fork the repo and create a pull request.
You can also simply open an issue with the tag "enhancement".
Any contribution is welcome and greatly appreciated!
- Fork the Project
- Create a Feature Branch (
git checkout -b feature-feature_description
) - Commit your Changes (
git commit -m 'Add some feature description'
) - Push to the Branch (
git push origin feature-feature_description
) - Open a Pull Request
Distributed under the GNU General Public License. See LICENSE.txt
for more information.
Pierre Houdayer - pierre.houdayer@obspm.fr
Project Link: https://github.com/pierrehoudayer/RUBIS
Find the A&A article here: https://ui.adsabs.harvard.edu/abs/2023arXiv230308077H/abstract
Bibtex entry :
@ARTICLE{2023A&A...675A.181H,
author = {{Houdayer}, P.~S. and {Reese}, D.~R.},
title = "{RUBIS: A simple tool for calculating the centrifugal deformation of stars and planets}",
journal = {\aap},
keywords = {stars: rotation, stars: interiors, planets and satellites: interiors, stars: oscillations, methods: numerical, Astrophysics - Solar and Stellar Astrophysics, Astrophysics - Earth and Planetary Astrophysics, Astrophysics - Instrumentation and Methods for Astrophysics},
year = 2023,
month = jul,
volume = {675},
eid = {A181},
pages = {A181},
doi = {10.1051/0004-6361/202346403},
archivePrefix = {arXiv},
eprint = {2303.08077},
primaryClass = {astro-ph.SR},
adsurl = {https://ui.adsabs.harvard.edu/abs/2023A&A...675A.181H},
adsnote = {Provided by the SAO/NASA Astrophysics Data System}
}