forked from Zomega/garnets
-
Notifications
You must be signed in to change notification settings - Fork 2
/
stellar_system.py
235 lines (191 loc) · 7.91 KB
/
stellar_system.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
from constants import DISK_ECCENTRICITY
from constants import B
from constants import SUN_MASS_IN_MOON_MASSES
from constants import SUN_MASS_IN_EARTH_MASSES
from constants import SUN_MASS_IN_JUPITER_MASSES
from constants import MILLIBARS_PER_ATM
from math import sqrt
from attr import attrs, attrib
import attr
from enviroment import PlanetType
from tabulate import tabulate
@attrs(repr=False)
class Star():
mass_ratio = attrib()
age = attrib()
name = attrib(default="Unnamed Star")
planets = attrib(default=attr.Factory(list))
@property
# Approximates the luminosity of the star.
# TODO: express only as ratio?
# Source: http://en.wikipedia.org/wiki/Mass%E2%80%93luminosity_relation
# SM - this can and should be udpated to a function. Also as function of time, luminosity increases?
def luminosity_ratio(self):
if (self.mass_ratio < .43):
return .23 * (self.mass_ratio ** 2.3)
if (self.mass_ratio < 2):
return (self.mass_ratio ** 4)
# Main Sequence Stars
if (self.mass_ratio < 20):
return 1.5 * (self.mass_ratio ** 3.5)
# For HUGE stars...
return 3200 * self.mass_ratio
@property
# Source: StarGen, TODO Verify against current data.
def stellar_dust_limit(self):
return 200.0 * (self.mass_ratio ** 0.3333)
@property
def r_ecosphere(self): # Source: StarGen, TODO Name? Value?
return sqrt(self.luminosity_ratio)
@property
def life(self): # Source: StarGen, TODO Name? Value?
return 10**10 * (self.mass_ratio / self.luminosity_ratio)
def __repr__(self):
return self.name + ": mass = " + str(self.mass_ratio) + " solar mass; age = " + str(self.age) + '\n' + '\n'.join([repr(planet) for planet in self.planets])
@attrs
class StellarSystem:
star = attrib()
planets = attrib()
@attrs(repr=False)
class Orbit:
a = attrib() # semi-major axis of solar orbit (in AU)
e = attrib() # eccentricity of solar orbit
@property
def periapsis(self):
return (1 - self.e) * self.a
@property
def apoapsis(self):
return (1 + self.e) * self.a
def __repr__(self):
return 'a = ' + str(self.a) + ' e = ' + str(self.e)
@attrs
class Planetoid():
orbit = attrib()
dust_mass = attrib()
gas_mass = attrib()
@property
def mass(self):
return self.dust_mass + self.gas_mass
@property
def reduced_mass(self):
# To understand what this is all about...
# http://spiff.rit.edu/classes/phys440/lectures/reduced/reduced.html
# But some sort of 3 body case, see dole.
# TODO: Understand better?
return (self.mass / (1.0 + self.mass)) ** 0.25
@property
def inner_effect_limit(self):
temp = (self.orbit.a * (1.0 - self.orbit.e) *
(1.0 - self.mass) / (1.0 + DISK_ECCENTRICITY))
if temp < 0:
return 0
return temp
@property
def outer_effect_limit(self):
return (
self.orbit.a *
(1.0 + self.orbit.e) *
(1.0 + self.mass) /
(1.0 - DISK_ECCENTRICITY)
)
@attrs
class Planetesimal(Planetoid):
disk = attrib()
@property
def critical_mass(self):
perihelion_dist = self.orbit.a * (1.0 - self.orbit.e)
temp = perihelion_dist * sqrt(self.disk.star.luminosity_ratio)
return B * (temp ** -0.75)
def mass_repr(mass):
if mass * SUN_MASS_IN_MOON_MASSES <= 50:
return str(mass * SUN_MASS_IN_MOON_MASSES) + " M_moon"
if mass * SUN_MASS_IN_EARTH_MASSES <= 50:
return str(mass * SUN_MASS_IN_EARTH_MASSES) + " M_earth"
return str(mass * SUN_MASS_IN_JUPITER_MASSES) + " M_jupiter"
@attrs(repr=False)
class Protoplanet(Planetoid):
star = attrib()
moons = attrib(default=attr.Factory(list))
def add_moon(self, moon):
self.moons.append(moon)
@property
def mass_of_moons(self):
return sum([moon.mass for moon in self.moons])
@property
def critical_mass(self):
perihelion_dist = self.orbit.a * (1.0 - self.orbit.e)
temp = perihelion_dist * sqrt(self.star.luminosity_ratio)
return B * (temp ** -0.75)
def __repr__(self):
return (
"\tMass: " + mass_repr(self.mass) + " = attrib() Orbit: " + str(self.orbit.a) +
" AU, Moons: " + str(len(self.moons)) + "\n"
)
@attrs
class Protomoon(Planetoid):
protoplanet = attrib()
# TODO(woursler): Go over these with a fine tooth comb. Many are not relevant, or only relevant during initialization.
# Many should be properties.
# Many should be initialized differently.
@attrs(repr=False)
class Planet():
# Orbital details.
orbit = attrib()
axial_tilt = attrib() # units of degrees
mass = attrib() # mass (in solar masses)
dust_mass = attrib() # mass, ignoring gas
gas_mass = attrib() # mass, ignoring dust
moons = attrib(default=attr.Factory(list))
# ZEROES start here -- TODO(woursler): A bunch of these should be other Zero-like types.
gas_giant = attrib(default=False) # TRUE if the planet is a gas giant
moon_a = attrib(default=0) # semi-major axis of lunar orbit (in AU)
moon_e = attrib(default=0) # eccentricity of lunar orbit
core_radius = attrib(default=0) # radius of the rocky core (in km)
radius = attrib(default=0) # equatorial radius (in km)
orbit_zone = attrib(default=0) # the 'zone' of the planet
density = attrib(default=0) # density (in g/cc)
orb_period = attrib(default=0) # length of the local year (days)
day = attrib(default=0) # length of the local day (hours)
resonant_period = attrib(default=0) # TRUE if in resonant rotation
esc_velocity = attrib(default=0) # units of cm/sec
surf_accel = attrib(default=0) # units of cm/sec2
surf_grav = attrib(default=0) # units of Earth gravities
rms_velocity = attrib(default=0) # units of cm/sec
molec_weight = attrib(default=0) # smallest molecular weight retained
volatile_gas_inventory = attrib(default=0)
surf_pressure = attrib(default=0) # units of millibars (mb)
greenhouse_effect = attrib(default=0) # runaway greenhouse effect?
boil_po = attrib(default=0) # the boiling po of water (Kelvin)
albedo = attrib(default=0) # albedo of the planet
exospheric_temp = attrib(default=0) # units of degrees Kelvin
estimated_temp = attrib(default=0) # quick non-iterative estimate (K)
# for terrestrial moons and the like
estimated_terr_temp = attrib(default=0)
surf_temp = attrib(default=0) # surface temperature in Kelvin
greenhs_rise = attrib(default=0) # Temperature rise due to greenhouse
high_temp = attrib(default=0) # Day-time temperature
low_temp = attrib(default=0) # Night-time temperature
max_temp = attrib(default=0) # Summer/Day
min_temp = attrib(default=0) # Wer/Night
hydrosphere = attrib(default=0) # fraction of surface covered
cloud_cover = attrib(default=0) # fraction of surface covered
ice_cover = attrib(default=0) # fraction of surface covered
sun = attrib(default=0)
atmosphere = attrib(default=None)
type = attrib(default=PlanetType.UNKNOWN) # Type code
# ZEROES end here
def __repr__(self):
if self.atmosphere is None:
atmosphere_string = "No Atmosphere"
else:
atmosphere_string = tabulate([[gas.symbol, str(amount) + ' mb'] for gas, amount in self.atmosphere])
return tabulate([
['Type', self.type],
['Mass', mass_repr(self.mass)],
['Radius', str(self.radius) + " km"],
['Orbit', self.orbit],
['Surface gravity', str(self.surf_grav) + ' g'],
['Surface pressure', str(self.surf_pressure / MILLIBARS_PER_ATM) + ' atm'],
['Atmosphere', atmosphere_string],
['Moons', '\n'.join(repr(moon) for moon in self.moons) if len(self.moons) > 0 else 'No Moons'],
])