Source code for auto2.continuations.fixed_points

"""

    Fixed Point Continuation class definition
    =========================================

    This module implements the fixed point continuation methods.

"""

import os
import sys
import warnings
import logging
import traceback
import glob

logger = logging.getLogger('logger')

try:
    auto_directory = os.environ['AUTO_DIR']

    for path in sys.path:
        if auto_directory in path:
            break
    else:
        # sys.path.append(auto_directory + '/python/auto')
        sys.path.append(auto_directory + '/python')
except KeyError:
    logger.warning('Unable to find auto directory environment variable.')

import auto.AUTOCommands as ac
import auto.runAUTO as ra
from auto.parseS import AUTOSolution
from auto.AUTOExceptions import AUTORuntimeError
from auto2.continuations.base import Continuation

import auto2.continuations.periodic_orbits as poc


[docs] class FixedPointContinuation(Continuation): """Class for the fixed point continuations in auto-AUTO. Parameters ---------- model_name: str The name of the model to load. Used to load the .f90 file of the provided model. config_object: ~auto2.parsers.config.ConfigParser A loaded ConfigParser object. path_name: str, optional The directory path where files are read/saved. If `None`, defaults to current working directory. Attributes ---------- model_name: str The name of the loaded model. Used to load the .f90 file of the provided model. config_object: ~auto2.parsers.config.ConfigParser A loaded ConfigParser object. continuation: dict Dictionary holding the forward and backward continuation data. branch_number: int The AUTO branch number attributed to the continuation(s). This number can be set manually by passing the IBR AUTO parameter when starting the continuations (see the :meth:`make_continuation` documentation for more details). initial_data: ~numpy.ndarray or AUTO solution object The initial data used to start the continuation(s). auto_filename_suffix: str Suffix for the |AUTO| files used to save the continuation(s) data and parameters on disk. """ def __init__(self, model_name, config_object, path_name=None): Continuation.__init__(self, model_name, config_object, path_name) # plots default behaviours self._default_marker = 'x' self._default_markersize = 6. self._default_linestyle = ' ' self._default_linewidth = 1.2
[docs] def make_continuation(self, initial_data, auto_suffix="", only_forward=False, **continuation_kwargs): """Make both forward and backward (if possible) continuation of fixed points. Parameters ---------- initial_data: ~numpy.ndarray or AUTOSolution Initial data used to start the continuation(s). auto_suffix: str, optional Suffix to use for the |AUTO| and Pickle files used to save the continuation(s) data, parameters and metadata on disk. If not provided, does not save the data on disk. only_forward: bool, optional If `True`, compute only the forward continuation (positive `DS` parameter). If `False`, compute in both backward and forward direction. Default to `False`. continuation_kwargs: dict Keyword arguments to be passed to the |AUTO| continuation. See below for further details. Notes ----- **AUTO Continuation Keyword Arguments**: See Section 10.8 in the |AUTO| documentation for further details. We provide below the most important ones: Other Parameters ---------------- DS: float, optional AUTO uses pseudo-arclength continuation for following solution families. The pseudo-arclength stepsize is the distance between the current solution and the next solution on a family. By default, this distance includes all state variables (or state functions) and all free parameters. The constant `DS` defines the pseudo-arclength stepsize to be used for the first attempted step along any family. DS may be chosen positive or negative; changing its sign reverses the direction of computation. The relation `DSMIN` ≤ | `DS` | ≤ `DSMAX` must be satisfied. The precise choice of `DS` is problem-dependent. DSMIN: float, optional This is minimum allowable absolute value of the pseudo-arclength stepsize. `DSMIN` must be positive. It is only effective if the pseudo-arclength step is adaptive, i.e., if `IADS`>0. The choice of `DSMIN` is highly problem-dependent. DSMAX: float, optional The maximum allowable absolute value of the pseudo-arclength stepsize. `DSMAX` must be positive. It is only effective if the pseudo-arclength step is adaptive, i.e., if `IADS`>0. The choice of `DSMAX` is highly problem-dependent. NMX: int, optional The maximum number of steps to be taken along the branch. IBR: int, optional This constant specifies the initial branch number BR that is used. The default `IBR=0` means that that this number is determined automatically. ILP: int, optional * If `ILP=0`: No detection of folds. This is the recommended choice in the AUTO documentation. * If `ILP=1`: Detection of folds. To be used if subsequent fold continuation is intended. SP: list(str), optional This constant controls the detection of bifurcations and adds stopping conditions. It is specified as a list of bifurcation type strings followed by an optional number. If this number is `0`, then the detection of this bifurcation is turned off, and if it is missing then the detection is turned on. A number `n` greater than zero specifies that the continuation should stop as soon as the nth bifurcation of this type has been reached. Examples: - `SP=[’LP0’]` turn off detection of folds. - `SP=[’LP’,’HB3’,’BP0’,’UZ3’]` turn on the detection of folds and Hopf bifurcations, turn off detection of branch points and stop at the third Hopf bifurcation or third user defined point, whichever comes first. ISP: int, optional This constant controls the detection of Hopf bifurcations, branch points, period-doubling bifurcations, and torus bifurcations: * If `ISP=0` This setting disables the detection of Hopf bifurcations, branch points, period doubling bifurcations, and torus bifurcations and the computation of Floquet multipliers. * If `ISP=1` Branch points and Hopf bifurcations are detected for algebraic equations. Branch points, period-doubling bifurcations and torus bifurcations are not detected for periodic solutions and boundary value problems. However, Floquet multipliers are computed. * If `ISP=2` This setting enables the detection of all special solutions. For periodic solutions and rotations, the choice `ISP=2` should be used with care, due to potential inaccuracy in the computation of the linearized Poincar ́e map and possible rapid variation of the Floquet multipliers. * If `ISP=3` Hopf bifurcations will not be detected. Branch points will be detected, and AUTO will monitor the Floquet multipliers. Period-doubling and torus bifurcations will go undetected. This option is useful for certain problems with non-generic Floquet behavior. * If `ISP=4` Branch points and Hopf bifurcations are detected for algebraic equations. Branch points are not detected for periodic solutions and boundary value problems. AUTO will monitor the Floquet multipliers, and period-doubling and torus bifurcations will be detected. ISW: int, optional This constant controls branch switching at branch points for the case of differential equations. Note that branch switching is automatic for algebraic equations. * If `ISW=1` This is the normal value of `ISW`. * If `ISW=-1` If `IRS` is the label of a branch point or a period-doubling bifurcation then branch switching will be done. For period doubling bifurcations it is recommended that `NTST` be increased. * If `ISW=2` If IRS is the label of a fold, a Hopf bifurcation point, a period-doubling, a torus bifurcation, or, in a non-generic (symmetric) system, a branch point then a locus of such points will be computed. * If `ISW=3` If `IRS` is the label of a branch point in a generic (non-symmetric) system then a locus of such points will be computed. Two additional free parameters must be specified for such continuations MXBF: int, optional This constant, which is effective for algebraic problems only, sets the maximum number of bifurcations to be treated. Additional branch points will be noted, but the corresponding bifurcating families will not be computed. dat: str, optional This constant, where `dat=’filename’`, sets the name of a user-supplied ASCII data file `filename.dat`, from which the continuation is to be restarted. The first column in the data file denotes the time, which does not need to be rescaled to the interval `[0, 1]`, and further columns the coordinates of the solution. The parameter `IRS` must be set to `0`. PAR: dict, optional Determines the parameter values for the continuation to start from. Should be entred as `{'parameter_name': parameter_value}`. IRS: int or str, optional This constant sets the label of the solution where the computation is to be restarted. Setting `IRS=0` is typically used in the first run of a new problem. To restart the computation at the `n`-th label, use `IRS=n`. To restart the computation at a specific label (for example `HB12`), use `IRS=HB12`. TY: str, optional This constant modifies the type from the restart solution. This is sometimes useful in conservative or extended systems, declaring a regular point to be a Hopf bifurcation point `TY=’HB’` or a branch point `TY=’BP’`. IPS: int, optional This constant defines the problem type: * If `IPS=0` algebraic bifurcation problem. * If `IPS=1` stationary solutions of ODEs with detection of Hopf bifurcations. * If `IPS=-1` fixed points of the discrete dynamical systems. * If `IPS=-2` time integration using implicit Euler. * If `IPS=2` computation of periodic solutions. * If `IPS=4` boundary value problems. * If `IPS=5` algebraic optimization problems. * If `IPS=7` boundary value problem with computation of Floquet multipliers. * If `IPS=9` option is used in connection with the HomCont algorithms * If `IPS=11` spatially uniform solutions of a system of parabolic PDEs, with detection of traveling wave bifurcations. * If `IPS=12` continuation of traveling wave solutions to a system of parabolic PDEs. * If `IPS=14` time evolution for a system of parabolic PDEs subject to periodic boundary conditions. """ self.make_forward_continuation(initial_data, "", **continuation_kwargs) if not only_forward: self.make_backward_continuation(initial_data, "", **continuation_kwargs) if auto_suffix: self.auto_save(auto_suffix)
[docs] def make_forward_continuation(self, initial_data, auto_suffix="", **continuation_kwargs): """Make the forward continuation of fixed points. Parameters ---------- initial_data: ~numpy.ndarray or AUTOSolution Initial data used to start the continuation(s). auto_suffix: str, optional Suffix to use for the |AUTO| and Pickle files used to save the continuation(s) data, parameters and metadata on disk. If not provided, does not save the data on disk. continuation_kwargs: dict Keyword arguments to be passed to the |AUTO| continuation. See below for further details Notes ----- **AUTO Continuation Keyword Arguments**: See Section 10.8 in the |AUTO| documentation for further details. The most important ones are provided in the documentation of the :meth:`make_continuation` method. """ runner = ra.runAUTO() ac.load(self.model_name, runner=runner) if 'MXBF' in continuation_kwargs: warnings.warn('Disabling automatic continuation of branch points (MXBF set to 0)') continuation_kwargs['MXBF'] = 0 if 'IBR' not in continuation_kwargs and self.branch_number is not None: continuation_kwargs['IBR'] = self.branch_number self.initial_data = initial_data if isinstance(initial_data, AUTOSolution): for retry in range(self._retry): try: cf = ac.run(initial_data, runner=runner, **continuation_kwargs) except AUTORuntimeError: print(traceback.format_exc()) warnings.warn('AUTO continuation failed, possibly retrying.') else: break else: warnings.warn('Problem to complete the forward AUTO continuation, returning nothing.') cf = None else: u = {i + 1: initial_data[i] for i in range(self.config_object.ndim)} for retry in range(self._retry): try: cf = ac.run(self.model_name, U=u, runner=runner, **continuation_kwargs) except AUTORuntimeError: print(traceback.format_exc()) warnings.warn('AUTO continuation failed, possibly retrying.') else: break else: warnings.warn('Problem to complete the forward AUTO continuation, returning nothing.') cf = None if not self.continuation: self.continuation['backward'] = None self.continuation['forward'] = cf if self.branch_number is None: self.branch_number = abs(self.continuation['forward'].data[0].BR) if auto_suffix: self.auto_save(auto_suffix)
[docs] def make_backward_continuation(self, initial_data, auto_suffix="", **continuation_kwargs): """Make the backward continuation of fixed points. Parameters ---------- initial_data: ~numpy.ndarray or AUTOSolution Initial data used to start the continuation(s). auto_suffix: str, optional Suffix to use for the |AUTO| and Pickle files used to save the continuation(s) data, parameters and metadata on disk. If not provided, does not save the data on disk. continuation_kwargs: dict Keyword arguments to be passed to the |AUTO| continuation. See below for further details Notes ----- **AUTO Continuation Keyword Arguments**: See Section 10.8 in the |AUTO| documentation for further details. The most important ones are provided in the documentation of the :meth:`make_continuation` method. """ runner = ra.runAUTO() ac.load(self.model_name, runner=runner) if 'MXBF' in continuation_kwargs: warnings.warn('Disabling automatic continuation of branch points (MXBF set to 0)') continuation_kwargs['MXBF'] = 0 if 'IBR' not in continuation_kwargs and self.branch_number is not None: continuation_kwargs['IBR'] = self.branch_number self.initial_data = initial_data if isinstance(initial_data, AUTOSolution): if 'DS' in continuation_kwargs: continuation_kwargs['DS'] = - continuation_kwargs['DS'] for retry in range(self._retry): try: cb = ac.run(initial_data, runner=runner, **continuation_kwargs) except AUTORuntimeError: print(traceback.format_exc()) warnings.warn('AUTO continuation failed, possibly retrying.') else: break else: warnings.warn('Problem to complete the backward AUTO continuation, returning nothing.') cb = None else: for retry in range(self._retry): try: cb = ac.run(initial_data, DS='-', runner=runner, **continuation_kwargs) except AUTORuntimeError: print(traceback.format_exc()) warnings.warn('AUTO continuation failed, possibly retrying.') else: break else: warnings.warn('Problem to complete the backward AUTO continuation, returning nothing.') cb = None else: u = {i + 1: initial_data[i] for i in range(self.config_object.ndim)} if 'DS' in continuation_kwargs: continuation_kwargs['DS'] = - continuation_kwargs['DS'] for retry in range(self._retry): try: cb = ac.run(self.model_name, U=u, runner=runner, **continuation_kwargs) except AUTORuntimeError: print(traceback.format_exc()) warnings.warn('AUTO continuation failed, possibly retrying.') else: break else: warnings.warn('Problem to complete the backward AUTO continuation, returning nothing.') cb = None else: for retry in range(self._retry): try: cb = ac.run(self.model_name, DS='-', U=u, runner=runner, **continuation_kwargs) except AUTORuntimeError: print(traceback.format_exc()) warnings.warn('AUTO continuation failed, possibly retrying.') else: break else: warnings.warn('Problem to complete the backward AUTO continuation, returning nothing.') cb = None if not self.continuation: self.continuation['forward'] = None self.continuation['backward'] = cb if self.branch_number is None: self.branch_number = abs(self.continuation['backward'].data[0].BR) if auto_suffix: self.auto_save(auto_suffix)
[docs] def point_stability(self, idx): """Given a point index, returns the stability values (Eiganvalues) for this point. Parameters ---------- idx: str, int |AUTO| index of the point to give the diagnostic from. If `idx`: * is a string, it should identify the label of the point, assuming the point has a label. Backward continuation label must be preceded by a `'-'` character. E.g. `'HB2'` would identify the second Hopf bifurcation point of the forward continuation, while `'-UZ3'` identifies the third user-defined label of the backward branch. * is an integer, it corresponds to the index of the point. Positive integers identify points on the forward continuation, while negative integers identify points on the backward continuation. The zero index identifies the initial point of the continuations. Returns ------- ~numpy.ndarray """ if isinstance(idx, str): if idx[0] == '-': if self.continuation['backward'] is not None: s = self.get_solution_by_label(idx) idx = s['PT'] ix_map = self._solutions_index_map(direction='backward') if idx is not None: return self.continuation['backward'].data[0].diagnostics[ix_map[idx]]['Eigenvalues'] else: warnings.warn('No point stability to show.') return None else: warnings.warn('No backward branch to show the stability for.') return None else: if self.continuation['forward'] is not None: s = self.get_solution_by_label(idx) idx = s['PT'] ix_map = self._solutions_index_map(direction='forward') if idx is not None: return self.continuation['forward'].data[0].diagnostics[ix_map[idx]]['Eigenvalues'] else: warnings.warn('No point stability to show.') return None else: warnings.warn('No forward branch to show the stability for.') return None if idx >= 0: if self.continuation['forward'] is not None: ix_map = self._solutions_index_map(direction='forward') if idx in ix_map: return self.continuation['forward'].data[0].diagnostics[ix_map[idx]]['Eigenvalues'] else: warnings.warn('Point index not found. No point stability to show.') return None else: warnings.warn('No forward branch to show the stability for.') return None else: if self.continuation['backward'] is not None: ix_map = self._solutions_index_map(direction='backward') if -idx in ix_map: return self.continuation['backward'].data[0].diagnostics[ix_map[-idx]]['Eigenvalues'] else: warnings.warn('Point index not found. No point stability to show.') return None else: warnings.warn('No backward branch to show the stability for.') return None
def _set_from_dict(self, state, load_initial_data=True): # store the pathname to pass to the updated class state['_path_name'] = self._path_name self.__dict__.clear() self.__dict__.update(state) if isinstance(self.initial_data, dict) and load_initial_data: branch_number = abs(self.initial_data['BR']) fp_file_list = glob.glob('fp*.pickle') fp_branch_numbers = list(map(lambda filename: int(filename.split('_')[1].split('.')[0]), fp_file_list)) if branch_number in fp_branch_numbers: fp = FixedPointContinuation(model_name=self.model_name, config_object=self.config_object, path_name=self._path_name) try: fp.load('fp_'+str(branch_number)+'.pickle', load_initial_data=False) for s in fp.full_solutions_list: if s['PT'] == self.initial_data['PT']: self.initial_data = s break except FileNotFoundError: warnings.warn('Unable to load initial data. Parent branch was not saved.') self.initial_data = None else: po_file_list = glob.glob('po*.pickle') po_branch_numbers = list(map(lambda s: int(s.split('_')[1].split('.')[0]), po_file_list)) if branch_number in po_branch_numbers: hp = poc.PeriodicOrbitContinuation(model_name=self.model_name, config_object=self.config_object, path_name=self._path_name) try: hp.load('po_' + str(branch_number) + '.pickle', load_initial_data=False) for s in hp.full_solutions_list: if s['PT'] == self.initial_data['PT']: self.initial_data = s break except FileNotFoundError: warnings.warn('Unable to load initial data. Parent branch was not saved.') self.initial_data = None else: warnings.warn('Unable to find initial data.') self.initial_data = None if self.auto_filename_suffix: self.auto_load(self.auto_filename_suffix) else: warnings.warn('No AUTO filename suffix specified. Unable to load data.') @property def isfixedpoint(self): """bool: Whether this class stores fixed point continuations.""" return True @property def isperiodicorbit(self): """bool: Whether this class stores periodic orbit continuations.""" return False