Coverage for rocketcea\blends.py : 0%

Hot-keys on this page
r m x p toggle line displays
j k next/prev highlighted chunk
0 (zero) top of page
1 (one) first highlighted chunk
#!/usr/bin/env python # -*- coding: ascii -*-
""" Handle propellant blends. Make new input cards for various ox and fuel blends. """ from rocketcea.input_cards import oxCards, fuelCards, propCards
def all_in_dict( nameL, D ): """If all members of nameL are keys in D, then return True, otherwise False.""" for name in nameL: if name not in D: return False return True
def get_propellant_name( Name=None, PcentL=None): '''::
#: Return the name of the blend defined by "Name". (string or list of strings) #: Might be a defined blend such as MON25 or FLOX80. #: Might be in the library such as "MMH" or "CLF5" #: Might need to create a long name from percentages Name=["N2H4","NH3"], PcentL=[90,10] #: (if new name, add card to oxCards, fuelCards or propCards) '''
if type(Name)==list:
if all_in_dict( Name, oxCards ): blend_name = newOxBlend(oxL=Name, oxPcentL=PcentL) return blend_name
if all_in_dict( Name, fuelCards ): blend_name = newFuelBlend( fuelL=Name, fuelPcentL=PcentL) return blend_name
raise Exception('Blend Components NOT Recognized: Name=%s, PcentL=%s'%(str(Name), str(PcentL)))
return None else: if (Name in oxCards) or (Name in fuelCards) or (Name in propCards): return Name if is_HYD_Ammonia_Blend( Name ): return Name if isAPeroxide_Blend( Name ): return Name if isAnMMH_N2H4_Blend( Name ): return Name if isMON_Ox_Blend( Name ): return Name if isFLOX_Ox_Blend( Name ): return Name
print('WARNING... Blend Name NOT Recognized "%s"'%str(Name)) return Name
def is_HYD_Ammonia_Blend( name ): """ check if name is for a blend of N2H4 and undissociated Ammonia (UA) HYD40 is in standard deck, however can make HYD30, HYD50 etc. "on-the-fly" """ try: dissPcent = float( name[3:] ) except: return 0 # fails to have an Frac in legal range
if name.upper()[:3] == 'HYD': if dissPcent>=0.0 and dissPcent<=100.0: return 1 return 0
def addHYD_AmmoniaBlend( name, cea_deck ): """ Add new N2H4 + undissociated Ammonia (UA) Blend to the CEA_Obj.cea_deck. Use name like "HYD30" to represent hydrazine with 30% dissociated ammonia. """ try: pcent = float( name[3:] ) x = pcent / 100.0 xUA = (4.0 * (1.0-x)) / 3.0 xN = (2.0 * (2.0*x + 1.0)) / 3.0 xH = (2.0 * 6.0 * x) / 3.0 except: raise Exception('N2H4 un-dissociated Ammonia Blend NOT Recognized: %s'%name)
if (pcent>=0.0) and pcent<=100.0: cardL = [ " name = diss UA %g H %g N %g wt%%= 99.5 "%(xUA, xH, xN), " h,cal=12094. t(k)=298.15 rho,g/cc = 1.004 ", " ", " name = water H 2.0 O 1.0 wt%=0.5 ", " h,cal=-68308. t(k)=298.15 rho,g/cc = 0.9998 ", " ! THIS IS %g%% AMMONIA DISSOCIATION "%(100.0*x,), " omit NH3 "]
if pcent >= 100.0: cardL[0] = " name = diss H 4 N 2 wt%%= 99.5 " cardL.pop() cardL.pop() else: raise Exception('N2H4 un-dissociated Ammonia Blend NOT Recognized: %s'%name)
# put cardL into propCards dictionary propCards[name] = cardL #print(cardL)
# name will be in propCards cea_deck.append( propCards[ name ] ) # --------------------------------------------------------------------------
def isAPeroxide_Blend( name ): """ check if name is for a blend of peroxide and water. Peroxide98 and Peroxide90 are in standard deck, however can make Peroxide95 etc. "on-the-fly" """ if name.lower()[:8]=='peroxide': return 1 else: return 0
def addPeroxideBlend( oxName, cea_deck ): """ Add new Peroxide Blend to the CEA_Obj.cea_deck. Use name like Peroxide95 to represent 95% Peroxide and 5% water. """ try: h2o2Pcent = float( oxName[8:] ) waterPcent = 100.0 - h2o2Pcent except: raise Exception('Peroxide/Water Ox Blend NOT Recognized: %s'%oxName)
if (waterPcent>0.0) and waterPcent<100.0: blendName = newOxBlend(oxL=["H2O2","H2O"], oxPcentL=[h2o2Pcent, waterPcent]) else: raise Exception('Peroxide/Water Ox Blend NOT Recognized: %s'%oxName)
# need to cover both naming conventions for the blend cardL = oxCards[ blendName ] oxCards[oxName] = cardL
# blendName will be in oxCards cea_deck.append( oxCards[ blendName ] ) # --------------------------------------------------------------------------
def isAnMMH_N2H4_Blend( name ): # M20 is in the standard deck # check for an MMH + N2H4 blend """ check if name is for a blend of MMH + N2H4 M20 is in standard deck, however can make M10, M30 etc. "on-the-fly" """
try: mmhPcent = float( name[1:] ) except: return 0 # fails to have an mmhFrac in legal range
if name[0] in ['M','m']: if mmhPcent>0.0 and mmhPcent<100.0: return 1 return 0
def addMMH_N2H4_Blend( fuelName, cea_deck ): """ Add new MMH + N2H4 Blend to the CEA_Obj.cea_deck Use name like M10 to represent 10% MMH and 90% N2H4 """
mmhPcent = float( fuelName[1:] ) # e.g. M10, M15, M23.789
if mmhPcent>0.0 and mmhPcent<100.0: blendName = newFuelBlend( fuelL=["MMH","N2H4"], fuelPcentL=[mmhPcent,100.0-mmhPcent]) else: raise Exception('MMH + N2H4 Blend NOT Recognized: %s'%fuelName)
# need to cover both naming conventions for the blend cardL = fuelCards[ blendName ] fuelCards[fuelName] = cardL
# blendName will be in fuelCards cea_deck.append( fuelCards[ blendName ] )
# --------------------------------------------------------------------------
def isMON_Ox_Blend( name ): """ check for MON oxidizer (MON1 to MON40) 48% is theoretical max MON15 and MON25 are in standard deck, however can make MON20, MON30 etc. "on-the-fly" """
try: noPcent = float( name[3:] ) except: return 0 # fails to have an Frac in legal range
if name[:3] in ['MON','mon']: if noPcent>0.0 and noPcent<40.0: # 48% is theoritical max return 1 return 0
def addMON_Blend( oxName, cea_deck ): """ Add new MON oxidizer Blend to the CEA_Obj.cea_deck MON15 and MON25 are in standard deck, however can make MON20, MON30 etc. "on-the-fly" """ mwN2O3 = 76.01 mwNO = 30.01
try: massNO = float( oxName[3:] )
molesNO = massNO / mwNO molesN2O3 = molesNO massN2O3 = molesN2O3 * mwN2O3 massN2O4 = 100.0 - massN2O3
except: raise Exception('MON Ox Blend NOT Recognized: %s'%oxName)
if (massN2O4>0.0) and (massN2O4<100.0): blendName = newOxBlend(oxL=['N2O4','N2O3'], oxPcentL=[massN2O4, massN2O3]) else: raise Exception('MON Ox Blend NOT Recognized: %s'%oxName)
# need to cover both naming conventions for the blend cardL = oxCards[ blendName ] oxCards[oxName] = cardL
# blendName will be in oxCards cea_deck.append( oxCards[ blendName ] ) # --------------------------------------------------------------------------
def isFLOX_Ox_Blend( name ): """ check for FLOX oxidizer (e.g. FLOX70 or FLOX82.5) No FLOX blends are in standard deck, however can make FLOX70, FLOX82.5 etc. "on-the-fly" """
try: f2Pcent = float( name[4:] ) except: return 0 # fails to have an Frac in legal range
if name[:4] in ['FLOX','flox']: if f2Pcent>0.0 and f2Pcent<100.0: return 1 return 0
def addFLOX_Blend( oxName, cea_deck ): # add FLOX oxidizer (e.g. FLOX70 or FLOX82.5) """ Add new add FLOX oxidizer to the CEA_Obj.cea_deck FLOX70, for example, represents 70% F2 and 30% LOX """
try: f2Pcent = float( oxName[4:] ) o2Pcent = 100.0 - f2Pcent
except: raise Exception('FLOX Ox Blend NOT Recognized: %s'%oxName)
if f2Pcent>0.0: blendName = newOxBlend(oxL=['F2','O2'], oxPcentL=[f2Pcent, o2Pcent]) else: blendName = 'O2'
# need to cover both naming conventions for the blend cardL = oxCards[ blendName ] oxCards[oxName] = cardL
# blendName will be in oxCards cea_deck.append( oxCards[ blendName ] )
# --------------------------------------------------------------------------
def tightenUpEquals( card ): """make sure there's no spaces around equal signs.""" c = None while 1: c = card.replace(' =','=') c = c.replace('= ','=') if c==card: break card = c return c
def giveCardNewHfAndTref( card, newHfCalPerMole, newTrefDegR ): """Change the values of "h,cal" and "t(k)" on propellant cards. """ c = tightenUpEquals( card ) cNew = '' spL = c.split() TrefDegK = newTrefDegR / 1.8 for sp in spL: speL = sp.split('=') if len(speL)==2: if speL[0]=='h,cal': speL[1]='%.1f'%newHfCalPerMole if speL[0]=='t(k)': speL[1]='%.2f'%TrefDegK s = '='.join(speL) else: s = sp cNew += ' ' + s
return cNew + ' '
def giveCardMassPercent( card, fuelPcent ): """set the value of mass percentage (wt%) on propellant card.""" c = tightenUpEquals( card ) cNew = '' spL = c.split() for sp in spL: speL = sp.split('=') if len(speL)==2: if speL[0]=='wt%': speL[1]='%g'%fuelPcent s = '='.join(speL) else: s = sp cNew += ' ' + s
#print( 'cNew =',cNew ) return cNew + ' '
def newOxBlend( oxL=None, oxPcentL=None): ''' create ox blends such as MON25 given the oxidizer names and weight percentages. e.g. oxL=["N2O4","N2O3"], oxPcentL=[36.67,63.33] ''' if 1:#try: newName = '' newCardL = [] for i,name in enumerate(oxL): newName += name + '_%g'%oxPcentL[i] if i<len(oxL)-1: newName += '_'
if len(oxCards[ name ])>2: raise Exception('ERROR... can NOT specify ox Blend for multi propellant card= ' + name)
cardL = oxCards[ name ]
for card in cardL: if float( oxPcentL[i] ) > 0.0: newCardL.append( giveCardMassPercent( card, oxPcentL[i] ) )
#newCardL.extend( cardL )
oxCards[newName] = newCardL
#print( 'new cards for newName=',newName ) #for card in newCardL: # print( card ) #print()
return newName else:#except: raise Exception('ERROR... in newOxBlend')
def newFuelBlend( fuelL=None, fuelPcentL=None): ''' create fuel blends such as M20 given the fuel names and weight percentages. e.g. fuelL=["MMH","N2H4"], fuelPcentL=[20,80] ''' if 1:#try: newName = '' newCardL = [] for i,name in enumerate(fuelL): newName += name + '_%g'%fuelPcentL[i] if i<len(fuelL)-1: newName += '_'
if len(fuelCards[ name ])>2: raise Exception('ERROR... can NOT specify Fuel Blend for multi propellant card= '+ name)
cardL = fuelCards[ name ]
for card in cardL: if float( fuelPcentL[i] ) > 0.0: newCardL.append( giveCardMassPercent( card, fuelPcentL[i] ) )
#newCardL.extend( cardL )
fuelCards[newName] = newCardL
#print( 'new cards for newName=',newName ) #for card in newCardL: # print( card ) #print()
return newName else:#except: raise Exception('ERROR... in newFuelBlend')
def newPropWithNewState( cardDict, name, newHfCalPerMole, newTrefDegR): ''' Take name as it exists in fuelCards or oxCards and change Hf and Tref to be the input values newHfCalPerMole and newTrefDegR ''' try: suffix = '_%g_%g'%(newHfCalPerMole, newTrefDegR) suffix = suffix.replace('-','m') newName = name + suffix
if newName not in cardDict: cardL = cardDict[ name ] if len(cardL)>2: raise Exception('ERROR... can NOT specify new Hf, Tref for multi propellant card: '+ name) #return name newCardL = [] for card in cardL: newCardL.append( giveCardNewHfAndTref( card, newHfCalPerMole, newTrefDegR ) )
cardDict[newName] = newCardL
return newName except:
raise Exception('ERROR... in newPropWithNewState')
def newFuelWithNewState( name, newHfCalPerMole, newTrefDegR): ''' Take name as it exists in fuelCards and change Hf and Tref to be the input values newHfCalPerMole and newTrefDegR ''' return newPropWithNewState( fuelCards, name, newHfCalPerMole, newTrefDegR)
def newOxWithNewState( name, newHfCalPerMole, newTrefDegR): ''' Take name as it exists in oxCards and change Hf and Tref to be the input values newHfCalPerMole and newTrefDegR ''' return newPropWithNewState( oxCards, name, newHfCalPerMole, newTrefDegR)
def turnCardsIntoTokenL( cardL ): """turn the card list into one long list of tokens""" tokenL = [] for card in cardL: c = card.replace('=',' ') spL = c.split() for s in spL: if s: tokenL.append(s) return tokenL
def getFloatTokenFromCards( cardL, token='t(k)' ): """Get the float value of the desired token. (e.g. 't(k)' )""" tokenL = turnCardsIntoTokenL( cardL ) #print( tokenL ) for i,tok in enumerate(tokenL): if tok.lower()==token: try: return float( tokenL[i+1] ) except: return None return None
def getFuelRefTempDegK( name ): """Get the fuel float value of the reference temperature 't(k)'""" try: cardL = fuelCards[ name ] return getFloatTokenFromCards( cardL, 't(k)' ) except: return None
def getPropRefTempDegK( name ): """Get the propellant float value of the reference temperature 't(k)'""" try: cardL = propCards[ name ] return getFloatTokenFromCards( cardL, 't(k)' ) except: return None
def getOxRefTempDegK( name ): """Get the oxidizer float value of the reference temperature 't(k)'""" try: cardL = oxCards[ name ] return getFloatTokenFromCards( cardL, 't(k)' ) except: return None
def getFuelHfCalPerMole( name ): """Get the fuel float value of the reference temperature 'h,cal'""" try: cardL = fuelCards[ name ] return getFloatTokenFromCards( cardL, 'h,cal' ) except: return None
def getOxHfCalPerMole( name ): """Get the ox float value of the reference temperature 'h,cal'""" try: cardL = oxCards[ name ] return getFloatTokenFromCards( cardL, 'h,cal' ) except: raise Exception('Could NOT find "h,cal" in getOxHfCalPerMole for %s'%name)
def getPropHfCalPerMole( name ): """Get the propellant float value of the reference temperature 'h,cal'""" try: cardL = propCards[ name ] return getFloatTokenFromCards( cardL, 'h,cal' ) except: return None
def makeCardForNewTemperature( ceaName='CH4', newTdegR=536.0, CpAve=0.791, MolWt=16.04 ): """ Create a new propellant card that reflects a change in temperature of the propellant from the original reference temperature on the original card to the new input value of temperature, newTdegR.
CpAve = BTU/lbm degR """
if ceaName in oxCards: cardD = oxCards prop_type = 'ox' elif ceaName in fuelCards: cardD = fuelCards prop_type = 'fuel' elif ceaName in propCards: cardD = propCards prop_type = 'prop' else: raise Exception('Could NOT find ceaName=%s in makeCardForNewTemperature'%ceaName)
# find Hf and Tref on card cardL = cardD[ceaName] #print( cardL )
# make one big line and look for Hf and Tref line = ' '.join(cardL) line = line.replace('=',' ') sL = line.split() Hf, TrefR = None, None for i,s in enumerate(sL): token = s.strip() if token=='h,cal': Hf = float( sL[i+1] ) if token=='t(k)': TrefR = float( sL[i+1] ) * 1.8
if Hf==None or TrefR==None: raise Exception('Did NOT find Heat of Formation and/or Reference Temperature in makeCardForNewTemperature')
#print( 'Hf=',Hf,' TrefR=',TrefR )
HrefCalPerMole = Hf #print( 'HrefCalPerMole=',HrefCalPerMole )
dTdegR = newTdegR - TrefR newTrefDegR = newTdegR deltaH = dTdegR * CpAve delCALperMOLE = deltaH * MolWt / 1.8
#print( 'deltaH=',deltaH,' delCALperMOLE=',delCALperMOLE )
HfNew = Hf + delCALperMOLE #print( 'HfNew=',HfNew )
if prop_type=='fuel': newName = newFuelWithNewState(ceaName, HfNew, newTdegR) elif prop_type=='ox': newName = newOxWithNewState(ceaName, HfNew, newTdegR) elif prop_type=='prop': newName = newPropWithNewState(ceaName, HfNew, newTdegR) else: # should never get here raise Exception('Could NOT find prop_type of ceaName=%s in makeCardForNewTemperature'%ceaName)
return newName
def renamePropIfNewHfOrTrefInName( cardDict, name ): '''Look for "h,cal OR "t(k)"" in name. If present, then create new modified name and create new card in cardDict if necessary
for example to tweak LH2 run might look like: "LH2 h,cal=-2155.0 t(k)=21.0" '''
if name in cardDict: # if name in dictionary already, simply return it return name
# w/o any equals sign, assume that the name is unchanged if name.find("=") == -1: return name
spL = name.split() #check for extra entries if len(spL)>1: dictName = spL[0] if dictName in cardDict: n = tightenUpEquals( name ) #print('tightenUpEquals( name )',n) nL = n.split() if len(nL) != 3: s = 'ERROR in adjusted propellant format:\n "%s"\n'%name +\ ' Should be: "LH2 h,cal=-2155.0 t(k)=21.0"' raise Exception(s) #return name
try: valD = {} for s in nL[1:]: eL = s.split('=') valD[ eL[0].lower() ] = float( eL[1] )
newHfCalPerMole, newTrefDegR = valD['h,cal'], valD['t(k)'] newName = newPropWithNewState( cardDict, dictName, newHfCalPerMole, newTrefDegR) return newName except: s = 'ERROR in adjusted propellant format:\n "%s"\n'%name +\ ' Should be: "LH2 h,cal=-2155.0 t(k)=21.0"' raise Exception(s) #return name
# if all else fails, simply return input name return name
if __name__ == '__main__':
for name in ['FLOX80', 'MON15', 'M13', 'Peroxide88']: L = []
if isAPeroxide_Blend( name ): addPeroxideBlend( name, L )
elif isMON_Ox_Blend( name ): addMON_Blend(name, L)
elif isFLOX_Ox_Blend( name ): addFLOX_Blend( name, L )
elif isAnMMH_N2H4_Blend( name ): addMMH_N2H4_Blend( name, L )
print( name ) print( L ) print( '-'*55 )
|