Source code for posydon.binary_evol.binarystar

"""The binary object describes current and past state of the binary.

The binary object is composed of two star objects and contains the current and
past states of the binary. Only parameters in the BINARYPROPERTIES list are
stored in the history.

The current parameter value of the star object is accessed as, e.g.
`binary.orbital_period` while its past history with
`binary.orbital_period_history`.

The two stars are accessed as, e.g. `binary.star_1.mass`
while their past history with `binary.star_1.mass_history`.

"""


__authors__ = [
    "Konstantinos Kovlakas <Konstantinos.Kovlakas@unige.ch>",
    "Kyle Akira Rocha <kylerocha2024@u.northwestern.edu>",
    "Simone Bavera <Simone.Bavera@unige.ch>",
    "Jeffrey Andrews <jeffrey.andrews@northwestern.edu>",
    "Nam Tran <tranhn03@gmail.com>",
    "Philipp Moura Srivastava <philipp.msrivastava@gmail.com>",
    "Devina Misra <devina.misra@unige.ch>",
    "Scott Coughlin <scottcoughlin2014@u.northwestern.edu>",
]


import signal
import copy
import numpy as np
import pandas as pd
from posydon.binary_evol.simulationproperties import SimulationProperties
from posydon.binary_evol.singlestar import SingleStar, STARPROPERTIES
from posydon.utils.common_functions import (
    check_state_of_star, orbital_period_from_separation,
    orbital_separation_from_period, get_binary_state_and_event_and_mt_case)
from posydon.popsyn.io import (clean_binary_history_df, clean_binary_oneline_df)
from posydon.binary_evol.flow_chart import UNDEFINED_STATES
from posydon.utils.posydonerror import FlowError 


# star property: column names in binary history for star 1 and star 2
STAR_ATTRIBUTES_FROM_BINARY_HISTORY = {
    "mass": ["star_1_mass", "star_2_mass"],
    "lg_mdot": ["lg_mstar_dot_1", "lg_mstar_dot_2"],
    "lg_system_mdot": ["lg_system_mdot_1", "lg_system_mdot_2"],
    "lg_wind_mdot": ["lg_wind_mdot_1", "lg_wind_mdot_2"],
}

# only mention those with names different from the column names in history data
STAR_ATTRIBUTES_FROM_STAR_HISTORY = {
    'state': None,                      # to be computed after loading
    'metallicity': None,                # from initial values
    'mass': None,                       # from binary history
    'lg_mdot': None,                    # from binary history
    'lg_system_mdot': None,             # from binary history
    'lg_wind_mdot': None,               # from binary history
    'spin': 'spin_parameter',
    'profile': None
}


BINARY_ATTRIBUTES_FROM_HISTORY = {
    'state': None,
    'event': None,
    'time': 'age',
    'separation': 'binary_separation',
    'orbital_period': 'period_days',
    'eccentricity': None,
    'V_sys': None,
    'mass_transfer_case': None,
    'nearest_neighbour_distance': None
}


BINARYPROPERTIES = [
    # The state and event of the system. For more information, see
    # `posydon.utils.common_functions.get_binary_state_and_event_and_mt_case()
    'state',                    #
    'event',
    'time',                     # age of the system (yr)
    'separation',               # binary orbital separation (solar radii)
    'orbital_period',           # binary orbital period (days)
    'eccentricity',             # binary eccentricity
    'V_sys',                    # list of the 3 systemic velocity coordinates
    # (R_{star} - R_{Roche_lobe}) / R_{Roche_lobe}...
    'rl_relative_overflow_1',   # ...for star 1
    'rl_relative_overflow_2',   # ...for star 2
    'lg_mtransfer_rate',        # log10 of mass lost from the donor (Msun/yr)
    'mass_transfer_case',       # current mass transfer case of the system.
                                # See `get_binary_state_and_event_and_mt_case`
                                # in `posydon.utils.common_functions`.
    'trap_radius',
    'acc_radius',
    't_sync_rad_1',
    't_sync_conv_1',
    't_sync_rad_2',
    't_sync_conv_2',
    'nearest_neighbour_distance',   # the distance of system from its nearest
                                    # neighbour of MESA binary system  in case
                                    # of interpolation during the the end of
                                    # the previous step including MESA psygrid.
                                    # The distance is normalized in the
                                    # parameter space and limits at which it
                                    # was calculated. See `mesa_step` for more.
]


MAXIMUM_STEP_TIME = 120


[docs] def signal_handler(signum, frame): """React to a maximum time signal.""" raise RuntimeError("Binary Step Exceeded Alloted Time: {}". format(MAXIMUM_STEP_TIME))
signal.signal(signal.SIGALRM, signal_handler)
[docs] class BinaryStar: """A class containing the state and history of a stellar binary.""" def __init__(self, star_1=None, star_2=None, index=None, properties=None, **binary_kwargs): """Initialize a binary star. Arguments --------- properties : SimulationProperties Instance of the SimulationProperties class (default: None) star_1 : SingleStar The first star of the binary. star_2 : Star The second star of the binary. **binary_kwargs : dictionary List of initialization parameters for a binary """ # Binary Index self.index = index # Create the two stars self.star_1 = star_1 if star_1 is not None else SingleStar() self.star_2 = star_2 if star_2 is not None else SingleStar() # Set the initial binary properties for item in BINARYPROPERTIES: if item == 'V_sys': setattr(self, item, binary_kwargs.pop(item, [0,0,0])) elif item == 'mass_transfer_case': setattr(self, item, binary_kwargs.pop(item, 'None')) elif item == 'nearest_neighbour_distance': setattr(self, item, binary_kwargs.pop(item, ['None', 'None', 'None'])) else: setattr(self, item, binary_kwargs.pop(item, None)) setattr(self, item + '_history', [getattr(self, item)]) for key, val in binary_kwargs.items(): setattr(self, key, val) if getattr(self.star_1, "mass") is not None and getattr(self.star_2, "mass") is not None: if getattr(self, "separation") is None and getattr(self, "orbital_period") is not None: setattr(self, "separation", orbital_separation_from_period(self.orbital_period, self.star_1.mass, self.star_2.mass)) elif getattr(self, "orbital_period") is None and getattr(self, "separation") is not None: setattr(self, "orbital_period", orbital_period_from_separation(self.separation, self.star_1.mass, self.star_2.mass)) if not hasattr(self, 'inspiral_time'): self.inspiral_time = None if not hasattr(self, 'mass_transfer_case'): self.mass_transfer_case = 'None' if not hasattr(self, 'true_anomaly_first_SN'): self.true_anomaly_SN1 = None if not hasattr(self, 'true_anomaly_second_SN'): self.true_anomaly_SN2 = None if not hasattr(self, 'first_SN_already_occurred'): self.first_SN_already_occurred = False if not hasattr(self, 'history_verbose'): self.history_verbose = False # store interpolation_class and mt_history for each step_MESA for grid_type in ['HMS_HMS','CO_HMS_RLO','CO_HeMS','CO_HeMS_RLO']: if not hasattr(self, f'interp_class_{grid_type}'): setattr(self, f'interp_class_{grid_type}', None) if not hasattr(self, f'mt_history_{grid_type}'): setattr(self, f'mt_history_{grid_type}', None) if not hasattr(self, f'culmulative_mt_case_{grid_type}'): setattr(self, f'culmulative_mt_case_{grid_type}', None) # SimulationProperties object - parameters & parameterizations if isinstance(properties, SimulationProperties): self.properties = properties else: self.properties = SimulationProperties()
[docs] def evolve(self): """Evolve a binary from start to finish.""" self.properties.pre_evolve(self) # Code to make sure start time is less than max_simulation_time if self.time > self.properties.max_simulation_time: raise ValueError( "The binary's birth time ({0}) is greater than " "`max_simulation_time` ({1}).".format( self.time, self.properties.max_simulation_time)) max_n_steps = self.properties.max_n_steps_per_binary n_steps = 0 while (self.event != 'END' and self.event != 'FAILED' and self.event not in self.properties.end_events and self.state not in self.properties.end_states): signal.alarm(MAXIMUM_STEP_TIME) self.run_step() n_steps += 1 if max_n_steps is not None: if n_steps > max_n_steps: raise RuntimeError("Exceeded maximum number of steps ({})".format(max_n_steps)) signal.alarm(0) # turning off alarm self.properties.post_evolve(self)
[docs] def run_step(self): """Evolve the binary through one evolutionary step.""" total_state = (self.star_1.state, self.star_2.state, self.state, self.event) if total_state in UNDEFINED_STATES: raise FlowError(f"Binary failed with a known undefined state in the flow:\n{total_state}") next_step_name = self.properties.flow.get(total_state) if next_step_name is None: raise ValueError("Undefined next step given stars/binary states {}.".format(total_state)) next_step = getattr(self.properties, next_step_name, None) if next_step is None: raise ValueError("Next step name '{}' does not correspond to a function in " "SimulationProperties.".format(next_step_name)) self.properties.pre_step(self, next_step_name) next_step(self) self.append_state() self.properties.post_step(self, next_step_name)
[docs] def append_state(self): """Update the history of the binaries' properties.""" ## do not append redirect steps to the binary history if history_verbose=False if not self.history_verbose and getattr(self, "event") is not None: if "redirect" in getattr(self, "event"): return # Append to the binary history lists for item in BINARYPROPERTIES: getattr(self, item + '_history').append(getattr(self, item)) # Append to the individual star history lists self.star_1.append_state() self.star_2.append_state()
[docs] def switch_star(self): """Switch stars.""" self.star_1, self.star_2 = self.star_2, self.star_1
[docs] def restore(self, i=0): """Restore the BinaryStar() object to its i-th state, keeping the binary history before the i-th state. Parameters ---------- i : int The index of the binary object history to reset the binary to. By default 0, i.e. the star will be restored to its initial state. """ # Move current binary properties to the ith step, using its history for p in BINARYPROPERTIES: setattr(self, p, getattr(self, '{}_history'.format(p))[i]) ## delete the binary history after the i-th index setattr(self, p + '_history', getattr(self, p + '_history')[0:i+1]) ## if running with extra hooks, restore any extra hook columns for hook in self.properties.all_hooks_classes: if hasattr(hook, 'extra_binary_col_names'): extra_columns = getattr(hook, 'extra_binary_col_names') for col in extra_columns: setattr(self, col, getattr(self, col)[0:i+1]) for star in (self.star_1, self.star_2): star.restore(i, hooks=self.properties.all_hooks_classes)
[docs] def reset(self, properties=None): """Reset the binary to its ZAMS state. Parameters ---------- properties : SimulationProperties Instance of the SimulationProperties class (default: None) """ # If provided, update the simulation properties class if properties is not None: self.properties = SimulationProperties(properties) # Use the restore function to move the binary back to its initial state self.restore(i=0)
[docs] def update_star_states(self): """Update the states of the two stars in the binary.""" if self.star_1.state != 'massless_remnant': self.star_1.state = check_state_of_star( self.star_1, star_CO=self.star_1.state in ["WD", "NS", "BH"]) if self.star_2.state != 'massless_remnant': self.star_2.state = check_state_of_star( self.star_2, star_CO=self.star_2.state in ["WD", "NS", "BH"])
[docs] def to_df(self, **kwargs): """Return history parameters from the binary in a DataFrame. Includes star 1 and 2 (S1, S2) data and an extra column 'binary_index'. Parameters ---------- extra_columns : dict( 'name':dtype, .... ) Extra binary parameters to return in DataFrame that are not included in BINARYPROPERTIES. All columns must have an associated pandas data type. Can be used in combination with `only_select_columns`. Assumes names have no suffix. ignore_columns : list Names of binary parameters to ignore. Assumes names have `_history` suffix. only_select_columns : list Names of the only columns to include. Assumes names have `_history` suffix. Can be used in combination with `extra_columns`. null_value : float Replace all None values with something else (for saving). Default is np.nan. include_S1, include_S2 : bool Choose to include star 1 or 2 data to the DataFrame. The default is to include both. S1_kwargs, S2_kwargs : dict kwargs to pass to each star's 'to_df' method (extra/ignore columns) Returns ------- pandas DataFrame """ extra_binary_cols_dict = kwargs.get('extra_columns', {}) extra_columns = list(extra_binary_cols_dict.keys()) all_keys = (["binary_index"] + [key+'_history' for key in BINARYPROPERTIES] + extra_columns) ignore_cols = list(kwargs.get('ignore_columns', [])) keys_to_save = [i for i in all_keys if not ( (i.split('_history')[0] in ignore_cols) or (i in ignore_cols))] if bool(kwargs.get('only_select_columns')): user_keys_to_save = list(kwargs.get('only_select_columns')) keys_to_save = (["binary_index"] + [key+'_history' for key in user_keys_to_save] + extra_columns) try: data_to_save = [getattr(self, key) for key in keys_to_save[1:]] col_lengths = [len(x) for x in data_to_save] max_col_length = np.max(col_lengths) # binary_index data_to_save.insert(0, [self.index]*max_col_length) # If a binary fails, usually history cols have diff lengths. # This should append NAN to create even columns. all_equal_length_cols = len(set(col_lengths)) == 1 if not all_equal_length_cols: for col in data_to_save: col.extend([np.nan] * abs(max_col_length - len(col))) where_none = np.array([[True if var is None else False for var in column] for column in data_to_save], dtype=bool) except AttributeError as err: raise AttributeError( str(err) + "\n\nAvailable attributes in BinaryStar: \n{}". format(self.__dict__.keys())) # Convert None to np.nan by default bin_data = np.array(data_to_save, dtype=object) bin_data[where_none] = kwargs.get('null_value', np.nan) bin_data = np.transpose(bin_data) # remove the _history at the end of all column names column_names = [name.split('_history')[0] for name in keys_to_save] bin_df = pd.DataFrame(bin_data, columns=column_names) # Add 3 columns for V_sys if 'V_sys' in column_names: V_sys_x = np.zeros(len(bin_df)) V_sys_y = np.zeros(len(bin_df)) V_sys_z = np.zeros(len(bin_df)) for i in range(len(bin_df)): V_sys_x[i] = bin_df.iloc[i]['V_sys'][0] V_sys_y[i] = bin_df.iloc[i]['V_sys'][1] V_sys_z[i] = bin_df.iloc[i]['V_sys'][2] bin_df['V_sys_x'] = copy.deepcopy(V_sys_x) bin_df['V_sys_y'] = copy.deepcopy(V_sys_y) bin_df['V_sys_z'] = copy.deepcopy(V_sys_z) # Lose the V_sys list bin_df = bin_df.drop(['V_sys'], axis=1) frames = [bin_df] if kwargs.get('include_S1', True): # we are hard coding the prefix frames.append(self.star_1.to_df( prefix='S1_', null_value=kwargs.get('null_value', np.nan), **kwargs.get('S1_kwargs', {}))) if kwargs.get('include_S2', True): frames.append(self.star_2.to_df( prefix='S2_', null_value=kwargs.get('null_value', np.nan), **kwargs.get('S2_kwargs', {}))) binary_df = pd.concat(frames, axis=1) binary_df.set_index('binary_index', inplace=True) extra_s1_cols_dict = kwargs.get('S1_kwargs', {}).get('extra_columns', {}) extra_s2_cols_dict = kwargs.get('S2_kwargs', {}).get('extra_columns', {}) binary_df = clean_binary_history_df(binary_df, extra_binary_dtypes_user=extra_binary_cols_dict, extra_S1_dtypes_user=extra_s1_cols_dict, extra_S2_dtypes_user=extra_s2_cols_dict) return binary_df
[docs] @classmethod def from_df(cls, dataframe, **kwargs): """Convert a binary from a pandas DataFrame to BinaryStar instance. Parameters ---------- dataframe : Pandas DataFrame data to turn into a BinaryStar instance. index : int, optional Sets the binary index. extra_columns : dict, optional Column names to be added directly to binary not in BINARYPROPERTIES. Returns ------- New instance of BinaryStar """ if isinstance(dataframe, pd.Series): dataframe = pd.DataFrame(dataframe.to_dict(), index=[0]) # split input dataframe into kwargs dicts binary_params, star1_params, star2_params = dict(), dict(), dict() extra_params = dict() extra_columns = kwargs.get('extra_columns', {}) hist_lengths = [] for name in list(dataframe.columns): if 'S1' in name: corr_name = name.split('S1_')[-1] + '_history' star1_params[corr_name] = list(dataframe[name]) hist_lengths.append(len(star1_params[corr_name])) elif 'S2' in name: corr_name = name.split('S2_')[-1] + '_history' star2_params[corr_name] = list(dataframe[name]) hist_lengths.append(len(star2_params[corr_name])) elif name in extra_columns: # this assumes extra cols in binary, not star 1 or 2 extra_params[name] = list(dataframe[name]) hist_lengths.append(len(extra_params[name])) else: corr_name = name + '_history' binary_params[corr_name] = list(dataframe[name]) hist_lengths.append(len(binary_params[corr_name])) # make sure all history columns have equal length assert len(set(hist_lengths)) == 1 history_length = set(hist_lengths).pop() if ('binary_index' in dataframe.index.name and not kwargs.get('index', False)): binary_index = set(dataframe.index).pop() else: binary_index = kwargs.get('index', None) binary = cls(index=binary_index, star_1=SingleStar(**star1_params), star_2=SingleStar(**star2_params), **binary_params) # set extra history columns directly for key, val in extra_params.items(): setattr(binary, key, val) # set some orbital parameters that should exist by hand bp_keys = binary_params.keys() if 'eccentricity_history' not in bp_keys: setattr(binary, 'eccentricity_history', [0]*history_length) setattr(binary, 'eccentricity', 0) if ('separation_history' not in bp_keys and 'orbital_period_history' in bp_keys): separation = orbital_separation_from_period( np.array(binary.orbital_period_history), np.array(binary.star_1.mass_history), np.array(binary.star_2.mass_history),) setattr(binary, 'separation_history', list(separation)) setattr(binary, 'separation', list(separation)[-1]) if ('orbital_period_history' not in bp_keys and 'seperation_history' in bp_keys): period = orbital_period_from_separation( np.array(binary.seperation_history), np.array(binary.star_1.mass_history), np.array(binary.star_2.mass_history),) setattr(binary, 'orbital_period_history', list(period)) setattr(binary, 'orbital_period', list(period)[-1]) # set the binary, star1, star2 parameters to last history value in df for params, pointer in zip( [star1_params, star2_params, binary_params], [binary.star_1, binary.star_2, binary]): for key, val in params.items(): setattr(pointer, key.split('_history')[0], val[-1]) # make BINARYPROPERTIES, history columns same length if not given already_included_cols = [name.split('_history')[0] for name in binary_params.keys()] valid_binaryprop_keys = [p for p in BINARYPROPERTIES if (p not in already_included_cols)] for prop_key in valid_binaryprop_keys: last_default_value = getattr(binary, prop_key+'_history')[-1] len_default = len(getattr(binary, prop_key+'_history')) diff = history_length - len_default getattr(binary, prop_key+'_history').extend([last_default_value] * diff) # make STARPROPERTIES, history columns same length if not given for params, star in zip( [star1_params, star2_params], [binary.star_1, binary.star_2]): already_included_cols = [name.split('_history')[0] for name in params.keys()] valid_starprop_keys = [p for p in STARPROPERTIES if not(p in already_included_cols)] for prop_key in valid_starprop_keys: last_default_value = getattr(star, prop_key+'_history')[-1] len_default = len(getattr(star, prop_key+'_history')) diff = history_length - len_default getattr(star, prop_key+'_history').extend([last_default_value] * diff) return binary
[docs] def to_oneline_df(self, scalar_names=[], history=True, **kwargs): """Convert binary into a single row DataFrame.""" if history: bin_kwargs = kwargs.copy() bin_kwargs['include_S1'] = False bin_kwargs['include_S2'] = False output_df = self.to_df(**bin_kwargs) initial_final_data = output_df.values[[0, -1], :] # first/last row col_names = list(output_df.columns) oneline_names = (['binary_index'] + [s + '_i' for s in col_names] + [s + '_f' for s in col_names]) oneline_data = [self.index] + [d for d in initial_final_data.flatten()] bin_df = pd.DataFrame(data=[oneline_data], columns=oneline_names) else: bin_df = pd.DataFrame() s1_kwargs = kwargs.get('S1_kwargs', {}) if bool(s1_kwargs): s1_df = self.star_1.to_oneline_df(prefix='S1_', **s1_kwargs) else: s1_df = pd.DataFrame() s2_kwargs = kwargs.get('S2_kwargs', {}) if bool(s2_kwargs): s2_df = self.star_2.to_oneline_df(prefix='S2_', **s2_kwargs) else: s2_df = pd.DataFrame() oneline_df = pd.concat([bin_df, s1_df, s2_df], axis=1) for name in scalar_names: if hasattr(self, name): oneline_df[name] = [getattr(self, name)] # check for variables set in BinaryPopulation handling safe evolution if hasattr(self, 'traceback'): oneline_df['FAILED'] = [1] else: oneline_df['FAILED'] = [0] if hasattr(self, 'warnings'): oneline_df['WARNING'] = [1] else: oneline_df['WARNING'] = [0] oneline_df.set_index('binary_index', inplace=True) # try to coerce data types automatically oneline_df = oneline_df.infer_objects() # Set data types for all columns explicitly # we are assuming you may pass the same kwargs to both to_df and oneline extra_binary_cols_dict = kwargs.get('extra_columns', {}) extra_s1_cols_dict = kwargs.get('S1_kwargs', {}).get('extra_columns', {}) extra_s2_cols_dict = kwargs.get('S2_kwargs', {}).get('extra_columns', {}) oneline_df = clean_binary_oneline_df(oneline_df, extra_binary_dtypes_user=extra_binary_cols_dict, extra_S1_dtypes_user=extra_s1_cols_dict, extra_S2_dtypes_user=extra_s2_cols_dict) return oneline_df
[docs] @classmethod def from_oneline_df(cls, oneline_df, **kwargs): """Convert a oneline DataFrame into a BinaryStar. The oneline DataFrame is expected to have initial-final values from history and any individual values that don't have histories. Parameters ---------- oneline_df : DataFrame A oneline DataFrame describing a binary. index : int, None Binary index extra_columns : dict Names of any extra history columns not inlcuded in BINARYPROPERTIES Returns ------- A new BinaryStar instance. """ if isinstance(oneline_df, pd.Series): oneline_df = pd.DataFrame(oneline_df.to_dict(), index=[0]) binary_params, star1_params, star2_params = dict(), dict(), dict() extra_params = dict() extra_columns = kwargs.get('extra_columns', {}) hist_lengths = [] for name in list(oneline_df.columns): if '_f' in name[-2:]: continue # ignore final values param_name = name.split('_i')[0] special_cases = ['natal_kick_array'] if any([i in param_name for i in special_cases]): continue # deal with special cases later # ignore error and warning if name in ['FAILED', 'WARNING']: continue if 'S1' in name: param_name = param_name.split('S1_')[-1] ending_str = '_history' if param_name in STARPROPERTIES else '' star1_params[param_name + ending_str] = list(oneline_df[name]) hist_lengths.append(len(list(oneline_df[name]))) elif 'S2' in name: param_name = param_name.split('S2_')[-1] ending_str = '_history' if param_name in STARPROPERTIES else '' star2_params[param_name + ending_str] = list(oneline_df[name]) hist_lengths.append(len(list(oneline_df[name]))) elif param_name in extra_columns: # this assumes extra cols in binary, not star 1 or 2 extra_params[param_name] = list(oneline_df[name]) hist_lengths.append(len(list(oneline_df[name]))) else: # binary ending_str = ('_history' if param_name in BINARYPROPERTIES else '') binary_params[param_name + ending_str] = list(oneline_df[name]) hist_lengths.append(len(list(oneline_df[name]))) # make sure all history columns have equal length assert len(set(hist_lengths)) == 1 history_length = set(hist_lengths).pop() if any(['S1_natal_kick_array' in name for name in oneline_df.columns]): natalkick_names = ['S1_natal_kick_array_{}'.format(i) for i in range(4)] star1_params['natal_kick_array'] = \ oneline_df[natalkick_names].values if any(['S2_natal_kick_array' in name for name in oneline_df.columns]): natalkick_names = ['S2_natal_kick_array_{}'.format(i) for i in range(4)] star2_params['natal_kick_array'] = \ oneline_df[natalkick_names].values if ('binary_index' in oneline_df.index.name and not kwargs.get('index', None)): binary_index = set(oneline_df.index).pop() else: binary_index = kwargs.get('index', None) binary = cls(index=binary_index, star_1=SingleStar(**star1_params), star_2=SingleStar(**star2_params), **binary_params) # set extra history columns directly for key, val in extra_params.items(): setattr(binary, key, val) # set some orbital parameters that should exist by hand bp_keys = binary_params.keys() if 'eccentricity_history' not in bp_keys: setattr(binary, 'eccentricity_history', [0]) setattr(binary, 'eccentricity', 0) if ('separation_history' not in bp_keys and 'orbital_period_history' in bp_keys): separation = orbital_separation_from_period( np.array(binary.orbital_period_history), np.array(binary.star_1.mass_history), np.array(binary.star_2.mass_history),) setattr(binary, 'separation_history', list(separation)) setattr(binary, 'separation', list(separation)[-1]) if ('orbital_period_history' not in bp_keys and 'seperation_history' in bp_keys): period = orbital_period_from_separation( np.array(binary.seperation_history), np.array(binary.star_1.mass_history), np.array(binary.star_2.mass_history),) setattr(binary, 'orbital_period_history', list(period)) setattr(binary, 'orbital_period', list(period)[-1]) # set the binary, star1, star2 parameters to last history value in df for params, pointer in zip( [star1_params, star2_params, binary_params], [binary.star_1, binary.star_2, binary]): for key, val in params.items(): setattr(pointer, key.split('_history')[0], val[-1]) # make BINARYPROPERTIES, history columns same length if not given already_included_cols = [name.split('_history')[0] for name in binary_params.keys()] valid_binaryprop_keys = [p for p in BINARYPROPERTIES if not(p in already_included_cols)] for prop_key in valid_binaryprop_keys: last_default_value = getattr(binary, prop_key+'_history')[-1] len_default = len(getattr(binary, prop_key+'_history')) diff = history_length - len_default getattr(binary, prop_key+'_history').extend([last_default_value] * diff) # if the binary errored, then set the final event if bool(oneline_df['FAILED'].values[-1]): setattr(binary, 'event', 'FAILED') return binary
def __repr__(self): """Return the object representation when print is called.""" s = '<{}.{} at {}>\n'.format(self.__class__.__module__, self.__class__.__name__, hex(id(self))) if hasattr(self, "error_message"): s += "BINARY FAILED: {}\n".format(self.error_message) if hasattr(self, "warning_message"): s += "WARNING FOUND: {}\n".format(self.warning_message) for p in BINARYPROPERTIES: s += '{}: {}\n'.format(p, getattr(self, p)) for star in (self.star_1, self.star_2): s += '\n{}\n'.format(star) return s[:-1] def __str__(self): """Get a printable description of the binary star.""" s = '' gap = ', ' for name in ['state', 'event']: s += str(getattr(self, name)) + gap def nan_if_not_int_or_float(value): """Return nan if `value` is neither int nor float.""" if isinstance(value, (float, int)): return value return np.nan orb_p = nan_if_not_int_or_float(self.orbital_period) m1 = nan_if_not_int_or_float(self.star_1.mass) m2 = nan_if_not_int_or_float(self.star_2.mass) s += 'p={0:.2f}'.format(orb_p) + gap s += 'S1=({0},M={1:.2f})'.format(self.star_1.state, m1) + gap s += 'S2=({0},M={1:.2f})'.format(self.star_2.state, m2) return 'BinaryStar(' + s + ')'
[docs] @staticmethod def from_run(run, history=False, profiles=False): """Create a BinaryStar object from a PSyGrid run.""" binary = BinaryStar() # get the data for the binary if run.binary_history is not None: n_steps = len(run.binary_history["age"]) if history else 1 bh_colnames = run.binary_history.dtype.names for attr in BINARYPROPERTIES: colname = BINARY_ATTRIBUTES_FROM_HISTORY.get(attr, attr) if colname is not None and colname in bh_colnames: final_value = run.final_values[colname] if history: col_history = list(run.binary_history[colname]) else: col_history = [final_value] else: final_value = None col_history = [None] * n_steps assert n_steps == len(col_history) setattr(binary, attr + "_history", col_history) setattr(binary, attr, final_value) else: return binary # if no binary history, return defaults # get the data for each companion star for star_history, star, prefix in zip([run.history1, run.history2], [binary.star_1, binary.star_2], ["S1", "S2"]): if star_history is not None: h_colnames = star_history.dtype.names for attr in STARPROPERTIES: # get the corresponding column name (default=same name) colname = STAR_ATTRIBUTES_FROM_STAR_HISTORY.get(attr, attr) if colname is not None and colname in h_colnames: final_value = run.final_values[prefix + "_" + colname] if history: col_history = list(star_history[colname]) else: col_history = [final_value] else: final_value = None col_history = [None] * n_steps assert n_steps == len(col_history) setattr(star, attr + "_history", col_history) setattr(star, attr, final_value) # set metallicities (if defined in the track)... try: metallicity = run.initial_values["Z"] except AttributeError: metallicity = None # ...and other star parameters taken from the binary history for star_index, star in enumerate([binary.star_1, binary.star_2]): star.metallicity = metallicity star.metallicity_history = [metallicity] * n_steps for attr, colnames in STAR_ATTRIBUTES_FROM_BINARY_HISTORY.items(): colname = colnames[star_index] final_value = run.final_values[colname] if history: col_history = list(run.binary_history[colname]) else: col_history = [final_value] assert n_steps == len(col_history) setattr(star, attr, final_value) setattr(star, attr + "_history", col_history) # add values at He depletion for colname in run.final_values.dtype.names: if "at_He_depletion" in colname: if colname[0:3]=="S1_": attr = colname[3:] final_value = run.final_values[colname] setattr(binary.star_1, attr, final_value) elif colname[0:3]=="S2_": attr = colname[3:] final_value = run.final_values[colname] setattr(binary.star_2, attr, final_value) else: attr = colname final_value = run.final_values[colname] setattr(binary, attr, final_value) # update eccentricity binary.eccentricity = 0.0 binary.eccentricity_history = [0.0] * n_steps # update star states n_history = len(binary.time_history) for star, track in zip([binary.star_1, binary.star_2], [run.history1, run.history2]): is_CO = track is None state_history = [check_state_of_star(star, i=i, star_CO=is_CO) for i in range(n_history)] star.state_history = state_history star.state = state_history[-1] # update binary state, event and MT case binary.state_history = [] binary.event_history = [] binary.mass_transfer_case_history = [] for i in range(n_history): # step-by-step: previous states matter! result = get_binary_state_and_event_and_mt_case(binary, i=i) binary.state, binary.event, binary.mass_transfer_case = result binary.state_history.append(binary.state) binary.event_history.append(binary.event) binary.mass_transfer_case_history.append(binary.mass_transfer_case) if profiles: binary.star_1.profile = run.final_profile1 binary.star_2.profile = run.final_profile2 return binary
[docs] def initial_condition_message(self, ini_params=None): """Generate a message with the initial conditions. Parameters ---------- ini_params : None or iterable of str If None take the initial conditions from the binary, otherwise add each item of it to the message. Returns ------- string The message with the binary initial conditions. """ if ini_params is None: ini_params = ["\nFailed Binary Initial Conditions:\n", f"S1 mass: {self.star_1.mass_history[0]} \n", f"S2 mass: {self.star_2.mass_history[0]} \n", f"S1 state: {self.star_1.state_history[0]} \n", f"S2 state: {self.star_2.state_history[0]}\n", f"orbital period: {self.orbital_period_history[0] } \n", f"eccentricity: {self.eccentricity_history[0]} \n", f"binary state: {self.state_history[0] }\n", f"binary event: {self.state_history[0] }\n", f"S1 natal kick array: {self.star_1.natal_kick_array }\n", f"S2 natal kick array: {self.star_2.natal_kick_array}\n"] message = "" for i in ini_params: message += i return message