-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdisk_gd.py
358 lines (326 loc) · 14.9 KB
/
disk_gd.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
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
import math
import numpy
import matplotlib.pyplot as plt
from scipy import integrate
from scipy.interpolate import interp1d
from scipy.special import gamma
import warnings
# A copy of disk.py with the addition of a grain size distribution. Here, grainSize represents the minimum grain size, whereas grainSize_max represents the maximum.
class Disk:
stefbolt_const = 5.67e-8
h_const = 6.6260695729e-34
k_const = 1.380648813e-23
c_const = 2.99792458e8
"""
inner and outer radius in units of AU
grain size in units of microns
disk mass in units of earth masses
"""
def __init__(self, innerRadius, outerRadius, grainSize, diskMass, powerLaw, grainEfficiency, beltMass):
# set the grain density to 2.7 grams/cm^3 in kilograms/m^3
self.grainDensity = 2.7e3
# set the star temperature to solar temperature in Kelvin
self.starTemperature = 5500.0
# set the star radius to solar radius in meters
self.starRadius = 6.955e8*.84
# set the star luminosity to solar luminosity in Watts
self.starLuminosity = 3.839e26*.583
# set the star distance to 35 parsecs in meters
self.starDistance = 35*3.08568e16
# disable overflow warnings
warnings.filterwarnings("ignore", "overflow encountered in double_scalars")
# convert the radii to meters
self.innerRadius = float(innerRadius)*1.496e11
self.outerRadius = float(outerRadius)*1.496e11
# convert grain size to meters
self.grainSize = float(grainSize)/1e6
self.grainSize_max = 0.01 #Max grain size is set to 1 cm
self.grainpowerLaw = -2.5 #Since dN/da = -3.5
# convert to kilograms
self.diskMass = float(diskMass)*5.9742e24
self.powerLaw = float(powerLaw)
self.grainEfficiency = float(grainEfficiency)
self.beltMass = float(beltMass)*5.9742e24
# calculate zeta(grainEfficency + 4)
argument = self.grainEfficiency + 4
self.zeta = 0
for n in range(1, 100):
self.zeta += 1.0/(n**argument)
# calculate the surface density at 100 AU (sigma_100)
j = 2 - self.powerLaw
m = (self.outerRadius**j) - (self.innerRadius**j)
self.surfaceSigma = self.diskMass*j/(2*math.pi*((100*1.496e11)**self.powerLaw)*m)
# read and store data from the SED model
f = open('Model Spectrum.txt','r')
radius = 0.84*6.955*1e8
dist = 35*3.09e16
self.data_lambda = []
self.data_flux = []
line = f.readline()
while line != '':
# convert nanometers to meters
lamma = float(line[4:12])*1e-9
flux = float(line[29:38])*4*3.14159*(radius/dist)**2*1e23
# multiply by nu
flux *= self.c_const/lamma
self.data_lambda.append(lamma)
self.data_flux.append(flux)
line = f.readline()
f.close()
self.data_lambda.append(1e4*1e-6)
self.data_flux.append(2.914e-13*4*3.14159*(radius/dist)**2*1e23*self.c_const/1e-2)
self.data_lambda = self.convertToMicrons(self.data_lambda)
# read and store observed fluxes
f = open('Observed Fluxes.txt', 'r')
self.sample_lambda = []
self.sample_flux = []
self.sample_error = []
line = f.readline()
while line != '':
# read wavelength, flux, and error
info = line.split()
# convert microns to meters, but use microns
lam = float(info[0])*1e-6
self.sample_lambda.append(float(info[0]))
# make flux in Janksys into Jansky*Hz by muliplying with nu
flu = float(info[1])*self.c_const/lam
self.sample_flux.append(flu)
# do the same thing with error
err = float(info[2])*self.c_const/lam
self.sample_error.append(err)
line = f.readline()
f.close()
#read and store IRAS fluxes, which will only be used for display purposes, not fitting
f = open('IRS Spectrum.txt', 'r')
self.IRS_lambda = []
self.IRS_flux = []
self.IRS_error = []
line = f.readline()
while line != '':
info = line.split()
lam = float(info[0])
self.IRS_lambda.append(lam)
flu = float(info[1])*self.c_const/(lam*1e-6) #Want nu*F_nu
self.IRS_flux.append(flu)
err = float(info[2])/flu*self.c_const/(lam*1e-6)*100
self.IRS_error.append(err)
line = f.readline()
self.ast_avg_err = numpy.mean(self.IRS_error)
#print "Average % error in IRS Spectrum =",self.ast_avg_err,"%" #Turns out it's 5.89%
f.close()
# generate interpolation function
loglamb = map (math.log10, self.data_lambda)
logflux = map(math.log10, self.data_flux)
# if out of bounds, interpolates to 0
logFunct = interp1d(loglamb, logflux, bounds_error=False, fill_value=0)
self.interpol_funct = lambda x: 10**(logFunct(math.log10(x)))
#Add in an asteroid belt with a fixed temperature and mass
self.asteroid_radius = 1.0e-6 #arbitrary
self.M_aster = 4/3*math.pi*self.asteroid_radius**3*self.grainDensity
self.n_asteroids = self.beltMass/self.M_aster
self.Temp_a = 100
"""
changes the paramters to the disk
"""
def changeParameters(self, innerRadius, outerRadius, grainSize, diskMass, powerLaw, grainEfficiency, beltMass):
# convert the radii to meters
self.innerRadius = float(innerRadius)*1.496e11
self.outerRadius = float(outerRadius)*1.496e11
# convert grain size to meters
self.grainSize = float(grainSize)/1e6
# convert to kilograms
self.diskMass = float(diskMass)*5.9742e24
self.powerLaw = float(powerLaw)
self.grainEfficiency = float(grainEfficiency)
self.beltMass = float(beltMass)*5.9742e24
self.n_asteroids = self.beltMass/self.M_aster
# calculate zeta(grainEfficency + 4)
argument = self.grainEfficiency + 4
self.zeta = 0
for n in range(1, 100):
self.zeta += 1.0/(n**argument)
# calculate the surface density at 100 AU (sigma_100)
j = 2 - self.powerLaw
m = (self.outerRadius**j) - (self.innerRadius**j)
self.surfaceSigma = self.diskMass*j/(2*math.pi*((100*1.496e11)**self.powerLaw)*m)
self.grainMass = self.grainDensity*(4.0/3.0*math.pi*(self.grainSize**3))
"""
gets the inner and outer radii in AU
"""
def getOuterRadius(self):
return self.outerRadius/1.496e11
def getInnerRadius(self):
return self.innerRadius/1.496e11
"""
gets the star distance in parsecs
"""
def getStarDistance(self):
return self.starDistance/3.08568e16
"""
converts a list of meters into microns
"""
def convertToMicrons(self, lst):
return [x*1e6 for x in lst]
# TODO: make sure this is the same as calculate flux
"""
Takes a radius (in AU) and frequency (in GHz)
Returns the point flux at that radius and frequency in Jansky's
"""
def calculatePointFlux(self, radius, frequency):
radius = float(radius)*1.496e11
# make sure that we are inside of the ring
if radius > self.outerRadius or radius < self.innerRadius:
return 0
# convert frequency to GHz
lamma = self.c_const/(frequency*1e9)
#Integrate over grain sizes
fluxIntegral = integrate.quad(lambda size: (size**2)*(1-math.exp(-(2*math.pi*size/lamma)**self.grainEfficiency))*\
self.calculateGrainSizeDistribution(radius, size)*self.calculateGrainBlackbody(radius, lamma, size),\
self.grainSize, self.grainSize_max)[0]
flux = 1e26/(self.starDistance**2)*fluxIntegral
return flux
"""
input lambda wavelength in meters
return nu*B_nu(lambda) in Jansky*Hz
"""
def calculateFlux(self, lamma):
# integrate returns a list of integral value and error, we only want value
def radIntegral(lamma, size):
return integrate.quad(lambda radius: radius*self.calculateGrainBlackbody(radius, lamma, size)*self.calculateGrainSizeDistribution(radius, size),\
self.innerRadius, self.outerRadius)[0]
fluxIntegral = integrate.quad(lambda size: radIntegral(lamma, size)*size**2*(1-math.exp(-(2*math.pi*size/lamma)**self.grainEfficiency)),\
self.grainSize, self.grainSize_max)[0]
# scale by nu
nu = self.c_const/lamma
flux = nu*2*math.pi*1e26/(self.starDistance**2)*fluxIntegral
return flux
"""
returns B_nu(T) in Janskys ( = 10^-26 * Watts / m^2 / Hz )
"""
def calculateGrainBlackbody(self, radius, lamma, size):
try:
nu = self.c_const/lamma
exponent = self.h_const*nu/(self.k_const*self.calculateGrainTemperature(radius, size))
numerator = 2*self.h_const*(nu**3)*math.pi # THIS PI IS ADDED BY ANGELO FOR SOLID ANGLE!
denominator = (self.c_const**2)*(math.e**exponent - 1)
grainBlackbody = numerator/denominator
except OverflowError:
return 0
return grainBlackbody
"""
return the temperature of a grain, in Kelvin, at a given radius
"""
def calculateGrainTemperature(self, radius, size):
lambdaNought = 2*math.pi*size
expr_1 = self.stefbolt_const*((lambdaNought*self.k_const/(self.h_const*self.c_const))**self.grainEfficiency)
# compute factorial(grainEfficiency+3) = gamma(grainEfficiency+4)
expr_2 = gamma(self.grainEfficiency + 4)
numerator = self.starLuminosity*(math.pi**3)
denominator = 240*(radius**2)*expr_1*expr_2*self.zeta
grainTemperature = (numerator/denominator)**(1/(self.grainEfficiency+4))
return grainTemperature
"""
returns nu*B_nu(lambda) in Jansky*Hz of the host star
"""
def calculateStarBlackbody(self, lamma):
nu = self.c_const/lamma
exponent = self.h_const*nu/(self.k_const*self.starTemperature)
numerator = 2*self.h_const*(nu**3)
denominator = (self.c_const**2)*(math.exp(exponent)-1)
nu = self.c_const/lamma
# not sure what to scale by (4pi?)
return numerator/denominator*nu*1e26*(self.starRadius**2)/(self.starDistance**2)
"""
generates lambda and nu*B_nu values in meters and Janksy*Hz, respectively
"""
def generateModel(self):
self.generateAsteroids()
# sample from .1 microns to 10^4 microns
x = numpy.arange(-7, -2, .01)
x = [10**power for power in x]
y = [self.calculateFlux(lamma) for lamma in x]
self.disk_lambda = self.convertToMicrons(x)
self.model_lambda = self.convertToMicrons(x)
self.disk_flux = y
z = self.asteroid_flux
self.model_flux = map(lambda lamma,flux1,flux2: flux1+flux2+self.interpol_funct(lamma*1e6), x, y, z)
def calculateAsteroidBelt(self, lamma):
try:
nu = self.c_const/lamma
exponent = self.h_const*nu/(self.k_const*self.Temp_a)
numerator = 2*self.h_const*(nu**3)*math.pi*self.n_asteroids #Multiply by number of asteroids
denominator = (self.c_const**2)*(math.e**exponent - 1)
asteroidBlackbody = numerator/denominator*nu*1e26*self.asteroid_radius**2/self.starDistance**2
except OverflowError:
return 0
return asteroidBlackbody
def generateAsteroids(self):
x = numpy.arange(-7, -2, 0.01)
x = [10**power for power in x]
y = [self.calculateAsteroidBelt(lamma) for lamma in x]
self.asteroid_lambda = self.convertToMicrons(x)
self.asteroid_flux = y
def generateInterpolation(self):
# generate interpolated data for the actual data we have
def calculateInterpol(lam):
# for less than 10^-6, ignore disk because so faint
if lam*1e-6 > 1e-6:
return self.calculateFlux(lam*1e-6)+self.interpol_funct(lam)+self.calculateAsteroidBelt(lam*1e-6)
else:
return self.interpol_funct(lam)
self.interpol_flux = map(calculateInterpol, self.sample_lambda)
"""
Computes the surface number density of grains that are of size "size." Replaces calculateGrainDistribution.
"""
def calculateGrainSizeDistribution(self, radius, size):
surfaceMassDensity = self.surfaceSigma*((radius/(100*1.496e11))**(-self.powerLaw))
numerator = surfaceMassDensity*3*(self.grainpowerLaw + 4)*size**(self.grainpowerLaw)
denominator = 4*math.pi*self.grainDensity*(self.grainSize_max**(self.grainpowerLaw + 4) - self.grainSize**(self.grainpowerLaw + 4))
return numerator/denominator
"""
plots the SED of the star and disk
"""
def plotSED(self):
self.generateModel()
self.generateInterpolation()
# plot the observed data
#plt.loglog(self.data_lambda, self.data_flux, label = 'Lejeune Model')
plt.errorbar(self.sample_lambda, self.sample_flux, yerr=self.sample_error, fmt='o', label = 'Observed Data')
# plot the disk model
plt.loglog(self.model_lambda, self.model_flux, label="Best Fit Model")
plt.loglog(self.disk_lambda, self.disk_flux, label="Disk Model")
plt.loglog(self.sample_lambda, self.interpol_flux, "o", label = 'Interpolated Model Data')
plt.loglog(self.IRS_lambda, self.IRS_flux, label = 'IRS Spectrum')
plt.loglog(self.asteroid_lambda, self.asteroid_flux, label='Asteroid Belt')
# format and display the plot
plt.xlabel('$\lambda$ $(\mu m)$')
plt.ylabel(r'$\nu F_\nu$ $(Jy \times Hz)$')
plt.title('SED of HD61005')
plt.legend()
plt.xlim(1e-1,1e4)
plt.ylim(1e6,1e16)
plt.legend(loc='lower left')
plt.show()
"""
computes the chi-squared value of the disk and model data
"""
def computeChiSquared(self):
self.generateInterpolation()
lamma = self.sample_lambda
model_flux = self.interpol_flux
actual_flux = self.sample_flux
error = self.sample_error
chi_squared = 0
numToFit = 0
for i in range(len(lamma)):
# the others are very close and we don't want them to mess up the chi-squared
if lamma[i] > 1e1 and lamma[i] < 1e3:
numToFit += 1
#print 'model:', model_flux[i]
#print 'actual:', actual_flux[i]
chi_squared += ((model_flux[i]-actual_flux[i])/error[i])**2
# degrees of freedom = [number of samples] - [number parameters to fit]
# TEMPO: parameters 6 normally
dof = numToFit - 4
chi_squared = chi_squared#/dof
return chi_squared