Source code for orbdot.models.theory

"""Theory
======
This module defines analytical models for investigating the source of long-term variations of
exoplanet orbits. These include equations for assessing the effects of tidal energy dissipation,
apsidal precession, systemic proper motion, and more.
"""

import numpy as np
import scipy.special as sci

# define constants
c = 2.99792458e8  # m/s
G = 6.6743e-11  # m^3 / kg / s^2

# define unit conversions
AU = 1.495978707e11  # 1 AU = 1.496 x 10^11 m
parsec = 3.0857e16  # 1 pc = 3.0857 x 10^16 m
R_earth = 6.371e6  # earth radius = 6,371.000 km
M_earth = 5.9722e24  # earth mass = 5.9722 x 10^24 kg
R_jup = 6.9911e7  # jupiter radius = 69911 km
M_jup = 1.89813e27  # jupiter mass = 1,898.13 x 10^24 kg
R_sun = 6.957e8  # solar radius = 695,700 km
M_sun = 1.9885e30  # solar mass = 1,988,500 x 10^30 kg


[docs] def companion_doppler_pdot_from_rv_trend(P, dvdt): """Calculate the apparent variation of an orbital period due to a line-of-sight acceleration. This method returns the time derivative of the observed orbital period of a transiting exoplanet due to the Doppler effect induced by an acceleration along the line-of-sight (i.e. a linear RV trend). It uses equation (6) from Bouma et al. (2020) [1]_, which is derived in convenient units. Parameters ---------- P : float The orbital period in days. dvdt : float The first-order radial acceleration term in m/s/day. Returns ------- float The apparent time derivative of the orbital period in milliseconds per year. Notes ----- When radial velocity data of a transiting exoplanet's host star show a linear trend, and assuming this trend is a real effect, the Doppler effect can cause the observed period between transits to vary. Equation (6) from Bouma et al. (2020) [1]_ expresses the expected period derivative in convenient units: .. math:: \\dot{P}_{\\mathrm{RV}} = 105.3 \\mathrm{\\,ms\\,yr^{-1}}\\left(\\frac{P}{\\mathrm{ day}}\\right)\\left(\\frac{\\dot{\\gamma}}{\\mathrm{m\\,s^{-1}\\,day^{-1}}}\\right) where :math:`P` is the orbital period of the transiting planet, :math:`\\dot{P}_{\\mathrm{ RV}}` is the time derivative of the period, and :math:`\\dot{\\gamma}` is the first-order RV acceleration term. References ---------- .. [1] :cite:t:`Bouma2020`. https://doi.org/10.3847/2041-8213/ab8563. """ return 105.3 * P * dvdt
[docs] def companion_doppler_rv_trend_from_pdot(P, dPdt): """Calculate the line-of-sight acceleration given the apparent variation of an orbital period. This method returns the slope of the linear radial velocity trend (acceleration) that corresponds to a given time derivative of the observed orbital period, assuming that the latter is a due to the Doppler effect. It uses equation (6) from Bouma et al. (2020) [1]_, which is derived in convenient units. Parameters ---------- P : float The orbital period in days. dPdt : float The apparent time derivative of the orbital period in milliseconds per year (ms/yr). Returns ------- float The first-order radial acceleration term in m/s/day. Notes ----- For a transiting exoplanet, if a variation in the observed period between transits is measured, it may be due to the Doppler effect induced by a line-of-sight acceleration. Assuming this trend is a real effect, Equation (6) from Bouma et al. (2020) [1]_ can be used to determine the corresponding slope of the linear radial velocity trend from the period derivative: .. math:: \\dot{\\gamma} = \\frac{\\dot{P}_{\\mathrm{RV}}}{105.3 \\mathrm{\\,ms\\,yr^{-1}}} \\left(\\frac{\\mathrm{day}}{P}\\right) where :math:`P` is the orbital period of the transiting planet, :math:`\\dot{P}_{\\mathrm{ RV}}` is the time derivative of the period, and :math:`\\dot{\\gamma}` is the first-order RV acceleration term. References ---------- .. [1] :cite:t:`Bouma2020`. https://doi.org/10.3847/2041-8213/ab8563. """ return dPdt / (105.3 * P)
[docs] def companion_mass_from_rv_trend(tau, dvdt, M_s): """Calculate the minimum mass of an outer companion given the slope of a linear RV trend. Given a measured linear trend in radial velocity data (acceleration), this method calculates the minimum mass of an unseen outer companion planet that could cause such an acceleration. While this approach was originally described in equation (1) of Feng et al. (2015) [1]_, this method implements a version given by equation (8) in Bouma et al. (2020) [2]_, which is derived in more convenient units. Parameters ---------- tau : float The time span of the observations in years. dvdt : float The first-order acceleration term in in m/s/day. M_s : float The mass of the host star in solar masses. Returns ------- float The companion planet mass in Earth masses. Notes ----- Given a measured acceleration in radial velocity measurements and the time span over which there are RV data available, this method calculates the minimum mass of an outer companion that could induce such an acceleration, using equation (8) from Bouma et al. (2020) [2]_: .. math:: M_{c} \\approx 5.99 M_{\\mathrm{Jup}}\\left(\\frac{\\tau}{\\mathrm{yr}}\\right)^{ 4/3}\\left|\\frac{\\dot{\\gamma}}{\\mathrm{m\\,s^{-1}\\,day^{-1}}}\\right|\\left( \\frac{M_{\\star}}{M_{\\odot}}\\right)^{2 / 3} where :math:`\\tau` is the time span of the observations in years, :math:`\\dot{\\gamma}` is the first-order acceleration term, and :math:`M_\\star` is the mass of the host star. The above equation assumes that the companion's orbit has an eccentricity of :math:`e=0.5`, an argument of pericenter that is exactly :math:`90` degrees, and a period that is :math:`1.25\\times` the time span of the observations. In this scenario, the observed linear trend manifests as a segment of the sawtooth-like RV curve of the companion planet (see Figure 1 of [1]_), for which the semi-amplitude can be approximated as half of the baseline multiplied by the acceleration, i.e. :math:`K_c = 0.5 \\, \\tau \\, \\dot{\\gamma}`. References ---------- .. [1] :cite:t:`Feng2015`. https://doi.org/10.1088/0004-637X/800/1/22 .. [2] :cite:t:`Bouma2020`. https://doi.org/10.3847/2041-8213/ab8563 """ # define the minimum possible companion period in years P_c = 1.25 * tau # calculate the corresponding semi major axis in AU a_c = get_semi_major_axis_from_period(1.25 * 365.25 * tau, M_s) / AU # calculate the companion RV semi-amplitude in m/s K_c = 0.5 * tau * 365.25 * dvdt # calculate the companion planet mass (Jupiter masses) M_c = 5.99 * tau ** (4 / 3) * np.abs(dvdt) * M_s ** (2 / 3) # convert to Earth masses M_c *= M_jup / M_earth return M_c, P_c, a_c, K_c
[docs] def companion_rv_trend_from_mass(tau, M_c, M_s): """Calculate the slope of a linear radial velocity trend given an estimate of a companion mass. This method computes the expected slope of a linear radial velocity trend (acceleration) induced by an unseen outer companion planet with an orbital period that is far longer than the observational baseline. It requires an estimate of the companion mass. While this approach was originally described in equation (1) of Feng et al. (2015) [1]_, this method implements a version given by equation (8) in Bouma et al. (2020) [ 2]_, which is derived in more convenient units. Parameters ---------- tau : float The time span of the observations in years. M_c : float The companion planet mass in earth masses. M_s : float The mass of the host star in solar masses. Returns ------- float The first-order acceleration term in in m/s/day. Notes ----- Given a companion planet's mass, the time span over which radial velocity measurements have been taken, and the mass of the host star, this method calculates the slope of the expected linear trend in the radial velocity data using equation (8) from Bouma et al. (2020) [2]_: .. math:: \\dot{\\gamma} \\approx \\frac{M_c}{5.99 M_{\\mathrm{Jup}}} \\left(\\frac{\\mathrm{yr}}{ \\tau}\\right)^{4/3} \\left(\\frac{M_{\\odot}}{M_{\\star}}\\right)^{2 / 3} \\mathrm{m\\, s^{-1}\\,day^{-1}} where :math:`\\tau` is the time span of the observations in years, :math:`M_c` is the mass of the companion planet, and :math:`M_\\star` is the mass of the host star. The above equation assumes that the companion's orbit has an eccentricity of :math:`e=0.5`, an argument of pericenter that is exactly :math:`90` degrees, and a period that is :math:`1.25\\times` the time span of the observations. In this scenario, the observed linear trend manifests as a segment of the sawtooth-like RV curve of the companion planet (see Figure 1 of [1]_), for which the semi-amplitude can be approximated as half of the baseline multiplied by the acceleration, i.e. :math:`K_c = 0.5 \\, \\tau \\, \\dot{\\gamma}`. References ---------- .. [1] :cite:t:`Feng2015`. https://doi.org/10.1088/0004-637X/800/1/22 .. [2] :cite:t:`Bouma2020`. https://doi.org/10.3847/2041-8213/ab8563 """ # convert earth masses to jupiter masses M_c *= M_earth / M_jup # calculate and return the RV slope in m/s/day return M_c / (5.99 * tau ** (4 / 3) * M_s ** (2 / 3))
[docs] def companion_from_quadratic_rv(P_min, t_pivot, dvdt, ddvdt, M_s): """Constrain properties of the orbit of an outer companion given a quadratic RV trend. Given the first and second-order acceleration terms that parameterize a long-term, quadratic radial velocity trend, :math:`\\dot{\\gamma}` and :math:`\\ddot{\\gamma}`, this method follows the derivations of Kipping et al. (2011) [1]_ (equations 1, 3, and 4) to constrain properties of a possible outer companion. This method requires an estimate of minimum possible orbital period of the companion, which is constrained by the timespan of the radial velocity measurements. This is determined by the :meth:`~orbdot.analysis.Analyzer.rv_trend_quadratic` method from the :class:`~orbdot.analysis.Analyzer` class. Parameters ---------- P_min : float The minimum orbital period of the companion in days. t_pivot : float The "pivot" point, defined as the reference mid-time of the transiting planet. dvdt : float First-order acceleration term in m/s/day. ddvdt : float Second-order acceleration term in m/s/day^2. M_s : float The mass of the host star in solar masses. Returns ------- tuple A tuple with the following elements: 1) the minimum companion period in days, 2) the corresponding lower limit on the RV semi-amplitude, 3) the companion mass in Earth masses, and 4) the time at which the companion RV signal is at a minimum. References ---------- .. [1] :cite:t:`Kipping2011`. https://doi.org/10.1088/0004-6256/142/3/95 """ # unit conversions M_s *= M_sun # solar masses to kg # calculate the lower limit of the RV semi-amplitude from the quadratic RV term in m/s K_min = np.abs(ddvdt) * P_min**2 / (4 * np.pi**2) # calculate the lower limit of the mass from K_min M_min = K_min * M_s ** (2 / 3) * ((P_min * 86400) / (2 * np.pi * G)) ** (1 / 3) # solve for the time when the outer companion RV signal is at a minimum tau_c = (-dvdt + ddvdt * t_pivot) / ddvdt - P_min / 4 return P_min, K_min, M_min / M_earth, tau_c
[docs] def companion_precession(P, M2, P2, M_s): """Calculates the rate of apsidal precession driven by a nonresonant planetary companion. This method returns the expected rate of apsidal precession due to a nonresonant companion planet, calculated using equation (8) from Heyl and Gladman (2007) [1]_. It is valid for a companion planet that is either interior or exterior to the observed (i.e. transiting) planet, but it assumes that the companion mass is much less than that of the host star. Parameters ---------- P : float The orbital period of the transiting planet in days. M2 : float The mass of the outer planet in Earth masses. P2 : float The orbital period of the outer planet in days. M_s : float The mass of the host star in solar masses. Returns ------- float The precession rate in radians per epoch. Notes ----- This method applies to the scenario in which a nonresonant companion in the system is driving apsidal precession of a transiting planet. It assumes that the companion object is on a circular, coplanar, and nonresonant orbit, and that its mass is far less than the host star. In the formulation of Heyl and Gladman (2007) [1]_, the induced precession rate in radians per orbit: .. math:: \\dot{\\omega} = \\frac{m_2}{M_\\star} \\frac{\\alpha}{(\\alpha+1)(\\alpha-1)^2}\\left[ \\left(\\alpha^2+1\\right) \\mathcal{E}\\left(\\frac{2 \\alpha^{1 / 2}}{\\alpha+1}\\right) -(\\alpha-1)^2 \\mathcal{K}\\left(\\frac{2 \\alpha^{1/2}}{\\alpha+1}\\right)\\right] where :math:`m_2` is the mass of the perturbing planet, :math:`\\alpha = a/a_2` is the semi major axis ratio, and :math:`\\mathcal{K}` and :math:`\\mathcal{E}` are the complete elliptic integrals of the first and second kind, respectively. .. important:: The ``scipy`` implementation of the complete elliptic integrals :math:`\\mathcal{K}` and :math:`\\mathcal{E}` expect an integrand of the form :math:`(1 - m \\sin^2 t)^{-1/2}`. However, the derivation of the equation above expects an integrand of the form :math:`(1 - m^2 \\sin^2 t)^{-1/2}` [1]_, which can be shown by expanding :math:`\\mathcal{K}` and :math:`\\mathcal{E}`. To maintain consistency, the input for the ``scipy.special.ellipk`` and ``scipy.special.ellipe`` methods must but squared, ie. such that :math:`\\mathcal{K}( 4 \\alpha/( \\alpha+1)^2)` and :math:`\\mathcal{E}(4 \\alpha/( \\alpha+1)^2)`. References ---------- .. [1] :cite:t:`Heyl2007`. https://doi.org/10.1111/j.1365-2966.2007.11697.x """ # derive parameters a1 = get_semi_major_axis_from_period(P, M_s) a2 = get_semi_major_axis_from_period(P2, M_s) # unit conversions M_s *= M_sun # solar masses to kg M2 *= M_earth # earth masses to kg # define the ratio of the semi major axes alpha = a1 / a2 # calculate the rate of apsidal precession t1 = alpha / ((alpha + 1) * (alpha - 1) ** 2) t2 = (alpha**2 + 1) * sci.ellipe(4 * alpha / (alpha + 1) ** 2) t3 = (alpha - 1) ** 2 * sci.ellipk(4 * alpha / (alpha + 1) ** 2) dwdE = (M2 / M_s) * t1 * (t2 - t3) # return the apsidal precession rate in radians per epoch (rad/E) return dwdE
[docs] def companion_mass_from_precession(P, P2, dwdE, M_s): """Calculate the mass of a nonresonant companion given a precession rate and orbital period. Given the orbital period of a nonresonant companion planet in the system, this method returns the necessary mass for it to be responsible for a measured apsidal precession rate of the studied orbit. It uses equation (8) from Heyl and Gladman (2007) [1]_, and is appropriate for a companion planet that is either interior or exterior to the observed (i.e., transiting) planet. However, it assumes that the companion mass is much less than that of the host star. Parameters ---------- P : float The orbital period of the transiting planet in days. P2 : float The orbital period of the outer planet in days. dwdE : float The rate of apsidal precession in radians per epoch. M_s : float The mass of the host star in solar masses. Returns ------- float The mass of the outer companion in earth masses. Notes ----- This method applies to the scenario in which a nonresonant companion in the system is driving apsidal precession of a transiting planet. It assumes that the companion object is on a circular, coplanar, and nonresonant orbit, and that its mass is far less than the host star. In the formulation of Heyl and Gladman (2007) [1]_, the mass of the companion planet can be determined from the measured precession rate by: .. math:: m_2 = \\dot{\\omega} \\frac{M_\\star (\\alpha+1)(\\alpha-1)^2}{\\alpha \\left[ \\left(\\alpha^2+1\\right) \\mathcal{E}\\left(\\frac{2 \\alpha^{1 / 2}}{\\alpha+1}\\right) - (\\alpha-1)^2 \\mathcal{K}\\left(\\frac{2 \\alpha^{1/2}}{\\alpha+1}\\right)\\right]} where :math:`\\dot{\\omega}` is the observed precession rate, :math:`\\alpha = a/a_2` is the semi major axis ratio, and :math:`\\mathcal{K}` and :math:`\\mathcal{E}` are the complete elliptic integrals of the first and second kind, respectively. .. important:: The ``scipy`` implementation of the complete elliptic integrals :math:`\\mathcal{K}` and :math:`\\mathcal{E}` expect an integrand of the form :math:`(1 - m \\sin^2 t)^{-1/2}`. However, the derivation of the equation above expects an integrand of the form :math:`(1 - m^2 \\sin^2 t)^{-1/2}` [1]_, which can be shown by expanding :math:`\\mathcal{K}` and :math:`\\mathcal{E}`. To maintain consistency, the input for the ``scipy.special.ellipk`` and ``scipy.special.ellipe`` methods must but squared, ie. such that :math:`\\mathcal{K}( 4 \\alpha/( \\alpha+1)^2)` and :math:`\\mathcal{E}(4 \\alpha/( \\alpha+1)^2)`. References ---------- .. [1] :cite:t:`Heyl2007`. https://doi.org/10.1111/j.1365-2966.2007.11697.x """ # derive parameters a1 = get_semi_major_axis_from_period(P, M_s) a2 = get_semi_major_axis_from_period(P2, M_s) # unit conversions M_s *= M_sun # solar masses to kg # define the ratio of the semi major axes alpha = a1 / a2 # calculate the mass of the companion planet t1 = alpha / ((alpha + 1) * (alpha - 1) ** 2) t2 = (alpha**2 + 1) * sci.ellipe(4 * alpha / (alpha + 1) ** 2) t3 = (alpha - 1) ** 2 * sci.ellipk(4 * alpha / (alpha + 1) ** 2) M2 = dwdE * M_s / (t1 * (t2 - t3)) # return the companion mass in Earth masses return M2 / M_earth
[docs] def decay_empirical_quality_factor(P_orb, P_rot_s): """Calculate the modified stellar tidal quality factor from an empirical law. This method calculates the tidal forcing period of a star-planet system and uses it to estimate the star's modified tidal quality factor with an empirical law derived by Penev et al. (2018) [1]_. Parameters ---------- P_orb : float Orbital period in days. P_rot_s : float Period of the host star's rotation in days. Returns ------- tuple The star's modified tidal quality factor and the tidal forcing period of the system in days. Notes ----- The "modified" tidal quality factor is a typical parameterization of the efficiency of tidal energy dissipation within a body. It is defined as: .. math:: Q_\\star^{'} = \\frac{3}{2} \\frac{Q_\\star}{k_{2,\\star}} where :math:`Q_\\star` is the tidal quality factor and :math:`k_{2,\\star}` is the second-order potential Love number. The theoretical upper limit is :math:`k_{2,\\star}=3/2`, that of a uniform density sphere, in which case :math:`Q_\\star^{'} = Q_\\star`. This method estimates a value for :math:`Q_\\star^{'}` using an empirical law derived by Penev et al. (2018) [1]_: .. math:: Q^{'}_\\star = \\max\\left(\\frac{10^6}{(P_{\\mathrm{tide}}/\\mathrm{day})^{3.1}}, \\, 10^5\\right) where :math:`P_{\\mathrm{tide}}` is the tidal forcing period of the system, defined as: .. math:: P_{\\mathrm{tide}} = \\frac{1}{2\\left(P_{\\mathrm{orb}}^{-1} - P_{\\mathrm{rot}}^{ -1}\\right)} where :math:`P_{\\mathrm{orb}}` is the planet's orbital period and :math:`P_{\\mathrm{rot}}` is the rotational period of the host star. References ---------- .. [1] :cite:t:`Penev2018`. https://doi.org/10.3847/1538-3881/aaaf71 """ # determine the tidal forcing period t1 = 1 / P_orb - 1 / P_rot_s P_tide = 1 / (2 * t1) # use the empirical law for Q' Q_star = 10**6.0 / P_tide**3.1 # return the tidal quality factor Q' return Q_star, P_tide
[docs] def decay_star_pdot_from_quality_factor( P, e, M_s, M_p, R_s, Q_star, epsilon_s=0.0, P_rot_s=None ): """Calculate the rate of orbital decay driven by equilibrium tides in the host star. This method computes the orbital decay rate for a given "modified" stellar tidal quality factor and, optionally, the host star’s rotation period and/or obliquity. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float Host star mass in solar masses. M_p : float Planet mass in earth masses. R_s : float Host star radius in solar radii. Q_star : float The host star's modified annual tidal quality factor. epsilon_s : float, optional Host star obliquity in degrees. Default is zero. P_rot_s : float, optional Host star rotation period in days. Returns ------- float Orbital decay rate in days per epoch. Notes ----- If equilibrium tides are the dominant driver of orbital evolution, the rate of orbital period decay due to stellar tides is given by: .. math:: \\frac{dP}{dt}_{\\mathrm{(star)}} = -\\frac{27\\pi}{Q_\\star^{'}} \\left(\\frac{M_p}{ M_\\star}\\right) \\left(\\frac{R_\\star}{a}\\right)^5 \\times \\left[f(e) - g(e) \\cos{ \\varepsilon_\\star} \\frac{\\dot{\\theta}_\\star}{n}\\right] where :math:`Q_\\star^{'}` is the modified annual tidal quality factor of the star, :math:`\\varepsilon_\\star` is the stellar obliquity (defined as the angle between the spin and orbital angular momentum vectors), :math:`\\dot{\\theta}_\\star` is the stellar spin frequency, :math:`n` is the orbital mean motion, :math:`a` is the semi major axis, :math:`M_p` is the planet mass, :math:`M_\\star` is the stellar mass, and :math:`R_\\star` is the stellar radius. The eccentricity-dependent functions :math:`f(e)` and :math:`g(e)` are defined in the :meth:`~orbdot.models.theory.decay_get_f_e` and :meth:`~orbdot.models.theory.decay_get_g_e` methods. This equation is derived using Equations (2), (5), and (19) from [1]_ and is valid for any value of :math:`\\dot{\\theta}_\\star`, :math:`\\varepsilon_\\star`, and :math:`e`. We see that if :math:`e=0` and :math:`\\varepsilon_\\star=0`, the expression simplifies to: .. math:: \\frac{dP}{dt}_{\\mathrm{(star)}} = -\\frac{27\\pi}{Q_\\star^{'}} \\left(\\frac{M_p}{ M_\\star}\\right) \\left(\\frac{R_\\star}{a}\\right)^5 \\left[1 - \\frac{\\dot{ \\theta}_\\star}{n}\\right] If the rotation rate of the star is unknown, it is assumed that :math:`\\dot{\\theta}_\\star \\ll n`, in which case the bracketed term approaches unity, leaving: .. math:: \\frac{dP}{dt}_{\\mathrm{(star)}} = -\\frac{27\\pi}{Q_\\star^{'}} \\left(\\frac{M_p}{ M_\\star}\\right) \\left(\\frac{R_\\star}{a}\\right)^5 References ---------- .. [1] :cite:t:`Leconte2010`. https://doi.org/10.1051/0004-6361/201014337 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) f_e = decay_get_f_e(e) g_e = decay_get_g_e(e) mean_motion = 2 * np.pi / P # calculate spin frequency if stellar rotation period is provided if P_rot_s is not None: spin_freq_s = (2 * np.pi) / P_rot_s beta = spin_freq_s / mean_motion # assume P_rot_s >> P_orb if stellar rotation period is not provided else: beta = 0 # as P_orb/P_rot_s --> 0 # unit conversions epsilon_s *= np.pi / 180 # deg to rad R_s *= R_sun # solar radii to m M_p *= M_earth # earth masses to kg M_s *= M_sun # solar masses to kg # calculate the predicted orbital decay rate (days/day) t1 = -27 * np.pi / Q_star * (M_p / M_s) * (R_s / a) ** 5 t2 = f_e - g_e * np.cos(epsilon_s) * beta pdot = t1 * t2 # return the orbital decay rate in days per epoch (days/E) return pdot * P
[docs] def decay_star_quality_factor_from_pdot( P, e, M_s, M_p, R_s, dPdE, epsilon_s=0.0, P_rot_s=None ): """Calculate the host star's modified stellar quality factor given an orbital decay rate. Given an orbital decay rate and, optionally, the host star's rotation period and obliquity, this method returns the host star's modified tidal quality factor under the assumption that the decay is driven by equilibrium tides in the host star. See the :meth:`~orbdot.models.theory.decay_star_pdot_from_quality_factor` docstring for a more detailed description of the relevant equations. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float Host star mass in solar masses. M_p : float Planet mass in earth masses. R_s : float Host star radius in solar radii. dPdE : float Orbital decay rate in days per epoch. epsilon_s : float, optional Host star obliquity in degrees. Default is zero. P_rot_s : float, optional Host star rotation period in days. Returns ------- float The host star's modified annual tidal quality factor. """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) f_e = decay_get_f_e(e) g_e = decay_get_g_e(e) mean_motion = 2 * np.pi / P # calculate spin frequency if stellar rotation period is provided if P_rot_s is not None: spin_freq_s = (2 * np.pi) / P_rot_s beta = spin_freq_s / mean_motion # assume P_rot_s >> P_orb if stellar rotation period is not provided else: beta = 0 # as P_orb/P_rot_s --> 0 # unit conversions epsilon_s *= np.pi / 180 # deg to rad R_s *= R_sun # solar radii to m M_p *= M_earth # earth masses to kg M_s *= M_sun # solar masses to kg dPdE *= 1 / P # days/E to days/day # compute and return the host star's modified annual tidal quality factor t1 = -27 * np.pi / dPdE * (M_p / M_s) * (R_s / a) ** 5 t2 = f_e - g_e * np.cos(epsilon_s) * beta return t1 * t2
[docs] def decay_planet_pdot_from_quality_factor( P, e, M_s, M_p, R_p, Q_planet, epsilon_p=0.0, P_rot_p=None ): """Calculate the rate of orbital decay driven by equilibrium tides in the planet. This method computes the orbital decay rate for a given "modified" planetary tidal quality factor and, optionally, the planet's rotation period and obliquity. This expression accounts for both eccentricity-driven and obliquity-driven tidal dissipation, providing a unified treatment of planetary tides. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float Host star mass in solar masses. M_p : float Planet mass in earth masses. R_p : float Planet radius in earth radii. Q_planet : float The planet's modified annual tidal quality factor. epsilon_p : float, optional Planetary obliquity in degrees. Default is zero. P_rot_p : float, optional Planet rotation period in days. Returns ------- float Orbital decay rate in days per epoch. Notes ----- If equilibrium tides are the dominant driver of orbital evolution, the rate of orbital period decay due to planetary tides is given by: .. math:: \\frac{dP}{dt}_{\\mathrm{(planet)}} = -\\frac{27\\pi}{Q_p^{'}} \\left(\\frac{M_\\star}{ M_p}\\right) \\left(\\frac{R_p}{a}\\right)^5 \\left[f(e) - g(e) \\cos{\\varepsilon_p} \\frac{\\dot{\\theta}_p}{n}\\right] where :math:`Q_p^{'}` is the modified annual tidal quality factor of the planet, :math:`\\varepsilon_p` is the planetary obliquity (defined as the angle between the planet's spin and orbital angular momentum vectors), :math:`\\dot{\\theta}_p` is the planet's spin frequency, :math:`n` is the orbital mean motion, :math:`a` is the semi major axis, :math:`M_p` is the planet mass, :math:`M_\\star` is the stellar mass, and :math:`R_p` is the planet's radius. The eccentricity-dependent functions :math:`f(e)` and :math:`g(e)` are defined in the :meth:`~orbdot.models.theory.decay_get_f_e` and :meth:`~orbdot.models.theory.decay_get_g_e` methods. This equation is derived using Equations (2), (5), and (19) from [1]_ and is valid for any value of :math:`\\dot{\\theta}_p`, :math:`\\varepsilon_p`, and :math:`e`. If a rotation period is not provided, the planet's spin frequency is calculated using the tidally-evolved equilibrium rate defined in :meth:`~orbdot.models.theory.decay_get_equilibrium_spin_freq`, which is dependent on the eccentricity and planetary obliquity. In this case, the above equation can be expressed as: .. math:: \\frac{dP}{dt}_{\\mathrm{(planet)}} = -\\frac{27\\pi}{Q_p^{'}} \\left(\\frac{M_\\star}{ M_p}\\right) \\left(\\frac{R_p}{a}\\right)^5 \\times \\left[f(e) - \\frac{g^2(e)}{h( e)}\\frac{2 \\cos^2{\\varepsilon_p}}{(1 + \\cos^2{\\varepsilon_p})}\\right] We see that if the orbit is circular and aligned, meaning :math:`e=0` and :math:`\\varepsilon_p=0`, the planetary contribution to orbital decay vanishes. In this case, the functions :math:`f(e)`, :math:`g(e)`, and :math:`h(e)` approach unity, and the equilibrium spin rate synchronizes with the mean motion, :math:`\\dot{\\theta}_p=n`, leading to no net tidal energy dissipation in the planet. """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) f_e = decay_get_f_e(e) g_e = decay_get_g_e(e) mean_motion = 2 * np.pi / P # calculate the planet spin frequency if P_rot_p is None: # if a rotation period is not provided spin_freq_p = decay_get_equilibrium_spin_freq(P, e, epsilon_p) else: # if a rotation period is given spin_freq_p = (2 * np.pi) / P_rot_p beta = spin_freq_p / mean_motion # unit conversions epsilon_p *= np.pi / 180 # deg to rad R_p *= R_earth # earth radii to m M_p *= M_earth # earth masses to kg M_s *= M_sun # solar masses to kg # calculate the predicted orbital decay rate (days/day) t1 = -27 * np.pi / Q_planet * (M_s / M_p) * (R_p / a) ** 5 t2 = f_e - g_e * np.cos(epsilon_p) * beta pdot = t1 * t2 # return the orbital decay rate in days per epoch (days/E) return pdot * P
[docs] def decay_planet_quality_factor_from_pdot( P, e, M_s, M_p, R_p, dPdE, epsilon_p=0.0, P_rot_p=None ): """Calculate the planet's modified planetary quality factor given an orbital decay rate. Given an orbital decay rate and, optionally, the planet's rotation period and obliquity, this method returns the planet's modified tidal quality factor under the assumption that the decay is driven by equilibrium tides in the planet. See the :meth:`~orbdot.models.theory.decay_planet_pdot_from_quality_factor` docstring for a more detailed description of the relevant equations. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float Host star mass in solar masses. M_p : float Planet mass in earth masses. R_p : float Planet radius in earth radii. dPdE : float Orbital decay rate in days per epoch. epsilon_p : float, optional Planetary obliquity in degrees. Default is zero. P_rot_p : float, optional Planet rotation period in days. Returns ------- float The planet's modified annual tidal quality factor. Notes ----- If a rotation period not provided for the planet, the planet's spin frequency is calculated with the :meth:`~orbdot.models.theory.decay_get_equilibrium_spin_freq` method. """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) f_e = decay_get_f_e(e) g_e = decay_get_g_e(e) mean_motion = 2 * np.pi / P # calculate the planet spin frequency if P_rot_p is None: # if a rotation period is not provided spin_freq_p = decay_get_equilibrium_spin_freq(P, e, epsilon_p) else: # if a rotation period is given spin_freq_p = (2 * np.pi) / P_rot_p beta = spin_freq_p / mean_motion # unit conversions epsilon_p *= np.pi / 180 # deg to rad R_p *= R_earth # earth radii to m M_p *= M_earth # earth masses to kg M_s *= M_sun # solar masses to kg dPdE *= 1 / P # days/E to days/day # compute and return the planet's modified annual tidal quality factor t1 = -27 * np.pi / dPdE * (M_s / M_p) * (R_p / a) ** 5 t2 = f_e - g_e * np.cos(epsilon_p) * beta return t1 * t2
[docs] def decay_get_f_e(e): """Computes the first eccentricity function for the equilibrium tide methods, f(e). This method compute one of three eccentricity-dependent functions called by the equilibrium tidal theory methods, given by Equation (4) in [1]_: .. math:: f(e) = \\frac{1+\\frac{31}{2} e^2+\\frac{255}{8} e^4+\\frac{185}{16} e^6+\\frac{25}{64} e^8}{(1-e^2)^{15/2}} Parameters ---------- e : float The orbit eccentricity. Returns ------- float The computed value of f(e). References ---------- .. [1] :cite:t:`Leconte2010`. https://doi.org/10.1051/0004-6361/201014337 """ t1 = 1 + (31 / 2) * e**2 + (255 / 8) * e**4 + (185 / 16) * e**6 + (25 / 64) * e**8 t2 = (1 - e**2) ** (15 / 2) return t1 / t2
[docs] def decay_get_g_e(e): """Computes the second eccentricity function for the equilibrium tide methods, g(e). This method compute one of three eccentricity-dependent functions called by the equilibrium tidal theory methods, given by Equation (3) in [1]_: .. math:: g(e) = \\frac{1+\\frac{15}{2} e^2+\\frac{45}{8} e^4+\\frac{5}{16} e^6}{(1-e^2)^6} Parameters ---------- e : float The orbit eccentricity. Returns ------- float The computed value of g(e). References ---------- .. [1] :cite:t:`Leconte2010`. https://doi.org/10.1051/0004-6361/201014337 """ t1 = 1 + (15 / 2) * e**2 + (45 / 8) * e**4 + (5 / 16) * e**6 t2 = (1 - e**2) ** 6 return t1 / t2
[docs] def decay_get_h_e(e): """Computes the third eccentricity function for the equilibrium tide methods, h(e). This method compute one of three eccentricity-dependent functions called by the equilibrium tidal theory methods, given by Equation (11) in [1]_: .. math:: h(e) = \\frac{1+3 e^2+\\frac{3}{8} e^4}{(1-e^2)^{9/2}} Parameters ---------- e : float The orbit eccentricity. Returns ------- float The calculates value of h(e). References ---------- .. [1] :cite:t:`Leconte2010`. https://doi.org/10.1051/0004-6361/201014337 """ t1 = 1 + 3 * e**2 + (3 / 8) * e**4 t2 = (1 - e**2) ** (9 / 2) return t1 / t2
[docs] def decay_get_equilibrium_spin_freq(P, e, epsilon_p): """Calculate the equilibrium spin frequency of a planet. This method returns the equilibrium spin frequency of a planet with nonzero obliquity and/or orbit eccentricity. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. epsilon_p : float The planetary obliquity. Returns ------- float Equilibrium spin frequency in days^{-1}. Notes ----- Planetary rotation periods are typically unconstrained, and it is not always appropriate to assume that they are tidally locked (synchronous rotation). When the orbit eccentricity and/or planetary obliquity are nonzero, the rotation state of the planet may be asynchronous. In this case, a reasonable assumption is that the planet's spin has tidally evolved toward an equilibrium rate, given by Equation (12) of [1]_: .. math:: \\dot{\\theta}_{eq} \\equiv n \\frac{g(e)}{h(e)}\\frac{2\\cos{\\varepsilon_p}}{(1 + \\cos^2{\\varepsilon_p})} where :math:`\\varepsilon_p` is the planetary obliquity (defined as the angle between the planet's spin and orbital angular momentum vectors), :math:`n` is the orbital mean motion, and :math:`g(e)` and :math:`h(e)` are eccentricity-dependent functions that are defined in :meth:`~orbdot.models.theory.decay_get_g_e` and :meth:`~orbdot.models.theory.decay_get_h_e`. References ---------- .. [1] :cite:t:`Leconte2010`. https://doi.org/10.1051/0004-6361/201014337 """ # derive parameters g_e = decay_get_g_e(e) h_e = decay_get_h_e(e) # unit conversions epsilon_p *= np.pi / 180 # deg to rad t1 = (2 * np.pi / P) * (g_e / h_e) t2 = 2 * np.cos(epsilon_p) / (1 + np.cos(epsilon_p) ** 2) return t1 * t2
[docs] def decay_timescale(P, dPdE): """Calculate the remaining lifetime of a planet on a decaying orbit. Parameters ---------- P : float Orbital period in days. dPdE : float Orbital decay rate in days per epoch. Returns ------- float The remaining lifetime in millions of years (Myr). Notes ----- For a planet on a decaying orbit, its remaining lifetime can be estimated as simply: .. math:: \\tau = \\frac{P_0}{|\\dot{P}_{\\mathrm{decay}}|} where :math:`\\dot{P}_{\\mathrm{decay}}` is the orbital decay rate, and :math:`P_0` is the observed orbital period. """ # convert days/orbit to days/yr dPdt = dPdE / P * 365.25 # calculate the remaining lifetime tau = P / np.abs(dPdt) # return the remaining lifetime in Myr return tau / 1e6
[docs] def decay_energy_loss(P, dPdE, M_s, M_p): """Calculate the rate of orbit energy loss due to tidal forces causing orbital decay. This function uses Equation (9) from Yee et al. (2020) [1]_ to compute the rate at which the energy of a planetary orbit decreases as it decays due to tidal forces. Parameters ---------- P : float Orbital period in days. dPdE : float Orbital decay rate in days per epoch. M_s : float Mass of the host star in solar masses. M_p : float Mass of the planet in earth masses. Returns ------- float The rate of orbital energy loss in watts. Notes ----- As a planet's orbit decays, the orbital energy and angular momentum will decrease over time. Equation (9) of Yee et al. (2020) [1]_ defines the rate of orbital energy loss as: .. math:: \\frac{dE}{dt} = \\frac{(2\\pi)^{2/3} M_{\\mathrm{p}}}{3} \\left(\\frac{G M_{\\star}}{ P}\\right)^{2/3} \\frac{1}{P} \\frac{dP}{dt} where :math:`G` is the gravitational constant, :math:`M_{\\star}` is the host star mass, :math:`M_{\\mathrm{p}}` is the planet mass, :math:`P` is the orbital period, and :math:`\\frac{dP}{dt}` is the orbital decay rate. References ---------- .. [1] :cite:t:`Yee2020`. https://doi.org/10.3847/2041-8213/ab5c16 """ # unit conversions M_p *= M_earth # earth masses to kg M_s *= M_sun # solar masses to kg dPdt = dPdE / P # days/E to days/day P *= 86400 # days to seconds # calculate the time derivative of the orbital energy t1 = (2 * np.pi) ** (2 / 3) * M_p / 3 t2 = (G * M_s / P) ** (2 / 3) t3 = (1 / P) * dPdt dEdt = t1 * t2 * t3 # return the rate of energy loss in Watts return dEdt
[docs] def decay_angular_momentum_loss(P, dPdE, M_s, M_p): """Calculate the rate of angular momentum loss due to tidal forces causing orbital decay. This function uses Equation (10) from Yee et al. (2020) [1]_ to compute the rate at which the angular momentum of a planetary orbit decreases as it decays due to tidal forces. Parameters ---------- P : float Orbital period in days. dPdE : float Orbital decay rate in days per epoch. M_s : float Host star mass in solar masses. M_p : float Planet mass in earth masses. Returns ------- float The rate of orbital angular momentum loss in kg m^2 / s^2. Notes ----- As a planet's orbit decays, the orbital energy and angular momentum will decrease over time. Equation (10) of Yee et al. (2020) [1]_ defines the rate of orbital angular momentum loss as: .. math:: \\frac{dL}{dt} = \\frac{M_{\\mathrm{p}}}{3(2\\pi)^{1/3}} \\left(\\frac{G M_{\\star}}{ P}\\right)^{2/3} \\frac{dP}{dt} where :math:`G` is the gravitational constant, :math:`M_{\\star}` is the host star mass, :math:`M_{\\mathrm{p}}` is the planet mass, :math:`P` is the orbital period, and :math:`\\frac{dP}{dt}` is the orbital decay rate. References ---------- .. [1] :cite:t:`Yee2020`. https://doi.org/10.3847/2041-8213/ab5c16 """ # unit conversions M_p *= M_earth # earth masses to kg M_s *= M_sun # solar masses to kg dPdt = dPdE / P # days/E to days/day P *= 86400 # days to seconds # calculate the time derivative of the orbit angular momentum t1 = M_p / (3 * (2 * np.pi) ** (1 / 3)) t2 = (G * M_s / P) ** (2 / 3) dLdt = t1 * t2 * dPdt # return the rate of angular momentum loss in kg m^2 / s^2 return dLdt
[docs] def precession_gr(P, e, M_s): """Calculate the rate of apsidal precession predicted by general relativity. This method returns the expected apsidal precession rate of the planet's orbit due to general relativistic (GR) effects, using Equation (12) from Ragozzine and Wolf (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float The host star mass in solar masses. Returns ------- float The precession rate in radians per epoch. Notes ----- This method applies the lowest order of the relativistic contribution to apsidal precession, given in Equation (12) of Ragozzine and Wolf (2009) [1]_ as: .. math:: \\dot{\\omega}_{\\mathrm{GR}} = \\frac{3 G M_{\\star} n}{a c^2(1 - e^2)} where :math:`G` is the gravitational constant, :math:`c` is the speed of light in a vacuum, :math:`M_{\\star}` is the host star mass, :math:`a` is the planet's semi major axis, :math:`e` is the eccentricity, and :math:`n = 2\\pi/P` is the mean motion. References ---------- .. [1] :cite:t:`Ragozzine2009`. https://doi.org/10.1088/0004-637X/698/2/1778 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) nu = 2 * np.pi / P # unit conversions M_s *= M_sun # solar masses to kg nu *= 1 / 86400 # 1/days to 1/s # calculate the precession rate wdot = 3 * G * M_s * nu / (a * c**2 * (1 - e**2)) # convert from rad/s to rad/E wdot *= 86400 * P # return the apsidal precession rate in radians per epoch return wdot
[docs] def precession_rotational_planet(P, e, M_s, M_p, R_p, k2_p, P_rot_p): """Calculates the rate of apsidal precession driven by planetary rotation. This method returns the expected apsidal precession rate of a planet's orbit due to its rotational bulge, calculated using Equations (10) and (11) from Ragozzine and Wolf (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float The host star mass in solar masses. M_p : float The planet mass in earth masses. R_p : float The planet radius in earth radii. k2_p : float The planet's Love number. P_rot_p : float The period of the planet's rotation in days. Returns ------- float The precession rate in radians per epoch. Notes ----- The rotation of a fluid body raises an oblate distortion (i.e., flattening) that perturbs the gravitational potential, driving apsidal precession. The rate of the apsidal precession induced by a planet's rotational bulge is given in Equation (10) of Ragozzine and Wolf (2009) [1]_ as: .. math:: \\dot{\\omega}_{\\mathrm{rot,p}} = \\frac{n {R_p}^5 k_{2,p} {\\dot{\\theta}_p}^2}{2 a^2 G M_p} g_2(e) where :math:`\\dot{\\theta}_p` is the rotational velocity of the planet, :math:`k_{2,p}` is its second-order potential Love number, :math:`G` is the gravitational constant, :math:`n = 2\\pi/P` is the orbit mean motion, :math:`a` is the semi major axis, and :math:`M_p` and :math:`R_p` are the planet mass and radius, respectively. The function :math:`g_2(e)` represents an expansion in eccentricity :math:`e`, defined in Equation (11) of Ragozzine and Wolf (2009) [1]_ as: .. math:: g_2(e) = (1-e^2)^{-2} References ---------- .. [1] :cite:t:`Ragozzine2009`. https://doi.org/10.1088/0004-637X/698/2/1778 .. [2] :cite:t:`Lissauer2019`. https://doi.org/10.1017/9781108304061 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) nu = 2 * np.pi / P # calculate the eccentricity expansion and rotational velocity g = (1 - e**2) ** (-2) v_p = 2 * np.pi / P_rot_p # rad/day # unit conversions M_p *= M_earth # earth masses to kg R_p *= R_earth # earth radii to m nu *= 1 / 86400 # 1/days to 1/s v_p *= 1 / 86400 # rad/day to rad/s # calculate precession rate in rad/E wdot = (k2_p / 2) * (R_p / a) ** 5 * (v_p**2 * a**3 / (G * M_p)) * g * nu # rad/s wdot *= 86400 * P # convert rad/s to rad/E return wdot # rad/E
[docs] def precession_rotational_planet_k2(P, e, M_s, M_p, R_p, P_rot_p, dwdE): """Calculate the Love number given a rate of apsidal precession driven by planetary rotation. Given an apsidal precession rate for a planetary orbit, this method returns the second-order potential Love number of the planet :math:`k_{2,p}` under the assumption that the precession is driven entirely by the rotational bulge of the planet. See the :meth:`~orbdot.models.theory.precession_rotational_planet` docstring for a more detailed description of the relevant equations, which are adopted from Ragozzine and Wolf (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float The host star mass in solar masses. M_p : float The planet mass in earth masses. R_p : float The planet radius in earth radii. P_rot_p : float The period of the planet's rotation in days. dwdE : float The precession rate in radians per epoch. Returns ------- float The planet's potential Love number of the second order. References ---------- .. [1] :cite:t:`Ragozzine2009`. https://doi.org/10.1088/0004-637X/698/2/1778 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) nu = 2 * np.pi / P # calculate the eccentricity expansion and rotational velocity g = (1 - e**2) ** (-2) v_p = 2 * np.pi / P_rot_p # rad/day # unit conversions M_p *= M_earth # earth masses to kg R_p *= R_earth # earth radii to m nu *= 1 / 86400 # 1/days to 1/s v_p *= 1 / 86400 # rad/day to rad/s dwdt = dwdE / (86400 * P) # rad/E to rad/s # compute and return the Love number t1 = (1 / 2) * (R_p / a) ** 5 * (v_p**2 * a**3 / (G * M_p)) * g * nu k2_p = dwdt * t1 ** (-1) return k2_p
[docs] def precession_rotational_star(P, e, M_s, R_s, k2_s, P_rot_s): """Calculate the rate of apsidal precession driven by stellar rotation. This method returns the expected apsidal precession rate of a planet's orbit due to the rotational bulge of the host star, calculated using Equations (10) and (11) from Ragozzine and Wolf (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float The host star mass in solar masses. R_s : float The host star radius in solar radii. k2_s : float The host star's Love number. P_rot_s : float The period of the host star's rotation in days. Returns ------- float The precession rate in radians per epoch. Notes ----- The rotation of a fluid body raises an oblate distortion (i.e., flattening) that perturbs the gravitational potential, driving apsidal precession. The rate of the apsidal precession induced by a star's rotational bulge is given in Equation (10) of Ragozzine and Wolf (2009) [1]_ as: .. math:: \\dot{\\omega}_{\\mathrm{rot,\\star}} = \\frac{n {R_\\star}^5 k_{2,\\star} {\\dot{ \\theta}_\\star}^2}{2 a^2 G M_\\star} g_2(e) where :math:`\\dot{\\theta}_\\star` is the rotational velocity of the star, :math:`k_{2, \\star}` is its second-order potential Love number, :math:`G` is the gravitational constant, :math:`a` is the semi major axis of the orbit, :math:`n = 2\\pi/P` is the mean motion, and :math:`M_\\star` and :math:`R_\\star` are the stellar mass and radius, respectively. The parameter :math:`g_2(e)` represents an expansion in eccentricity :math:`e`, defined in Equation (11) of Ragozzine and Wolf (2009) [1]_ as: .. math:: g_2(e) = (1-e^2)^{-2} References ---------- .. [1] :cite:t:`Ragozzine2009`. https://doi.org/10.1088/0004-637X/698/2/1778 .. [2] :cite:t:`Lissauer2019`. https://doi.org/10.1017/9781108304061 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) nu = 2 * np.pi / P # calculate the eccentricity expansion and rotational velocity g = (1 - e**2) ** (-2) v_s = 2 * np.pi / P_rot_s # rad/day # unit conversions M_s *= M_sun # solar masses to kg R_s *= R_sun # solar radii to m nu *= 1 / 86400 # 1/days to 1/s v_s *= 1 / 86400 # rad/day to rad/s # calculate precession rate wdot = (k2_s / 2) * (R_s / a) ** 5 * (v_s**2 * a**3 / (G * M_s)) * g * nu wdot *= 86400 * P # convert rad/s to rad/E return wdot # rad/E
[docs] def precession_rotational_star_k2(P, e, M_s, R_s, P_rot_s, dwdE): """Calculate the Love number given a rate of apsidal precession driven by stellar rotation. Given an apsidal precession rate for a planetary orbit, this method returns the second-order potential Love number of the host star :math:`k_{2,\\star}` under the assumption that the precession is driven entirely by the rotational bulge of the host star. See the :meth:`~orbdot.models.theory.precession_rotational_star` docstring for a more detailed description of the relevant equations, which are adopted from Ragozzine and Wolf (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float The host star mass in solar masses. R_s : float The host star radius in solar radii. P_rot_s : float The period of the host star's rotation in days. dwdE : float The precession rate in radians per epoch. Returns ------- float The host star's potential Love number of the second order. References ---------- .. [1] :cite:t:`Ragozzine2009`. https://doi.org/10.1088/0004-637X/698/2/1778 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) nu = 2 * np.pi / P # calculate the eccentricity expansion and rotational velocity g = (1 - e**2) ** (-2) v_s = 2 * np.pi / P_rot_s # rad/day # unit conversions M_s *= M_sun # solar masses to kg R_s *= R_sun # solar radii to m nu *= 1 / 86400 # 1/days to 1/s v_s *= 1 / 86400 # rad/day to rad/s dwdt = dwdE / (86400 * P) # rad/E to rad/s # compute and return the Love number t1 = (1 / 2) * (R_s / a) ** 5 * (v_s**2 * a**3 / (G * M_s)) * g * nu k2_s = dwdt * t1 ** (-1) return k2_s
[docs] def precession_tidal_planet(P, e, M_s, M_p, R_p, k2_p): """Calculate the rate of apsidal precession driven by the planet's tidal bulge. This method returns the expected apsidal precession rate of the planet's orbit due to its tidal bulge, calculated using Equations (6) and (7) from Ragozzine and Wolf (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float The host star mass in solar masses. M_p : float The planet mass in earth masses. R_p : float The planet radius in earth radii. k2_p : float The planet's Love number. Returns ------- float The precession rate in radians per epoch. Notes ----- The host star of a massive, close-in planet may raise a significant "tidal bulge" on the planet, an ellipsoidal distortion caused by the varying gravitational force experienced across the extent of the body, driving apsidal precession. The rate of apsidal precession that is induced by the planet's tidal bulge is given in Equation (6) of Ragozzine and Wolf (2009) [1]_ as: .. math:: \\dot{\\omega}_{\\mathrm{tide,p}} = \\frac{15}{2}k_{2,p} n \\left(\\frac{R_p}{ a}\\right)^5\\left(\\frac{M_\\star}{M_p}\\right)f_2(e) where :math:`k_{2,p}` is the planet's second-order potential Love number, :math:`n = 2\\pi/P` is the orbit mean motion, :math:`a` is the semi major axis, :math:`M_\\star` is the host star mass, and :math:`M_p` and :math:`R_p` are the planet mass and radius. The parameter :math:`f_2(e)` represents an expansion in eccentricity :math:`e`, defined in Equation (7) of Ragozzine and Wolf (2009) [1]_ as: .. math:: f_2(e)= (1-e^2)^{-5} \\left(1 + \\frac{3}{2}e^2 + \\frac{1}{8}e^4 \\right). References ---------- .. [1] :cite:t:`Ragozzine2009`. https://doi.org/10.1088/0004-637X/698/2/1778 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) nu = 2 * np.pi / P # calculate the eccentricity expansion f = (1 - e**2) ** (-5) * (1 + (3 / 2) * e**2 + (1 / 8) * e**4) # unit conversions M_p *= M_earth # earth masses to kg M_s *= M_sun # solar masses to kg R_p *= R_earth # earth radii to m nu *= 1 / 86400 # 1/days to 1/s # calculate precession rate in rad/E wdot = (15 / 2) * k2_p * nu * f * (R_p / a) ** 5 * (M_s / M_p) # rad/s wdot *= 86400 * P # convert rad/s to rad/E return wdot # rad/E
[docs] def precession_tidal_planet_k2(P, e, M_s, M_p, R_p, dwdE): """Calculate a planet's Love number given a rate of apsidal precession driven by tides. Given an apsidal precession rate for a planetary orbit, this method returns the second-order potential Love number of the planet :math:`k_{2,p}` under the assumption that the precession is driven entirely by the tidal bulge of the planet. See the :meth:`~orbdot.models.theory.precession_tidal_planet` docstring for a more detailed description of the relevant equations, which are adopted from Ragozzine and Wolf (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float The host star mass in solar masses. M_p : float The planet mass in earth masses. R_p : float The planet radius in earth radii. dwdE : float The precession rate in radians per epoch. Returns ------- float The planet's potential Love number of the second order. References ---------- .. [1] :cite:t:`Ragozzine2009`. https://doi.org/10.1088/0004-637X/698/2/1778 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) nu = 2 * np.pi / P # calculate the eccentricity expansion f = (1 - e**2) ** (-5) * (1 + (3 / 2) * e**2 + (1 / 8) * e**4) # unit conversions M_p *= M_earth # earth masses to kg M_s *= M_sun # solar masses to kg R_p *= R_earth # earth radii to m nu *= 1 / 86400 # 1/days to 1/s dwdt = dwdE / (86400 * P) # convert rad/E to rad/s # compute and return the Love number t1 = (15 / 2) * nu * f * (R_p / a) ** 5 * (M_s / M_p) # rad/s k2_p = dwdt * t1 ** (-1) return k2_p
[docs] def precession_tidal_star(P, e, M_s, M_p, R_s, k2_s): """Calculate the rate of apsidal precession driven by the star's tidal bulge. This method returns the expected apsidal precession rate of the planet's orbit due to the star's tidal bulge, calculated using Equations (6) and (7) from Ragozzine and Wolf (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float The host star mass in solar masses. M_p : float The planet mass in earth masses. R_s : float The host star radius in solar radii. k2_s : float The host star's Love number. Returns ------- float The precession rate in radians per epoch. Notes ----- A massive, close-in planet may raise a significant "tidal bulge" on the host star, an ellipsoidal distortion caused by the varying gravitational force experienced across the extent of the body, driving apsidal precession. The rate of apsidal precession that is induced by the star's tidal bulge is given in Equation (6) of Ragozzine and Wolf (2009) [1]_ as: .. math:: \\dot{\\omega}_{\\mathrm{tide,\\star}} = \\frac{15}{2}k_{2,\\star} n \\left(\\frac{ R_\\star}{a}\\right)^5\\left(\\frac{M_p}{M_\\star}\\right)f_2(e) where :math:`k_{2,\\star}` is the star's second-order potential Love number, :math:`n = 2\\pi/P` is the orbit mean motion, :math:`a` is the semi major axis, :math:`M_p` is the planet mass, and :math:`M_\\star` and :math:`R_\\star` are the stellar mass and radius. The parameter :math:`f_2(e)` represents an expansion in eccentricity :math:`e`, defined in Equation (7) of Ragozzine and Wolf (2009) [1]_ as: .. math:: f_2(e)= (1-e^2)^{-5} \\left(1 + \\frac{3}{2}e^2 + \\frac{1}{8}e^4 \\right). References ---------- .. [1] :cite:t:`Ragozzine2009`. https://doi.org/10.1088/0004-637X/698/2/1778 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) nu = 2 * np.pi / P # calculate the eccentricity expansion f = (1 - e**2) ** (-5) * (1 + (3 / 2) * e**2 + (1 / 8) * e**4) # unit conversions M_p *= M_earth # earth masses to kg M_s *= M_sun # solar masses to kg R_s *= R_sun # solar radii to m nu *= 1 / 86400 # 1/days to 1/s # calculate precession rate in rad/E wdot = (15 / 2) * k2_s * nu * f * (R_s / a) ** 5 * (M_p / M_s) # rad/s wdot *= 86400 * P # convert rad/s to rad/E return wdot # rad/E
[docs] def precession_tidal_star_k2(P, e, M_s, M_p, R_s, dwdE): """Calculate a host star's Love number given a rate of apsidal precession driven by tides. Given an apsidal precession rate for a planetary orbit, this method returns the second-order potential Love number of the host star :math:`k_{2,\\star}` under the assumption that the precession is driven entirely by the tidal bulge of the star. See the :meth:`~orbdot.models.theory.precession_tidal_star` docstring for a more detailed description of the relevant equations, which are adopted from Ragozzine and Wolf (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. M_s : float The host star mass in solar masses. M_p : float The planet mass in earth masses. R_s : float The host star radius in solar radii. dwdE : float The precession rate in radians per epoch. Returns ------- float The host star's potential Love number of the second order. References ---------- .. [1] :cite:t:`Ragozzine2009`. https://doi.org/10.1088/0004-637X/698/2/1778 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) nu = 2 * np.pi / P # calculate the eccentricity expansion f = (1 - e**2) ** (-5) * (1 + (3 / 2) * e**2 + (1 / 8) * e**4) # unit conversions M_p *= M_earth # earth masses to kg M_s *= M_sun # solar masses to kg R_s *= R_sun # solar radii to m nu *= 1 / 86400 # 1/days to 1/s dwdt = dwdE / (86400 * P) # rad/E to rad/s # compute and return the Love number t1 = (15 / 2) * nu * f * (R_s / a) ** 5 * (M_p / M_s) k2_s = dwdt * t1 ** (-1) return k2_s
[docs] def get_pdot_from_wdot(P, e, w, dwdE): """Calculate the apparent time derivative of the orbital period due to apsidal precession. This method determines the apparent time derivative of the orbital period due to apsidal precession in general, independent of the physical mechanism. It uses Equation (17) from Rafikov (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. w : float The argument of pericenter in radians. dwdE : float The precession rate in radians per epoch. Returns ------- float The time derivative of the orbital period in milliseconds per year (ms/yr). Notes ----- For transiting exoplanets, apsidal precession causes the sidereal period, i.e. the length of time between transits, to vary over time. Because apsidal precession periods are typically far greater than observational baselines, this effect yields an *apparent* time derivative :math:`\\dot{P}`. Equation (17) of Rafikov (2009) [1]_ expresses this as: .. math:: \\dot{P} = \\frac{4\\pi\\left(\\dot{\\omega}\\right)^2 e\\cos\\omega_p}{ n^2}\\frac{\\left(1-e^2\\right)^{3/2}}{(1+e\\sin\\omega_p)^3} where :math:`e` is the orbit eccentricity, :math:`\\omega_p` is the argument of pericenter, :math:`\\dot{\\omega}` is the apsidal precession rate, and :math:`n = 2\\pi/P` is the orbit mean motion. References ---------- .. [1] :cite:t:`Rafikov2009`. https://doi.org/10.1088/0004-637X/700/2/965 """ # derive parameters nu = 2 * np.pi / P # unit conversions dwdt = dwdE * (1 / P) * 365.25 # rad/E to rad/yr # calculate the apparent period derivative t1 = e * np.cos(w) * (1 - e**2) ** (3 / 2) / (1 + e * np.sin(w)) ** 3 t2 = 4 * np.pi * (dwdt / nu) ** 2 pdot = t1 * t2 # days^2/yr^2 # return the apparent period derivative in ms/yr return pdot * (1 / 365.25) * 8.64e7
[docs] def get_tdot_from_wdot(P, e, w, i, T, dwdE, M_s, R_s): """Calculate the expected transit duration variation (TDV) due to apsidal precession. This method determines the expected TDV signal for apsidal precession in general, independent of the physical mechanism. It uses Equation (9) from Rafikov (2009) [1]_, assuming there is no change in the line-of-sight inclination over time. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. w : float The argument of pericenter in radians. i : float The line-of-sight inclination in degrees. T : float The transit duration in minutes. dwdE : float The precession rate in radians per epoch. M_s : float The host star mass in solar masses. R_s : float The host star radius in solar radii. Returns ------- float The time derivative of the transit duration in milliseconds per year (ms/yr). Notes ----- For transiting exoplanets, apsidal precession causes the duration of the transits :math:`T` to vary over time. Assuming that the line-of-sight inclination :math:`i` of the orbit is constant, the expected transit duration variation (TDV) is given by [1]_: .. math:: \\dot{T} = -\\frac{T}{1 + e\\sin{\\omega_p}} \\left[e \\, \\dot{\\omega}\\cos{\\omega_p} - g \\, \\dot{\\omega}\\cos{i}\\frac{e\\cos{\\omega_p}}{1+e\\sin{\\omega_p}}\\right] where, .. math:: g = \\frac{a}{R_\\star}\\frac{b}{1 - b^2} and where :math:`e` is the orbit eccentricity, :math:`\\omega_p` is the argument of pericenter, :math:`\\dot{\\omega}` is the apsidal precession rate, :math:`b` is the impact parameter, :math:`a` is the semi major axis, and :math:`R_\\star` is the radius of the host star. References ---------- .. [1] :cite:t:`Rafikov2009`. https://doi.org/10.1088/0004-637X/700/2/965 """ # unit conversions i *= np.pi / 180 # degrees to radians R_s *= R_sun # solar radii to m T *= 60 * 1000 # minutes to ms wdot = dwdE * (1 / P) * 365.25 # rad/E to rad/yr # derive parameters a = get_semi_major_axis_from_period(P, M_s) b = (a / R_s) * np.cos(i) * (1 - e**2) / (1 + e * np.sin(w)) g = (a / R_s) * b / (1 - b**2) # calculate the transit duration variation t1 = -T / (1 + e * np.sin(w)) t2 = e * wdot * np.cos(w) t3 = -g * (wdot * np.cos(i) * e * np.cos(w) / (1 + e * np.sin(w))) # return the TDV rate in ms/yr return t1 * (t2 + t3)
[docs] def proper_motion_idot(mu, beta): """Calculate the apparent rate of change of the inclination due to systemic proper motion. This method returns the rate of the variation of the line-of-sight inclination of an orbit due to systemic proper motion, using Equation (3) from Rafikov (2009) [1]_. Parameters ---------- mu : float The proper motion of the system in mas/yr (milliarcseconds per year). beta : float The angle in radians between the proper motion vector and the angular momentum vector projected onto the sky-plane. Returns ------- float The apparent time derivative of the line-of-sight inclination in radians per year. Notes ----- The proper motion of a star-planet system alters its appearance on the sky-plane, which causes an evolution of the line-of-sight inclination :math:`\\dot{i}_\\mu`. Rafikov (2009) [1]_ derives a simple expression for this effect in terms of the magnitude of the proper motion vector, :math:`\\mu = |\\vec{\\mu}|`: .. math:: \\dot{i}_\\mu = -\\mu\\cos{\\beta} where :math:`\\beta` is defined as the angle in the sky-plane between the proper motion vector :math:`\\vec{\\mu}` and the projection of the orbital angular momentum vector :math:`\\vec{L}`. Thus, the magnitude of the inclination variation :math:`|\\dot{i}_\\mu|` is maximized when :math:`\\beta = 0^\\circ, 180^\\circ`. References ---------- .. [1] :cite:t:`Rafikov2009`. https://doi.org/10.1088/0004-637X/700/2/965 """ # unit conversions beta *= np.pi / 180 # degrees to radians mu *= 1 / (1000 * 206265) # mas/yr to rad/yr # return the time derivative of the inclination in rad/yr return -mu * np.cos(beta)
[docs] def proper_motion_wdot(mu, i, beta): """Calculate the rate of the apparent apsidal precession due to systemic proper motion. This method returns the rate of the apparent apsidal precession induced by systemic proper motion, using Equation (4) from Rafikov (2009) [1]_. Parameters ---------- mu : float The proper motion of the system in mas/yr (milliarcseconds per year). i : float The line-of-sight inclination of the orbit in degrees. beta : float The angle in radians between the proper motion vector and the angular momentum vector projected onto the sky-plane. Returns ------- float The apparent apsidal precession rate in radians per year. Notes ----- The proper motion of a star-planet system alters its appearance on the sky-plane, which causes an apparent precession of the argument of pericenter :math:`\\dot{ \\omega}_\\mu`. Rafikov (2009) [1]_ derives a simple expression for this effect in terms of the magnitude of the proper motion vector, :math:`\\mu = |\\vec{\\mu}|`: .. math:: \\dot{\\omega}_\\mu = -\\mu\\frac{\\sin{\\beta}}{\\sin{i}} where :math:`\\beta` is defined as the angle in the sky-plane between the proper motion vector :math:`\\vec{\\mu}` and the projection of the orbital angular momentum vector :math:`\\vec{L}`. The magnitude of the apparent precession rate :math:`|\\dot{\\omega}_\\mu|` is maximized when :math:`\\beta = 90^\\circ, 270^\\circ`. References ---------- .. [1] :cite:t:`Rafikov2009`. https://doi.org/10.1088/0004-637X/700/2/965 """ # unit conversions beta *= np.pi / 180 # degrees to radians i *= np.pi / 180 # degrees to radians mu *= 1 / (1000 * 206265) # mas/yr to rad/yr # return the apparent precession rate in rad/yr return -mu * np.sin(beta) / np.sin(i)
[docs] def proper_motion_pdot(P, e, w, mu): """Calculate the apparent time derivative of the orbital period due to systemic proper motion. This method returns the apparent time derivative of the orbital period that is expected as a result of the systemic proper motion-induced (apparent) apsidal precession. It uses Equation (15) from Rafikov (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. w : float The argument of pericenter in radians. mu : float The proper motion of the system in milliarcseconds per year (mas/yr). Returns ------- float The apparent time derivative of the orbital period in milliseconds per year (ms/yr). Notes ----- The systemic proper motion of a transiting star-planet system can have a significant influence on the observed period between transits due to the *apparent* apsidal precession :math:`\\dot{\\omega}_{\\mu}`. This manifests as a constant change in the observed period, which Rafikov (2009) [1]_ expresses as: .. math:: \\dot{P}_{\\omega,\\mu} = -\\frac{2\\pi}{n^2} \\frac{(1-e^2)^{3/2}}{(1+e\\sin{ \\omega_p})^2} \\times \\left[\\ddot{\\omega}_{\\mu}-2(\\dot{\\omega}_{\\mu})^2\\frac{ e\\cos{\\omega_p}}{1+e\\sin{\\omega_p}}\\right] where :math:`e` is the orbit eccentricity, :math:`\\omega_p` is the argument of pericenter, :math:`n` is the orbit mean motion, :math:`\\dot{\\omega}_{\\mu}` is the apparent precession rate, and :math:`\\ddot{\\omega}_{\\mu}` is the time derivative of the latter. Rafikov [1]_ posits that for transiting systems the following approximation is valid: .. math:: \\ddot{\\omega}_{\\mu} \\sim \\mu^2 \\sim (\\dot{\\omega}_{\\mu})^2 which effectively maximizes the first equation. References ---------- .. [1] :cite:t:`Rafikov2009`. https://doi.org/10.1088/0004-637X/700/2/965 """ # unit conversions mu *= 1 / (1000 * 206265) # mas/yr to rad/yr # derive parameters nu = 2 * np.pi / P dwdt = mu ddwdt = mu**2 # calculate apparent period derivative t1 = -2 * np.pi * (1 / nu) ** 2 t2 = (1 - e**2) ** (3 / 2) / (1 + e * np.sin(w)) ** 2 t3 = ddwdt - 2 * dwdt**2 * (e * np.cos(w)) / (1 + e * np.sin(w)) pdot_pm = t1 * t2 * t3 # days^2/yr^2 # return the apparent period derivative in ms/yr return pdot_pm * (1 / 365.25) * 8.64e7
[docs] def proper_motion_tdot(P, e, w, i, T, wdot_pm, idot_pm, M_s, R_s): """Calculate the time derivative of the transit duration due to systemic proper motion. This method returns the time derivative of the transit duration due to systemic proper motion using Equation (9) from Rafikov (2009) [1]_. Parameters ---------- P : float The orbital period in days. e : float The eccentricity of the orbit. w : float The argument of pericenter in radians. i : float The line-of-sight inclination of the orbit in degrees. T : float The transit duration in minutes. wdot_pm : float The apparent apsidal precession rate in radians per year. idot_pm : float The apparent time derivative of the line-of-sight inclination in radians per year. M_s : float The host star mass in solar masses. R_s : float The host star radius in solar radii. Returns ------- float A constant time derivative of the transit duration in milliseconds per year. Notes ----- The systemic proper motion of a transiting star-planet system can have a significant influence on the observed transit duration, :math:`T`, due to the proper motion-induced variation of the line-of-sight inclination :math:`\\dot{i}_\\mu` and the *apparent* apsidal precession :math:`\\dot{\\omega}_{\\mu}`. The transit duration variation (TDV) is governed by the following expression [1]_: .. math:: \\dot{T}_{\\mu} = - \\frac{T}{1 + e\\sin{\\omega_p}} \\times \\left[e\\dot{\\omega}_{ \\mu}\\cos{\\omega_p} - g \\left(\\dot{i}_{\\mu}\\sin{i} + \\dot{\\omega}_{\\mu}\\cos{ i}\\frac{e\\cos{\\omega_p}}{1+e\\sin{\\omega_p}}\\right)\\right] where, .. math:: g = \\frac{a}{R_\\star}\\frac{b}{1 - b^2} and where :math:`e` is the orbit eccentricity, :math:`\\omega_p` is the argument of pericenter, :math:`b` is the impact parameter, :math:`a` is the semi major axis, and :math:`R_\\star` is the host star radius. References ---------- .. [1] :cite:t:`Rafikov2009`. https://doi.org/10.1088/0004-637X/700/2/965 """ # derive parameters a = get_semi_major_axis_from_period(P, M_s) # unit conversions i *= np.pi / 180 # degrees to radians R_s *= R_sun # solar radii to m T *= 60 * 1000 # minutes to ms b = (a / R_s) * np.cos(i) * (1 - e**2) / (1 + e * np.sin(w)) g = (a / R_s) * b / (1 - b**2) # calculate the transit duration variation t1 = -T / (1 + e * np.sin(w)) t2 = e * wdot_pm * np.cos(w) t3 = -g * ( idot_pm * np.sin(i) + wdot_pm * np.cos(i) * e * np.cos(w) / (1 + e * np.sin(w)) ) # return the transit duration variation in ms/yr return t1 * (t2 + t3)
[docs] def proper_motion_shklovskii(P, mu, D): """Calculate the apparent time derivative of observed period due to the Shklovskii effect. This method returns the apparent rate of change of the period between transits due to the Shklovskii effect, using Equation (21) from Rafikov (2009) [1]_. Parameters ---------- P : float The orbital period in days. mu : float The proper motion of the system in mas/yr (milliarcseconds per year). D : float The distance to the system in parsecs. Returns ------- float The apparent time derivative of the orbital period in milliseconds per year (ms/yr). Notes ----- As a star-planet system moves through space, the radial component of its velocity, :math:`v_r`, varies in time as a result of the nonzero transverse velocity component: .. math:: v_t = \\mu D where :math:`D` is the distance to the system in parsecs and :math:`\\mu = |\\vec{\\mu}|` is the magnitude of the proper motion vector. Consequently, the radial velocity component is time-dependent, which leads to a secular variation in the observed period between transits, denoted :math:`\\dot{P}_{ \\mathrm{Shk}}`. This is known as the Shklovskii Effect [1]_, expressed in the following convenient form by Rafikov (2009) [2]_: .. math:: \\dot{P}_{\\mathrm{Shk}} = 20\\left(\\frac{\\mu}{100\\,{\\mathrm{mas\\, yr}^{-1}}}\\right)^2\\frac{D}{100}\\frac{P}{3\\,\\mathrm{day}}\\,\\mu {\\mathrm{s\\, yr}^{-1}} References ---------- .. [1] :cite:t:`Rafikov2009`. https://doi.org/10.1088/0004-637X/700/2/965 .. [2] :cite:t:`Shklovskii1970`. https://ui.adsabs.harvard.edu/abs/1970SvA....13..562S """ # calculate period change in microseconds per year pdot = 20 * (mu / 100) ** 2 * (D / 100) * (P / 3) pdot *= 0.001 # convert microseconds to milliseconds # return the apparent period derivative in ms/yr return pdot
[docs] def resolved_binary_mass_from_rv_trend(theta, D, dvdt): """Calculate the minimum mass of a resolved secondary star given a measured radial acceleration. This method applies to the case where an acceleration has been observed in radial velocity observations of a star, and there is a known secondary object for which an angular separation has been measured. It uses Equation (6) from Torres (1999) [1]_ to estimate a lower limit for the mass of the secondary object, under the assumption that the radial velocity trend is due entirely to its gravitational influence. In the derivation of this equation, the author makes no assumptions about the mass or brightness of the secondary [1]_. Parameters ---------- theta : float The angular separation of the binary in arcseconds. D : float The distance to the system in parsecs. dvdt : float The first-order RV acceleration term in m/s/day. Returns ------- float The mass of the secondary star in solar masses. Notes ----- Given the angular separation of a known companion object, the distance to the system, and a measured acceleration in radial velocity observations, this method uses Equation (6) from Torres (1999) [1]_ to estimate the minimum mass of the companion: .. math:: M_B = 5.341 \\times 10^{-6}(D \\rho)^2\\left|\\dot{\\gamma}\\right| \\Phi where :math:`\\rho` is the angular separation of the binary in arcseconds, :math:`D` is the distance to the system in parsecs, :math:`\\dot{\\gamma}` is the first-order radial acceleration term in m/s/day, and :math:`M_B` is in units of solar masses. The parameter :math:`\\Phi` is a function of the eccentricity, longitude of pericenter, and inclination of the companion's orbit. Assuming that these parameters are unconstrained, this method uses the minimum value :math:`\\Phi = \\frac{3\\sqrt{3}}{2}` to determine the minimum companion mass. References ---------- .. [1] :cite:t:`Torres1999`. https://doi.org/10.1086/316313. """ # convert the slope from m/s/day to ms/s/yr dvdt *= 365.25 # calculate the minimum value of the unknown component 'Phi' Phi = 3 * np.sqrt(3) / 2 # return the mass of the resolved companion star in solar masses return 5.341e-6 * (D * theta) ** 2 * np.abs(dvdt) * Phi
[docs] def resolved_binary_rv_trend_from_mass(theta, D, M_B): """Calculate the slope of a linear RV trend given properties of a resolved stellar companion. This method returns the minimum possible acceleration, induced by a bound stellar companion, that may be observed in radial velocity observations of the primary. The secondary star must have been resolved through imaging or astrometric measurements, such that the angular separation is known. It uses Equation (6) from Torres (1999) [1]_, which makes no assumptions about the mass or brightness of the secondary [1]_. Parameters ---------- theta : float The angular separation of the binary in arcseconds. D : float The distance to the system in parsecs. M_B : float The mass of the stellar companion in solar masses. Returns ------- float The first-order radial acceleration term in m/s/day. Notes ----- Given the angular separation of a known companion, the distance to the system, and the mass of the companion, this method uses Equation (6) from Torres et al. (1999) [1]_ to estimate the acceleration that could be measured in radial velocity data: .. math:: \\left|\\dot{\\gamma}\\right| = \\frac{M_B}{5.341 \\times 10^{-6}(D \\rho)^2 \\Phi} where :math:`\\rho` is the angular separation of the binary in arcseconds, :math:`D` is the distance to the system in parsecs, and :math:`M_B` is the mass of the stellar companion in solar masses. In the derivation of this equation, the author makes no assumptions about the mass or brightness of the secondary [1]_. The parameter :math:`\\Phi` is a function of the eccentricity, longitude of pericenter, and inclination of the companion's orbit. Assuming that these parameters are unconstrained, this method uses the minimum value :math:`\\Phi = \\frac{3\\sqrt{3}}{2}` to determine the minimum companion mass. References ---------- .. [1] :cite:t:`Torres1999`. https://doi.org/10.1086/316313. """ # calculate the minimum value of the unknown component 'Phi' Phi = 3 * np.sqrt(3) / 2 # calculate the expected linear RV trend in m/s/yr dvdt = M_B / (5.341e-6 * (D * theta) ** 2 * Phi) # return the slope in m/s/day return dvdt / 365.25
[docs] def get_semi_major_axis_from_period(P, M_s): """Calculate the semi major axis in meters given the orbital period and host star mass. Parameters ---------- P : float The orbital period in days. M_s : float The mass of the host star in solar masses. Returns ------- float The semi major axis of the orbit in meters. """ # unit conversions P *= 86400 # days to seconds M_s *= M_sun # solar masses to kg # return the semi major axis in meters return (G * M_s * P**2 / (4 * np.pi**2)) ** (1 / 3)