Skip to content

Commit 0d9fd26

Browse files
committed
first commit
0 parents  commit 0d9fd26

9 files changed

+352
-0
lines changed

README.md

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
# openfoam_python
2+
Modules for manipulating OpenFOAM runs and other utilities.

__init__.py

Whitespace-only changes.

boundary_conditions.py

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
#!/usr/bin/python
2+
3+
"""
4+
These functions offer easy ways to automate changes in OpenFOAM settings by
5+
modification of 'constant', 'system', '0' files and other files that define an
6+
OpenFOAM case.
7+
"""
8+
9+
import os
10+
import sys
11+
12+
def read_file(file_name):
13+
with open(file_name) as f:
14+
content = f.readlines()
15+
f.close()
16+
return content
17+
18+
def modify_empty_bc(file_path, patch_name, new_bc):
19+
file_contents = read_file(file_path)
20+
section_flag = False
21+
with open(file_path, 'w') as f:
22+
for line in file_contents:
23+
if patch_name in line:
24+
section_flag = True
25+
if section_flag and "type" in line:
26+
line = line.replace("patch", new_bc)
27+
if section_flag and "physicalType" in line:
28+
line = line.replace("patch", new_bc)
29+
if "}" in line and section_flag == True:
30+
section_flag = False
31+
f.write(line)
32+
f.close()
33+
return

clean_times.py

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
#!/usr/bin/python
2+
3+
import os
4+
import sys
5+
6+
def leave_last_time(time_path):
7+
"""
8+
This leaves the latest time in either processor or main case folders.
9+
"""
10+
files = os.listdir(time_path)
11+
12+
num_files = [ \
13+
float(x) for x in files if ('.' in x or x.isdigit()) \
14+
and os.path.isdir(x) \
15+
and (x!="0") ]
16+
num_files = [ int(x) if x.is_integer() else x for x in num_files ]
17+
18+
max_time = 0
19+
if len(num_files) > 1:
20+
max_time = max(num_files)
21+
num_files.remove(max_time)
22+
num_file_names = ' '.join(str(x) for x in num_files)
23+
os.system("rm -r "+num_file_names)
24+
25+
return
26+
27+
print leave_last_time("./")

controlDict.py

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
#!/usr/bin/python
2+
3+
import os
4+
import file_operations
5+
6+
def set_end_time(new_end_time, case_path="./"):
7+
controlDict_path = case_path+"/system/controlDict"
8+
output_file = open(controlDict_path+"_temp", 'w')
9+
with open(controlDict_path, 'r') as input_file:
10+
for line in input_file:
11+
if ("endTime" in line) and ("stopAt" not in line):
12+
time_value = line.split()[1]
13+
new_line = line.replace(time_value, str(new_end_time)+";")
14+
output_file.write(new_line)
15+
else:
16+
output_file.write(line)
17+
output_file.close()
18+
os.system("mv "+controlDict_path+"_temp "+controlDict_path)
19+
return
20+
21+
def set_write_interval(new_write_interval, case_path="./"):
22+
controlDict_path = case_path+"/system/controlDict"
23+
output_file = open(controlDict_path+"_temp", 'w')
24+
with open(controlDict_path, 'r') as input_file:
25+
for line in input_file:
26+
if ("writeInterval" in line):
27+
time_value = line.split()[1]
28+
new_line = line.replace(time_value, str(new_write_interval)+";")
29+
output_file.write(new_line)
30+
else:
31+
output_file.write(line)
32+
output_file.close()
33+
os.system("mv "+controlDict_path+"_temp "+controlDict_path)
34+
return
35+
36+
def set_courant_number(controlDict_path="system/controlDict", maxCo=1):
37+
file_operations.change_line(controlDict_path, ["maxCo"], \
38+
[], "\tmaxCo\t\t"+str(maxCo)+";\n")
39+
return
40+
41+
def set_force_output_interval(controlDict_path="system/controlDict", \
42+
outputInterval=50):
43+
file_operations.change_line(controlDict_path, ["outputInterval"], \
44+
[], "\t\toutputInterval\t\t"+str(outputInterval)+";\n")
45+
return

file_operations.py

+109
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,109 @@
1+
#!/usr/bin/python
2+
3+
import os
4+
5+
def safely_remove_file(path):
6+
if os.path.isfile(path):
7+
os.system("rm "+path)
8+
return
9+
10+
def safely_remove_dir(path):
11+
if os.path.isdir(path):
12+
os.system("rm -r "+path)
13+
return
14+
15+
def change_file_name(file_path, suffix):
16+
# Simply adds a suffix to an existing file.
17+
if os.path.isfile(file_path):
18+
os.system("mv "+file_path+" "+file_path+suffix)
19+
else:
20+
print "Couldn\'t find "+file_path
21+
return
22+
23+
24+
def get_highest_folder(path, prefix):
25+
# Assumes that the folders in path are part of a series with an integer ID
26+
# as the suffix.
27+
# Assumes integer ranks
28+
highest = -1
29+
folders = [x for x in os.listdir(path) if os.path.isdir(path+"/"+x)]
30+
for folder in folders:
31+
folder_id = int(folder.replace(prefix, ""))
32+
if folder_id > highest:
33+
highest = folder_id
34+
return highest
35+
36+
37+
def get_time_folders(case_path="./"):
38+
time_folders = [ x for x in os.listdir(case_path) \
39+
if os.path.isdir(case_path+"/"+x) \
40+
and (("." in x) or (x.isdigit())) \
41+
and x != "0"]
42+
return time_folders
43+
44+
def leave_last_time(case_path="./"):
45+
# if os.path.isdir(case_path+"/processor0"):
46+
# os.system("reconstructPar -latestTime -case "+case_path)
47+
# os.system("rm -r processor*")
48+
time_folders = get_time_folders(case_path)
49+
times = [ float(x) for x in time_folders ]
50+
times = [ int(x) if x.is_integer() else x for x in times ]
51+
times.sort()
52+
time_folders = [ str(x) for x in times ]
53+
# print "Here comes the annoying warning"
54+
if len(time_folders) > 1:
55+
time_folders_string = ' '.join(time_folders[0:-1])
56+
os.system("rm -r "+time_folders_string)
57+
# print "There goes the annoying warning"
58+
return
59+
60+
def clean_case(case_path="./"):
61+
# This removes all non-standard FOLDERS.
62+
current_folders = os.listdir(case_path)
63+
standard_folders = ["0", "system", "constant", "openfoam_python"]
64+
# First make sure this is a case!
65+
for sf in standard_folders:
66+
if sf not in current_folders:
67+
print "This is not an OpenFOAM case directory! Be careful! Exiting."
68+
return
69+
# Then filter out folders that are not standard folders and remove.
70+
folders = [ case_path+"/"+x for x in current_folders \
71+
if os.path.isdir(case_path+"/"+x) and x not in standard_folders ]
72+
if len(folders) > 0:
73+
os.system("rm -r "+' '.join(folders))
74+
# File removal
75+
if os.path.isfile(case_path+"/geometry.dat"):
76+
os.system("rm "+case_path+"/geometry.dat")
77+
if os.path.isdir(case_path+"/constant/polyMesh"):
78+
os.system("rm -r "+case_path+"/constant/polyMesh")
79+
if os.path.isfile(case_path+"/main.msh"):
80+
os.system("rm "+case_path+"/main.msh")
81+
if os.path.isfile(case_path+"/forces.txt"):
82+
os.system("rm "+case_path+"/forces.txt")
83+
# We assume we have reconstructed already.
84+
if os.path.isdir(case_path+"/processor0"):
85+
os.system("rm -r "+case_path+"/processor*")
86+
return
87+
88+
def change_line(file_path, included_strings, excluded_strings, replacement):
89+
"""
90+
Goes through file_path line-by-line and if all of included_strings are in
91+
the line and all of excluded_strings are not in the line, then the whole line
92+
is replaced by replacement.
93+
Assumes file_path+'_temp' is not already a file.
94+
"""
95+
if not os.path.isfile(file_path):
96+
print file_path+" file not found!"
97+
return
98+
temp_path = file_path+"_temp"
99+
temp_file = open(temp_path, 'w')
100+
with open(file_path, 'r') as f:
101+
for line in f:
102+
if all([x in line for x in included_strings]) and \
103+
all([x not in line for x in excluded_strings]):
104+
temp_file.write(replacement)
105+
else:
106+
temp_file.write(line)
107+
temp_file.close()
108+
os.system("mv "+temp_path+" "+file_path)
109+
return

force_read.py

+91
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,91 @@
1+
#!/usr/bin/python
2+
3+
import os
4+
import sys
5+
import math
6+
import numpy
7+
8+
def line2dict(line):
9+
tokens_unprocessed = line.split()
10+
tokens = [x.replace(")","").replace("(","") \
11+
for x in tokens_unprocessed]
12+
floats = [float(x) for x in tokens]
13+
data_dict = {}
14+
data_dict['time'] = floats[0]
15+
force_dict = {}
16+
force_dict['pressure'] = floats[1:4]
17+
force_dict['viscous'] = floats[4:7]
18+
force_dict['porous'] = floats[7:10]
19+
moment_dict = {}
20+
moment_dict['pressure'] = floats[10:13]
21+
moment_dict['viscous'] = floats[13:16]
22+
moment_dict['porous'] = floats[16:19]
23+
data_dict['force'] = force_dict
24+
data_dict['moment'] = moment_dict
25+
return data_dict
26+
27+
def get_forces_dict(forces_file):
28+
# Returns a list of lists: [time, drag, lift, moment]
29+
time = []
30+
drag = []
31+
lift = []
32+
moment = []
33+
with open(forces_file,"r") as datafile:
34+
for line in datafile:
35+
if line[0] == "#":
36+
continue
37+
data_dict = line2dict(line)
38+
time += [data_dict['time']]
39+
drag += [data_dict['force']['pressure'][0] + \
40+
data_dict['force']['viscous'][0]]
41+
lift += [data_dict['force']['pressure'][1] + \
42+
data_dict['force']['viscous'][1]]
43+
moment += [data_dict['moment']['pressure'][2] + \
44+
data_dict['moment']['viscous'][2]]
45+
datafile.close()
46+
return [time, drag, lift, moment]
47+
48+
def get_line_from_file(file_path, line_number):
49+
with open(file_path, "r") as temp_file:
50+
lines = temp_file.readlines()
51+
temp_file.close()
52+
return lines[line_number]
53+
54+
def get_overall_chord():
55+
# Searches file for the chord.
56+
return float(get_line_from_file("geometry.dat", 0))
57+
58+
def get_cell_depth():
59+
# Searches file for the cell_depth.
60+
return float(get_line_from_file("geometry.dat", 1))
61+
62+
def get_V():
63+
# Searches file for the velocity.
64+
# We assume we are in the case directory.
65+
velocity = 0
66+
with open("0/include/initialConditions","r") as ufile:
67+
for line in ufile:
68+
if "flowVelocity" in line:
69+
vector = line[line.find("(")+1:line.find(")")]
70+
velocity = float(vector.split()[0])
71+
ufile.close()
72+
return velocity
73+
74+
def trailing_average(time, data, trailing_time):
75+
end_time = time[-1]
76+
start_time = end_time - trailing_time
77+
diff = numpy.abs( [x-start_time for x in time] )
78+
start_index = diff.argmin()
79+
subdata = data[start_index:len(data)]
80+
return sum(subdata) / len(subdata)
81+
82+
def get_latest_force_time(patch):
83+
# Assumes in main case directory
84+
# Assumes all folders are floating-point named
85+
# Gets folder with highest floating-point value
86+
folders = [ float(x) for x in os.listdir("postProcessing/"+patch) ]
87+
folders.sort()
88+
folder = folders[-1]
89+
if folder.is_integer():
90+
folder = int(folder)
91+
return folder

fvSchemes.py

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
#!/usr/bin/python
2+
3+
import os
4+
import file_operations
5+
6+
def second_order_spatial(fvSchemes_path="system/fvSchemes"):
7+
"""
8+
Assumed format: each setting is on its own line.
9+
"""
10+
file_operations.change_line(fvSchemes_path, ['div', 'phi', 'U'], ['//'], \
11+
"\tdiv(phi,U)\t\tGauss linearUpwind grad(U);\n")
12+
file_operations.change_line(fvSchemes_path, ['div', 'phi', 'k'], ['//'], \
13+
"\tdiv(phi,k)\t\tGauss limitedLinear 1;\n")
14+
file_operations.change_line(fvSchemes_path, ['div', 'phi', 'omega'], \
15+
['//'], "\tdiv(phi,omega)\tGauss limitedLinear 1;\n")
16+
file_operations.change_line(fvSchemes_path, ['div', 'phi', 'nut'], \
17+
['//'], "\tdiv(phi,nut)\tGauss limitedLinear 1;\n")
18+
return
19+
20+
def first_order_spatial(fvSchemes_path="system/fvSchemes"):
21+
"""
22+
Assumed format: each setting is on its own line.
23+
"""
24+
file_operations.change_line(fvSchemes_path, ['div', 'phi', 'U'], ['//'], \
25+
"\tdiv(phi,U)\t\tGauss upwind;\n")
26+
file_operations.change_line(fvSchemes_path, ['div', 'phi', 'k'], ['//'], \
27+
"\tdiv(phi,k)\t\tGauss upwind;\n")
28+
file_operations.change_line(fvSchemes_path, ['div', 'phi', 'omega'], \
29+
['//'], "\tdiv(phi,omega)\tGauss upwind;\n")
30+
file_operations.change_line(fvSchemes_path, ['div', 'phi', 'nut'], \
31+
['//'], "\tdiv(phi,nut)\tGauss upwind;\n")
32+
return
33+

fvSolution.py

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
#!/usr/bin/python
2+
3+
import os
4+
import file_operations
5+
6+
def set_piso_correctors(nCorrectors, fvSolution_path="system/fvSolution"):
7+
"""
8+
Assumed format: each setting is on its own line.
9+
"""
10+
file_operations.change_line(fvSolution_path, ['nCorrectors', ';'], ['//'], \
11+
"\tnCorrectors\t\t"+str(nCorrectors)+";\n")
12+
return

0 commit comments

Comments
 (0)