Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Normalize profiles before fitting piecewise polynomial #205

Merged
merged 1 commit into from
Jan 8, 2025
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
75 changes: 62 additions & 13 deletions CADETProcess/processModel/process.py
Original file line number Diff line number Diff line change
Expand Up @@ -589,15 +589,19 @@ def add_concentration_profile(self, unit, time, c, components=None, s=1e-6):
if max(time) > self.cycle_time:
raise ValueError('Inlet profile exceeds cycle time.')

# Handle components and concentration shape
if components is None:
if c.shape[1] != self.n_comp:
raise ValueError(f'Expected shape ({len(time), self.n_comp}) for concentration array. Got {c.shape}.')
raise ValueError(
f'Expected shape ({len(time), self.n_comp}) for concentration array'
f'. Got {c.shape}.'
)
components = self.component_system.species
elif components == -1:
# Assume same profile for all components.
# Assume same profile for all components
if c.ndim > 1:
raise ValueError('Expected single concentration profile.')
c = np.column_stack([c]*self.n_comp)
c = np.column_stack([c] * self.n_comp)
components = self.component_system.species

if not isinstance(components, list):
Expand All @@ -607,18 +611,41 @@ def add_concentration_profile(self, unit, time, c, components=None, s=1e-6):
if len(indices) == 1 and c.ndim == 1:
c = np.array(c, ndmin=2).T

# Process each component's profile
for i, comp in enumerate(indices):
tck = interpolate.splrep(time, c[:, i], s=s)
# Extract the concentration profile for the current component
profile = c[:, i]

# Normalize the profile
min_val = np.min(profile)
max_val = np.max(profile)
range_val = max_val - min_val

if range_val == 0:
warn(
f'Component {comp} has no variation in concentration; '
'scaling will be skipped.'
)
range_val = 1 # Avoid division by zero, effectively skipping scaling

normalized_profile = (profile - min_val) / range_val

# Fit the spline on the normalized profile
tck = interpolate.splrep(time, normalized_profile, s=s)
ppoly = interpolate.PPoly.from_spline(tck)

for i, (t, sec) in enumerate(zip(ppoly.x, ppoly.c.T)):
if i < 3:
# Add events with properly unscaled coefficients
for j, (t, sec) in enumerate(zip(ppoly.x, ppoly.c.T)):
if j < 3:
continue
elif i > len(ppoly.x) - 5:
elif j > len(ppoly.x) - 5:
continue
evt = self.add_event(
f'{unit}_inlet_{comp}_{i-3}', f'flow_sheet.{unit}.c',
np.flip(sec), t, comp
# Scale the coefficients and adjust the constant term
scaled_sec = sec * range_val
scaled_sec[-1] += min_val # Adjust the constant term for shifting
self.add_event(
f'{unit}_inlet_{comp}_{j-3}', f'flow_sheet.{unit}.c',
np.flip(scaled_sec), t, comp
)

def add_flow_rate_profile(self, unit, time, flow_rate, s=1e-6):
Expand Down Expand Up @@ -652,17 +679,39 @@ def add_flow_rate_profile(self, unit, time, flow_rate, s=1e-6):
if max(time) > self.cycle_time:
raise ValueError('Inlet profile exceeds cycle time')

tck = interpolate.splrep(time, flow_rate, s=s)
# Compute min and max for scaling
min_val = np.min(flow_rate)
max_val = np.max(flow_rate)
range_val = max_val - min_val

if range_val == 0:
warn(
f'Component {comp} has no variation in concentration; '
'scaling will be skipped.'
)
range_val = 1 # Avoid division by zero, effectively skipping scaling

# Normalize flow_rate to [0, 1]
normalized_flow_rate = (flow_rate - min_val) / range_val

# Fit the spline with normalized flow_rate
tck = interpolate.splrep(time, normalized_flow_rate, s=s)
ppoly = interpolate.PPoly.from_spline(tck)

# Add events with unscaled coefficients
for i, (t, sec) in enumerate(zip(ppoly.x, ppoly.c.T)):
if i < 3:
continue
elif i > len(ppoly.x) - 5:
continue
evt = self.add_event(

# Unscale all coefficients
unscaled_sec = sec * range_val
# Adjust the constant term (last coefficient in the array) for shifting
unscaled_sec[-1] += min_val
self.add_event(
f'{unit}_flow_rate_{i-3}', f'flow_sheet.{unit}.flow_rate',
np.flip(sec), t
np.flip(unscaled_sec), t
)

def check_config(self):
Expand Down
Loading