LCOV - code coverage report
Current view: top level - synthesis/ImagerObjects - grpcInteractiveClean.cc (source / functions) Hit Total Coverage
Test: casacpp_coverage.info Lines: 0 1066 0.0 %
Date: 2024-10-10 19:51:30 Functions: 0 61 0.0 %

          Line data    Source code
       1             : #include <synthesis/ImagerObjects/grpcInteractiveClean.h>
       2             : #include <synthesis/ImagerObjects/SIMinorCycleController.h>
       3             : #include <casatools/Config/State.h>
       4             : #include <casacore/casa/Logging/LogIO.h>
       5             : #include <casacore/images/Images/PagedImage.h>
       6             : #include <stdcasa/StdCasa/CasacSupport.h>
       7             : #include <string.h>
       8             : #include <stdlib.h>
       9             : #include <sys/types.h>
      10             : #include <sys/wait.h>
      11             : #include <iostream>
      12             : 
      13             : #include <sys/types.h>
      14             : #include <sys/stat.h>
      15             : #include <unistd.h>
      16             : #include <array>
      17             : #include <regex>
      18             : #include <string>
      19             : 
      20             : #include <algorithm> 
      21             : #include <cctype>
      22             : #include <locale>
      23             : 
      24             : #include <grpc++/grpc++.h>
      25             : #include "shutdown.grpc.pb.h"
      26             : #include "img.grpc.pb.h"
      27             : #include "ping.grpc.pb.h"
      28             : 
      29             : #include <stdcasa/StdCasa/CasacSupport.h>
      30             : 
      31             : #ifdef __APPLE__
      32             : extern "C" char **environ;
      33             : #include <unistd.h>
      34             : #endif
      35             : 
      36             : using namespace casacore;
      37             : 
      38             : // https://stackoverflow.com/questions/216823/whats-the-best-way-to-trim-stdstring
      39             : // C++ is so ridiculous... trim from start (in place)
      40           0 : static inline void ltrim(std::string &s) {
      41           0 :     s.erase(s.begin(), std::find_if(s.begin(), s.end(), [](int ch) {
      42           0 :         return !std::isspace(ch);
      43             :     }));
      44           0 : }
      45             : 
      46             : // trim from end (in place)
      47           0 : static inline void rtrim(std::string &s) {
      48           0 :     s.erase(std::find_if(s.rbegin(), s.rend(), [](int ch) {
      49           0 :         return !std::isspace(ch);
      50           0 :     }).base(), s.end());
      51           0 : }
      52             : 
      53             : // trim from both ends (in place)
      54           0 : static inline void trim(std::string &s) {
      55           0 :     ltrim(s);
      56           0 :     rtrim(s);
      57           0 : }
      58             : 
      59             : // Executes the given program, with the given arguments and the given environment.
      60             : // The stdout from the program is collected and returned in output, up to outputlen characters.
      61             : // @param envp To get around the MPI issue from CAS-13252, this should probably come from getenv_sansmpi().
      62           0 : static void execve_getstdout(char *pathname, char *argv[], char *envp[], char *output, ssize_t outputlen)
      63             : {
      64             :     // We use execve here instead of popen to get around issues related to using MPI.
      65             :     // MPI crashes when starting a process that calls MPI_Init in a process spawned using popen or exec.
      66             :     // We can trick MPI into behaving itself by removing all the MPI environmental variables for
      67             :     // the child precess (thus getenv_sansmpi and execve).
      68             : 
      69             :     int filedes[2];
      70           0 :     if (pipe(filedes) == -1) {
      71           0 :         std::cerr << "pipe error" << std::endl;
      72           0 :         exit(1);
      73             :     }
      74             : 
      75           0 :     pid_t pid = fork();
      76           0 :     if (pid == -1) {
      77           0 :         std::cerr << "fork error" << std::endl;
      78           0 :         exit(1);
      79           0 :     } else if (pid == 0) { // child
      80             :         // close stdout and connect it to the input of the pipe
      81           0 :         while ((dup2(filedes[1], STDOUT_FILENO) == -1) && (errno == EINTR)) {}
      82           0 :         close(filedes[1]);
      83           0 :         close(filedes[0]);
      84             :         // exec on the child process
      85           0 :         execve(pathname, argv, envp);
      86           0 :         exit(1);
      87             :     } else { // parent
      88             :         // don't care about the input end of the pipe
      89           0 :         close(filedes[1]);
      90             : 
      91           0 :         const ssize_t tmplen = 128;
      92             :         char tmp[tmplen];
      93           0 :         ssize_t total = 0;
      94             :         while (1) {
      95           0 :             ssize_t count = read(filedes[0], tmp, tmplen);
      96           0 :             if (count == -1) {
      97           0 :                 if (errno == EINTR) {
      98           0 :                     continue;
      99             :                 } else {
     100           0 :                     std::cerr << "read error" << std::endl;
     101           0 :                     exit(1);
     102             :                 }
     103           0 :             } else if (count == 0) {
     104           0 :                 break;
     105             :             } else {
     106           0 :                 ssize_t remaining = outputlen - total;
     107           0 :                 ssize_t cpysize = (count < remaining) ? count : remaining;
     108           0 :                 memcpy(output+total, tmp, cpysize);
     109           0 :                 total += cpysize;
     110           0 :                 output[total] = '\0';
     111             :             }
     112           0 :         }
     113             : 
     114           0 :         close(filedes[0]);
     115             :     }
     116           0 : }
     117             : 
     118             : // Get all environment parameters (as from the "environ" posix variable),
     119             : // but don't include any environment parameters that match "*MPI*".
     120             : // @return A malloc'ed set of environment parameters. Should call free after use.
     121           0 : static char **getenv_sansmpi()
     122             : {
     123           0 :     int nvars = 0, nvars_sansmpi = 0;
     124           0 :     for (nvars = 0; environ[nvars] != NULL; nvars++) {
     125             :         // printf("%s\n", environ[nvars]);
     126           0 :         std::string envvar = environ[nvars];
     127           0 :         if (envvar.find("MPI") == std::string::npos) {
     128           0 :             nvars_sansmpi++;
     129             :         }
     130           0 :     }
     131             : 
     132           0 :     char **ret = (char**)malloc(sizeof(char*) * (nvars_sansmpi+1));
     133           0 :     int retidx = 0;
     134           0 :     for (int i = 0; environ[i] != NULL; i++) {
     135           0 :         std::string envvar = environ[i];
     136           0 :         if (envvar.find("MPI") == std::string::npos) {
     137           0 :             ret[retidx] = environ[i];
     138           0 :             retidx++;
     139             :         }
     140           0 :     }
     141           0 :     ret[nvars_sansmpi] = NULL;
     142             : 
     143           0 :     return ret;
     144             : }
     145             : 
     146             : namespace casa { //# NAMESPACE CASA - BEGIN
     147             : 
     148           0 :     grpcInteractiveCleanManager &grpcInteractiveClean::getManager( ) {
     149           0 :         static grpcInteractiveCleanManager mgr;
     150           0 :         return mgr;
     151             :     }
     152             : 
     153           0 :         void grpcInteractiveCleanManager::pushDetails() {
     154           0 :         }
     155             : 
     156           0 :     grpcInteractiveCleanState::grpcInteractiveCleanState( ) : SummaryMinor(casacore::IPosition(2, 
     157             :                                                                            SIMinorCycleController::nSummaryFields, // temporary CAS-13683 workaround
     158             :                                                               //                                   SIMinorCycleController::useSmallSummaryminor() ? 6 : SIMinorCycleController::nSummaryFields, // temporary CAS-13683 workaround
     159             :                                                                                                  0)),
     160           0 :                                                               SummaryMajor(casacore::IPosition(1,0)) {
     161           0 :                 LogIO os( LogOrigin("grpcInteractiveCleanState",__FUNCTION__,WHERE) );
     162           0 :         reset( );
     163           0 :     }
     164             : 
     165           0 :     void grpcInteractiveCleanState::reset( ) {
     166           0 :         Niter = 0;
     167           0 :         MajorDone = 0;
     168           0 :         CycleNiter = 0;
     169           0 :         InteractiveNiter = 0;
     170           0 :         Threshold = 0;
     171           0 :         CycleThreshold = 0;
     172           0 :         InteractiveThreshold = 0.0;
     173           0 :         IsCycleThresholdAuto = true;
     174           0 :         IsCycleThresholdMutable = true;
     175           0 :         IsThresholdAuto = false;
     176           0 :         CycleFactor = 1.0;
     177           0 :         LoopGain = 0.1;
     178           0 :         StopFlag = false;
     179           0 :         PauseFlag = false;
     180           0 :         InteractiveMode = false;
     181           0 :         UpdatedModelFlag = false;
     182           0 :         InteractiveIterDone = 0; 
     183           0 :         IterDone = 0;
     184           0 :         StopCode = 0;
     185           0 :         Nsigma = 0.0;
     186           0 :         MaxPsfSidelobe = 0.0;
     187           0 :         MinPsfFraction = 0.05;
     188           0 :         MaxPsfFraction = 0.8;
     189           0 :         PeakResidual = 0.0;
     190           0 :         MinorCyclePeakResidual = 0.0;
     191           0 :         PrevPeakResidual = -1.0;
     192           0 :         NsigmaThreshold = 0.0;
     193           0 :         PrevMajorCycleCount = 0;
     194           0 :         PeakResidualNoMask = 0.0;
     195           0 :         PrevPeakResidualNoMask = -1.0;
     196           0 :         MinPeakResidualNoMask = 1e+9;
     197           0 :         MinPeakResidual = 1e+9;
     198           0 :         MaskSum = -1.0;
     199           0 :         MadRMS = 0.0;
     200             :         //int nSummaryFields = SIMinorCycleController::useSmallSummaryminor() ? 6 : SIMinorCycleController::nSummaryFields; // temporary CAS-13683 workaround
     201           0 :         int nSummaryFields = SIMinorCycleController::nSummaryFields; // temporary CAS-13683 workaround
     202             :         //int nSummaryFields = !FullSummary ? 6 : SIMinorCycleController::nSummaryFields; // temporary CAS-13683 workaround
     203           0 :         SummaryMinor.reformOrResize(casacore::IPosition(2, nSummaryFields ,0));
     204           0 :         SummaryMajor.reformOrResize(casacore::IPosition(1,0));
     205           0 :         SummaryMinor = 0;
     206           0 :         SummaryMajor = 0;
     207           0 :     }
     208             : 
     209           0 :     void grpcInteractiveCleanManager::setControls( int niter, int ncycle, float threshold ) {
     210           0 :         LogIO os( LogOrigin("grpcInteractiveCleanManager", __FUNCTION__, WHERE) );
     211           0 :         static const auto debug = getenv("GRPC_DEBUG");
     212           0 :         if ( debug ) std::cerr << "setting clean controls:";
     213           0 :         access( (void*) 0,
     214           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
     215           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
     216             : 
     217           0 :                            state.Niter = niter;
     218           0 :                            if ( debug ) std::cerr << " niter=" << state.Niter;
     219           0 :                            state.CycleNiter = ncycle;
     220           0 :                            if ( debug ) std::cerr << " cycleniter=" << state.CycleNiter;
     221           0 :                            state.Threshold = threshold;
     222           0 :                            if ( debug ) std::cerr << " threshold=" << state.Threshold;
     223           0 :                            return dummy;
     224             : 
     225             :                        } ) );
     226             : 
     227           0 :         if ( debug ) {
     228           0 :             std::cerr << " (process " << getpid( ) << ", thread " <<
     229           0 :                 std::this_thread::get_id() << ")" << std::endl;
     230           0 :             fflush(stderr);
     231             :         }
     232             : 
     233           0 :     }
     234           0 :         void grpcInteractiveCleanManager::setControlsFromRecord(const casac::record &iterpars) {
     235           0 :                 LogIO os( LogOrigin("grpcInteractiveCleanManager", __FUNCTION__, WHERE) );
     236           0 :         static const auto debug = getenv("GRPC_DEBUG");
     237             : 
     238           0 :         if ( debug ) std::cerr << "initializing clean controls:";
     239             : 
     240           0 :         access( (void*) 0,
     241           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
     242           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
     243             : 
     244           0 :                            auto oldNiter = state.Niter;
     245           0 :                            auto oldCycleNiter = state.CycleNiter;
     246           0 :                            auto oldThreshold = state.Threshold;
     247           0 :                            auto oldCycleThreshold = state.CycleThreshold;
     248             : 
     249           0 :                            state.reset( );
     250             : 
     251             :                            /* Note it is important that niter get set first as we catch
     252             :                               negative values in the cycleniter, and set it equal to niter */
     253           0 :                            auto niter = iterpars.find("niter");
     254           0 :                            if ( niter != iterpars.end( ) ) {
     255           0 :                                state.Niter = niter->second.toInt( );
     256           0 :                                if ( debug ) std::cerr << " niter=" << state.Niter;
     257             :                            }
     258           0 :                            auto newNiter = state.Niter;
     259             : 
     260           0 :                            auto cycleniter = iterpars.find("cycleniter");
     261           0 :                            if ( cycleniter != iterpars.end( ) ) {
     262           0 :                                int val = cycleniter->second.toInt( );
     263           0 :                                if ( val <= 0 )
     264           0 :                                    state.CycleNiter = state.Niter;
     265             :                                else
     266           0 :                                    state.CycleNiter = val;
     267           0 :                                if ( debug ) std::cerr << " cycleniter=" << state.CycleNiter;
     268             :                            }
     269           0 :                            auto newCycleNiter = state.CycleNiter;
     270             : 
     271           0 :                            auto interactiveniter = iterpars.find("interactiveniter");
     272           0 :                            if ( interactiveniter != iterpars.end( ) ) {
     273           0 :                                state.InteractiveNiter = interactiveniter->second.toInt( );
     274           0 :                                if ( debug ) std::cerr << " interactiveniter=" << state.InteractiveNiter;
     275             :                            }
     276             : 
     277           0 :                            auto threshold = iterpars.find("threshold");
     278           0 :                            if ( threshold != iterpars.end( ) ) {
     279           0 :                                auto quant = casaQuantity(threshold->second);
     280           0 :                                if ( quant.getUnit( ) == "" ) quant.setUnit("Jy");
     281           0 :                                auto val = quant.getValue(Unit("Jy"));
     282           0 :                                if ( val == -1.0 ) {
     283           0 :                                    state.Threshold = 0.0;
     284           0 :                                    state.IsThresholdAuto = true;
     285             :                                } else {
     286           0 :                                    state.Threshold = (float) val; 
     287           0 :                                    state.IsThresholdAuto = false;
     288             :                                }
     289           0 :                                if ( debug ) {
     290           0 :                                    std::cerr << " threshold=" << state.Threshold;
     291             :                                    std::cerr << " isthresholdauto=" <<
     292           0 :                                        (state.IsThresholdAuto ? "true" : "false");
     293             :                                }
     294           0 :                            }
     295           0 :                            auto newThreshold = state.Threshold;
     296             : 
     297           0 :                            auto cyclethreshold = iterpars.find("cyclethreshold");
     298           0 :                            if ( cyclethreshold != iterpars.end( ) ) {
     299           0 :                                state.CycleThreshold = casaQuantity(cyclethreshold->second).getValue(Unit("Jy"));
     300           0 :                                state.IsCycleThresholdAuto = false;
     301           0 :                                if ( debug ) {
     302           0 :                                    std::cerr << " cyclethreshold=" << state.CycleThreshold;
     303             :                                    std::cerr << " iscyclethresholdauto=" <<
     304           0 :                                        (state.IsCycleThresholdAuto ? "true" : "false");
     305           0 :                                    fflush(stderr);
     306             :                                }
     307             :                            }
     308           0 :                            auto newCycleThreshold = state.CycleThreshold;
     309             : 
     310           0 :                            auto cyclethresholdismutable = iterpars.find("cyclethresholdismutable");
     311           0 :                            if ( cyclethresholdismutable != iterpars.end( ) ) {
     312           0 :                                state.IsCycleThresholdMutable = cyclethresholdismutable->second.toBool( );
     313           0 :                                state.IsCycleThresholdAuto = state.IsCycleThresholdAuto && state.IsCycleThresholdMutable;
     314           0 :                                if ( debug ) {
     315           0 :                                    std::cerr << " iscyclethresholdmutable=" << (state.IsCycleThresholdMutable ? "true" : "false");
     316           0 :                                    std::cerr << " iscyclethresholdauto=" << (state.IsCycleThresholdAuto ? "true" : "false");
     317           0 :                                    fflush(stderr);
     318             :                                }
     319             :                            }
     320             : 
     321           0 :                            auto interactivethreshold = iterpars.find("interactivethreshold");
     322           0 :                            if ( interactivethreshold != iterpars.end( ) ) {
     323           0 :                                state.InteractiveThreshold = casaQuantity(interactivethreshold->second).getValue(Unit("Jy"));
     324           0 :                                if ( debug ) std::cerr << " interactivethreshold=" << state.InteractiveThreshold;
     325             :                            }
     326             : 
     327           0 :                            auto loopgain = iterpars.find("loopgain");
     328           0 :                            if ( loopgain != iterpars.end( ) ) {
     329           0 :                                state.LoopGain = (float) loopgain->second.toDouble( );
     330           0 :                                if ( debug ) std::cerr << " loopgain=" << state.LoopGain;
     331             :                            }
     332           0 :                            auto cyclefactor = iterpars.find("cyclefactor");
     333           0 :                            if ( cyclefactor != iterpars.end( ) ) {
     334           0 :                                state.CycleFactor = (float) cyclefactor->second.toDouble( );
     335           0 :                                if ( debug ) std::cerr << " cyclefactor=" << state.CycleFactor;
     336             :                            }
     337             : 
     338           0 :                            auto interactivemode = iterpars.find("interactive");
     339           0 :                            if ( interactivemode != iterpars.end( ) ) {
     340           0 :                                state.InteractiveMode = interactivemode->second.toBool( );
     341           0 :                                if ( debug ) std::cerr << " interactive=" <<
     342           0 :                                                 (state.InteractiveMode ? "true" : "false");
     343             :                            }
     344             : 
     345           0 :                            auto minpsffraction = iterpars.find("minpsffraction");
     346           0 :                            if ( minpsffraction != iterpars.end( ) ) {
     347           0 :                                state.MinPsfFraction = (float) minpsffraction->second.toDouble( );
     348           0 :                                if ( debug ) std::cerr << " minpsffraction=" << state.MinPsfFraction;
     349             :                            }
     350             : 
     351           0 :                            auto maxpsffraction = iterpars.find("maxpsffraction");
     352           0 :                            if ( maxpsffraction != iterpars.end( ) ) {
     353           0 :                                state.MaxPsfFraction = (float) maxpsffraction->second.toDouble( );
     354           0 :                                if ( debug ) std::cerr << " maxpsffraction=" << state.MaxPsfFraction;
     355             :                            }
     356             : 
     357           0 :                            auto nsigma = iterpars.find("nsigma");
     358           0 :                            if ( nsigma != iterpars.end( ) ) {
     359           0 :                                state.Nsigma = (float) nsigma->second.toDouble( );
     360           0 :                                if ( debug ) std::cerr << " nsigma=" << state.Nsigma;
     361             :                            }
     362           0 :                            auto fullsummary = iterpars.find("fullsummary");
     363           0 :                            if ( fullsummary != iterpars.end() ) {
     364           0 :                                state.FullSummary = fullsummary->second.toBool();
     365           0 :                                if ( debug ) std::cerr << " fullsummry=" << state.FullSummary;
     366             :                            }
     367           0 :                            if ( debug ) std::cerr << std::endl;
     368             : 
     369           0 :                            if ( debug ) {
     370           0 :                                std::cerr << "-------------------------------------------" << this << " / " << &state << std::endl;
     371           0 :                                std::cerr << "  exported python state: " << std::endl;
     372           0 :                                std::cerr << "-------------------------------------------" << std::endl;
     373           0 :                                std::cerr << "    Niter            " << oldNiter <<
     374           0 :                                    " ---> " << newNiter << std::endl;
     375           0 :                                std::cerr << "    CycleNiter       " << oldCycleNiter <<
     376           0 :                                    " ---> " << newCycleNiter << std::endl;
     377           0 :                                std::cerr << "    Threshold        " << oldThreshold <<
     378           0 :                                    " ---> " << newThreshold << std::endl;
     379           0 :                                std::cerr << "    CycleThreshold   " << oldCycleThreshold <<
     380           0 :                                    " ---> " << newCycleThreshold << std::endl;
     381           0 :                                std::cerr << "    IsCycleThresholdAuto " << state.IsCycleThresholdAuto << std::endl;
     382           0 :                                std::cerr << "-------------------------------------------" << std::endl;
     383             :                            
     384             : 
     385             :                            }
     386           0 :                            return dummy;
     387             : 
     388             :                        } ) );
     389             : 
     390           0 :         if ( debug ) {
     391           0 :             std::cerr << " (process " << getpid( ) << ", thread " << 
     392           0 :                 std::this_thread::get_id() << ")" << std::endl;
     393           0 :             fflush(stderr);
     394             :         }
     395           0 :     }
     396             : /*
     397             :         Float SIIterBot_state::readThreshold( Record recordIn, String id )  {
     398             :                 LogIO os( LogOrigin("SIIterBot_state",__FUNCTION__,WHERE) );
     399             :                 std::lock_guard<std::recursive_mutex> guard(recordMutex);    
     400             :                 // Threshold can be a variant, either Float or String(with units).
     401             :                 Float fthresh=0.0;
     402             :                 // If a number, treat it as a number in units of Jy.
     403             :                 if( recordIn.dataType(id) == TpFloat || 
     404             :                     recordIn.dataType(id) == TpDouble || 
     405             :                     recordIn.dataType(id) == TpInt )
     406             :                   { fthresh = recordIn.asFloat( RecordFieldId(id)); }
     407             :                 // If a string, try to convert to a Quantity
     408             :                 else if( recordIn.dataType(id) == TpString )
     409             :                   {
     410             :                     Quantity thresh; 
     411             :                     // If it cannot be converted to a Quantity.... complain, and use zero.
     412             :                     if( ! casacore::Quantity::read( thresh, recordIn.asString( RecordFieldId(id) ) ) )
     413             :                       {os << LogIO::WARN << "Cannot parse threshold value. Setting to zero." << LogIO::POST;  
     414             :                         fthresh=0.0;}
     415             :                     // If converted to Quantity, get value in Jy. 
     416             :                     // ( Note : This does not check for wrong units, e.g. if the user says '100m' ! )
     417             :                     else { fthresh = thresh.getValue(Unit("Jy")); }
     418             :                   }
     419             :                 // If neither valid datatype, print a warning and use zero.
     420             :                 else {os << LogIO::WARN << id << " is neither a number nor a string Quantity. Setting to zero." << LogIO::POST;
     421             :                   fthresh=0.0; }
     422             : 
     423             :                 return fthresh;
     424             :         }
     425             : */
     426           0 :         void grpcInteractiveCleanManager::setIterationDetails(const casac::record &iterpars) {
     427           0 :                 LogIO os( LogOrigin("grpcInteractiveCleanManager",__FUNCTION__,WHERE) );
     428           0 :         static const auto debug = getenv("GRPC_DEBUG");
     429             : 
     430             :         // Setup interactive masking : list of image names.
     431           0 :         if ( clean_images.size( ) == 0 ) {
     432             :             try {
     433             :                 ////////////////////////////////////////////////////////////////////////////////////////////////////////
     434             :                 ///// START : code to get a list of image names for interactive masking
     435             : 
     436           0 :                 auto allimages = iterpars.find("allimages");
     437           0 :                 if ( allimages != iterpars.end( ) ) {
     438           0 :                     auto rec = allimages->second.getRecord( );
     439           0 :                     for ( auto it = rec.begin( ); it != rec.end( ); ++it ) {
     440           0 :                         auto oneimg = it->second.getRecord( );
     441           0 :                         auto img_name = oneimg.find("imagename");
     442           0 :                         auto img_multiterm = oneimg.find("multiterm");
     443           0 :                         if ( img_name != oneimg.end( ) && img_multiterm != oneimg.end( ) ) {
     444           0 :                             clean_images.push_back( std::make_tuple( img_name->second.getString(),
     445           0 :                                                                      img_multiterm->second.toBool( ), false, 0) );
     446             :                         }
     447           0 :                     }
     448           0 :                 } else {
     449           0 :                     throw( AipsError("Need image names and nterms in iteration parameter list") );
     450             :                 }
     451           0 :                 if ( clean_images.size( ) <= 0 ) {
     452           0 :                     throw( AipsError("Need image names for iteration") );
     453             :                 }
     454             : 
     455           0 :                 if ( debug ) {
     456           0 :                     std::cerr << "clean images specified: ";
     457           0 :                     for ( auto it = clean_images.begin( ); it != clean_images.end( ); ++it ) {
     458           0 :                         if ( it != clean_images.begin( ) ) std::cerr << ", ";
     459           0 :                         std::cerr << std::get<0>(*it) << " [" << (std::get<1>(*it) ? "true" : "false") << "]";
     460             :                     }
     461           0 :                     std::cerr << " (process " << getpid( ) << ", thread " << 
     462           0 :                         std::this_thread::get_id() << ")" << std::endl;
     463           0 :                     fflush(stderr);
     464             :                 }
     465             : 
     466             :                 ///// END : code to get a list of image names for interactive masking
     467             :                 ////////////////////////////////////////////////////////////////////////////////////////////////////////
     468             : 
     469           0 :                 setControlsFromRecord( iterpars );
     470           0 :                 Int nSummaryFields = !state.FullSummary ? 6 : SIMinorCycleController::nSummaryFields;
     471           0 :                 state.SummaryMinor.reformOrResize(casacore::IPosition(2, nSummaryFields ,0));
     472             : 
     473           0 :             } catch( AipsError &x ) {
     474           0 :                 throw( AipsError("Error in updating iteration parameters : " + x.getMesg()) );
     475           0 :             }
     476             :         }
     477           0 :         }
     478             : 
     479           0 :         void grpcInteractiveCleanManager::updateCycleThreshold( grpcInteractiveCleanState &state ) {
     480           0 :                 static const auto debug = getenv("GRPC_DEBUG");
     481             : 
     482           0 :                 Float psffraction = state.MaxPsfSidelobe * state.CycleFactor;
     483             : 
     484           0 :                 psffraction = max(psffraction, state.MinPsfFraction);
     485           0 :                 psffraction = min(psffraction, state.MaxPsfFraction);
     486             : 
     487           0 :                 if ( debug ) {
     488           0 :                         std::cerr << "------------------------------------------- " << this << " / " << &state << std::endl;
     489           0 :                         std::cerr << "  algorithmic update of cycle threshold: " << std::endl;
     490           0 :                         std::cerr << "    CycleThreshold   " << state.CycleThreshold <<
     491           0 :                             " ---> " << (state.PeakResidual * psffraction) << std::endl;
     492           0 :                         std::cerr << "    IsCycleThresholdAuto " << state.IsCycleThresholdAuto << std::endl;
     493           0 :                         std::cerr << "-------------------------------------------" << std::endl;
     494             :                 }
     495             : 
     496           0 :                 state.CycleThreshold = state.PeakResidual * psffraction;
     497           0 :                 pushDetails();
     498           0 :         }
     499             : 
     500           0 :         void grpcInteractiveCleanManager::addSummaryMajor( ) {
     501           0 :         access( (void*) 0,
     502           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
     503           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
     504           0 :                            IPosition shp = state.SummaryMajor.shape();
     505           0 :                            if( shp.nelements() != 1 )
     506           0 :                                throw(AipsError("Internal error in shape of major-cycle summary record"));
     507             : 
     508           0 :                            state.SummaryMajor.resize( IPosition( 1, shp[0]+1 ) , true );
     509           0 :                            state.SummaryMajor( IPosition(1, shp[0] ) ) = state.IterDone;
     510           0 :                            return dummy; } ) );
     511           0 :         }
     512             : 
     513           0 :     casacore::Record grpcInteractiveCleanManager::getDetailsRecord( bool includeSummary ) {
     514           0 :                 LogIO os( LogOrigin("grpcInteractiveCleanManager",__FUNCTION__,WHERE) );
     515             : 
     516           0 :                 Record returnRecord;
     517             : 
     518             :         Record result = access( returnRecord,
     519           0 :                        std::function< casacore::Record ( casacore::Record, grpcInteractiveCleanState & )>(
     520           0 :                        [&]( casacore::Record rec, grpcInteractiveCleanState &state )->casacore::Record {
     521             :                            //*** Control Variables **************************************************
     522           0 :                            rec.define( RecordFieldId("niter"), state.Niter );
     523           0 :                            rec.define( RecordFieldId("cycleniter"), state.CycleNiter );
     524           0 :                            rec.define( RecordFieldId("interactiveniter"), state.InteractiveNiter );
     525             : 
     526           0 :                            rec.define( RecordFieldId("threshold"),  state.Threshold );
     527           0 :                            rec.define( RecordFieldId("nsigma"),  state.Nsigma );
     528             : 
     529           0 :                            if( state.IsCycleThresholdAuto == true ) updateCycleThreshold(state);
     530           0 :                            state.IsCycleThresholdAuto = true && state.IsCycleThresholdMutable;        // Reset this, for the next round
     531             : 
     532           0 :                            rec.define( RecordFieldId("cyclethreshold"), state.CycleThreshold );
     533           0 :                            rec.define( RecordFieldId("interactivethreshold"), state.InteractiveThreshold );
     534             : 
     535           0 :                            rec.define( RecordFieldId("loopgain"), state.LoopGain );
     536           0 :                            rec.define( RecordFieldId("cyclefactor"), state.CycleFactor );
     537             : 
     538             :                            //*** Status Reporting Variables *****************************************
     539           0 :                            rec.define( RecordFieldId("iterdone"),  state.IterDone );
     540           0 :                            rec.define( RecordFieldId("cycleiterdone"), state.MaxCycleIterDone );
     541           0 :                            rec.define( RecordFieldId("interactiveiterdone"),
     542           0 :                                        state.InteractiveIterDone + state.MaxCycleIterDone);
     543             : 
     544           0 :                            rec.define( RecordFieldId("nmajordone"), state.MajorDone );
     545           0 :                            rec.define( RecordFieldId("maxpsfsidelobe"), state.MaxPsfSidelobe );
     546           0 :                            rec.define( RecordFieldId("maxpsffraction"), state.MaxPsfFraction );
     547           0 :                            rec.define( RecordFieldId("minpsffraction"), state.MinPsfFraction );
     548           0 :                            rec.define( RecordFieldId("interactivemode"), state.InteractiveMode );
     549             : 
     550           0 :                            rec.define( RecordFieldId("stopcode"), state.StopCode );
     551             : 
     552             :                            //*** report clean's state ***********************************************
     553           0 :                            rec.define( RecordFieldId("cleanstate"),
     554           0 :                                        state.StopFlag ? "stopped" : state.PauseFlag ? "paused" : "running" );
     555             : 
     556           0 :                            if ( includeSummary ) {
     557           0 :                                rec.define( RecordFieldId("summaryminor"), state.SummaryMinor );
     558           0 :                                rec.define( RecordFieldId("summarymajor"), state.SummaryMajor );
     559             :                            }
     560             : 
     561           0 :                            return rec; }) );
     562             : 
     563           0 :         return result;
     564             : 
     565             : 
     566             : /*      return access( returnRecord,
     567             :                        std::function< casacore::Record ( casacore::Record, grpcInteractiveCleanState & )>(
     568             :                        [&]( casacore::Record rec, grpcInteractiveCleanState &state )->casacore::Record {
     569             :                            //*** Control Variables **************************************************
     570             :                            rec.define( RecordFieldId("niter"), state.Niter );
     571             :                            rec.define( RecordFieldId("cycleniter"), state.CycleNiter );
     572             :                            rec.define( RecordFieldId("interactiveniter"), state.InteractiveNiter );
     573             : 
     574             :                            rec.define( RecordFieldId("threshold"),  state.Threshold );
     575             :                            rec.define( RecordFieldId("nsigma"),  state.Nsigma );
     576             :                            if( state.IsCycleThresholdAuto == true ) updateCycleThreshold(state);
     577             :                            state.IsCycleThresholdAuto = true && state.IsCycleThresholdMutable;        // Reset this, for the next round
     578             : 
     579             :                            rec.define( RecordFieldId("cyclethreshold"), state.CycleThreshold );
     580             :                            rec.define( RecordFieldId("interactivethreshold"), state.InteractiveThreshold );
     581             : 
     582             :                            rec.define( RecordFieldId("loopgain"), state.LoopGain );
     583             :                            rec.define( RecordFieldId("cyclefactor"), state.CycleFactor );
     584             : 
     585             :                            //*** Status Reporting Variables *****************************************
     586             :                            rec.define( RecordFieldId("iterdone"),  state.IterDone );
     587             :                            rec.define( RecordFieldId("cycleiterdone"), state.MaxCycleIterDone );
     588             :                            rec.define( RecordFieldId("interactiveiterdone"),
     589             :                                        state.InteractiveIterDone + state.MaxCycleIterDone);
     590             : 
     591             :                            rec.define( RecordFieldId("nmajordone"), state.MajorDone );
     592             :                            rec.define( RecordFieldId("maxpsfsidelobe"), state.MaxPsfSidelobe );
     593             :                            rec.define( RecordFieldId("maxpsffraction"), state.MaxPsfFraction );
     594             :                            rec.define( RecordFieldId("minpsffraction"), state.MinPsfFraction );
     595             :                            rec.define( RecordFieldId("interactivemode"), state.InteractiveMode );
     596             : 
     597             :                            rec.define( RecordFieldId("stopcode"), state.StopCode );
     598             : 
     599             :                            //*** report clean's state ***********************************************
     600             :                            rec.define( RecordFieldId("cleanstate"),
     601             :                                        state.StopFlag ? "stopped" : state.PauseFlag ? "paused" : "running" );
     602             : 
     603             :                            if ( includeSummary ) {
     604             :                                rec.define( RecordFieldId("summaryminor"), state.SummaryMinor );
     605             :                                rec.define( RecordFieldId("summarymajor"), state.SummaryMajor );
     606             :                            }
     607             : 
     608             :                            return rec; }) );
     609             : */
     610           0 :     }
     611             : 
     612             : 
     613           0 :         Record grpcInteractiveCleanManager::getMinorCycleControls( ){
     614           0 :                 LogIO os( LogOrigin("grpcInteractiveCleanManager",__FUNCTION__,WHERE) );
     615             : 
     616             :                 /* This returns a record suitable for initializing the minor cycle controls. */
     617           0 :                 Record returnRecord;
     618             : 
     619             :         return access( returnRecord,
     620           0 :                        std::function< casacore::Record ( casacore::Record, grpcInteractiveCleanState & )>(
     621           0 :                        [&]( casacore::Record rec, grpcInteractiveCleanState &state )->casacore::Record {
     622             : 
     623             :                            /* If autocalc, compute cyclethresh from peak res, cyclefactor and psf sidelobe
     624             :                               Otherwise, the user has explicitly set it (interactively) for this minor cycle */
     625           0 :                            if( state.IsCycleThresholdAuto == true ) { updateCycleThreshold(state); }
     626           0 :                            state.IsCycleThresholdAuto = true && state.IsCycleThresholdMutable; /* Reset this, for the next round */
     627             : 
     628             :                            /* The minor cycle will stop based on the cycle parameters. */
     629           0 :                            int maxCycleIterations = state.CycleNiter;
     630           0 :                            float cycleThreshold     = state.CycleThreshold;
     631           0 :                            maxCycleIterations = min(maxCycleIterations, state.Niter - state.IterDone);
     632           0 :                            cycleThreshold = max(cycleThreshold, state.Threshold);
     633           0 :                            bool thresholdReached = (cycleThreshold==state.Threshold)? True : False;
     634             : 
     635           0 :                            rec.define( RecordFieldId("cycleniter"),  maxCycleIterations);
     636           0 :                            rec.define( RecordFieldId("cyclethreshold"), cycleThreshold);
     637           0 :                            rec.define( RecordFieldId("loopgain"), state.LoopGain);
     638           0 :                            rec.define( RecordFieldId("thresholdreached"), thresholdReached);
     639           0 :                            rec.define( RecordFieldId("nsigma"), state.Nsigma);
     640             : 
     641           0 :                            return rec; }) );
     642           0 :     }
     643             : 
     644           0 :         int grpcInteractiveCleanManager::cleanComplete( bool lastcyclecheck, bool reachedMajorLimit ){
     645           0 :         LogIO os( LogOrigin("grpcInteractiveCleanManager",__FUNCTION__,WHERE) );
     646             : 
     647           0 :                 int stopCode=0;
     648             : 
     649           0 :         return access( stopCode,
     650           0 :                        std::function< int ( int, grpcInteractiveCleanState & )>(
     651           0 :                        [&]( int stop_code, grpcInteractiveCleanState &state ) -> int {
     652             : 
     653             :                            float usePeakRes;
     654             : 
     655           0 :                            if( lastcyclecheck==True ) { usePeakRes = state.MinorCyclePeakResidual; }
     656           0 :                            else { usePeakRes = state.PeakResidual; }
     657             : 
     658             :                            // for debugging, remove it later
     659           0 :                            os<<LogIO::DEBUG1<<"cleanComplete-- CycleThreshold without Threshold limit="<<state.CycleThreshold<<LogIO::POST;
     660             : 
     661           0 :                            if( state.PeakResidual > 0 && state.PrevPeakResidual>0 &&
     662           0 :                                fabs( state.PeakResidual - state.PrevPeakResidual)/fabs(state.PrevPeakResidual) > 2.0 ) {
     663           0 :                                os << "[WARN] Peak residual (within the mask) increased from " << state.PrevPeakResidual << " to " << state.PeakResidual << LogIO::POST;
     664             :                            }
     665             :                            // for debugging, remove it later
     666           0 :                            os <<LogIO::DEBUG1<<"Threshold="<<state.Threshold<<" itsNsigmaThreshold===="<<state.NsigmaThreshold<<LogIO::POST;
     667           0 :                            os <<LogIO::DEBUG1<<"usePeakRes="<<usePeakRes<<" itsPeakResidual="<<state.PeakResidual<<" itsPrevPeakRes="<<state.PrevPeakResidual<<LogIO::POST;
     668           0 :                            os <<LogIO::DEBUG1<<"itsIterDone="<<state.IterDone<<" itsNiter="<<state.Niter<<LogIO::POST;
     669           0 :                            os <<LogIO::DEBUG1<<"FullSummary="<<state.FullSummary<<LogIO::POST;
     670             : 
     671             :                            /// This may interfere with some other criterion... check.
     672           0 :                            float tol = 0.01; // threshold test torelance (CAS-11278)
     673           0 :                            if ( state.MajorDone==0 && state.IterDone==0 && state.MaskSum==0.0) {
     674           0 :                                stopCode=7; // if zero mask is detected it should exit right away
     675           0 :                            } else if ( state.IterDone >= state.Niter ||
     676           0 :                                        usePeakRes <= state.Threshold ||
     677           0 :                                        state.PeakResidual <= state.NsigmaThreshold ||
     678           0 :                                        fabs(usePeakRes-state.Threshold)/state.Threshold < tol ||
     679           0 :                                        fabs(state.PeakResidual - state.NsigmaThreshold)/state.NsigmaThreshold < tol ||
     680           0 :                                        state.StopFlag ) {
     681             :                                //                   os << "Reached global stopping criteria : ";
     682             : 
     683           0 :                                if ( state.IterDone >= state.Niter ) { stopCode=1; }
     684             :                                //os << "Numer of iterations. "; // (" << state.IterDone << ") >= limit (" << state.Niter << ")" ;
     685           0 :                                if( usePeakRes <= state.Threshold || (usePeakRes-state.Threshold)/state.Threshold < tol) {stopCode=2; }
     686           0 :                                else if ( usePeakRes <= state.NsigmaThreshold || (state.PeakResidual - state.NsigmaThreshold)/state.NsigmaThreshold < tol ) {
     687           0 :                                    if (state.NsigmaThreshold!=0.0) { stopCode=8; } // for nsigma=0.0 this mode is turned off
     688             :                                }
     689             : 
     690             :                                //os << "Peak residual (" << state.PeakResidual << ") <= threshold(" << state.Threshold << ")";
     691           0 :                                if( state.StopFlag ) {stopCode=3;}
     692             :                                //os << "Forced stop. ";
     693             :                                //                   os << LogIO::POST;
     694             : 
     695             :                                //return true;
     696             : 
     697             :                            } else { // not converged yet... but....if nothing has changed in this round... also stop
     698             : 
     699           0 :                                if (state.MaskSum==0.0) {
     700             :                                    //cout << "(7) Mask is all zero.Stopping" << endl;
     701           0 :                                    stopCode = 7;
     702             :                                }
     703             :                                // Nothing has changed across the last set of minor cycle iterations and major cycle.
     704           0 :                                else if( state.IterDone>0 && (state.MajorDone>state.PrevMajorCycleCount) &&
     705           0 :                                         fabs(state.PrevPeakResidual - state.PeakResidual)<1e-10)
     706           0 :                                    {stopCode = 4;}
     707             : 
     708             :                                // another non-convergent condition: diverging (relative increase is more than 3 times across one major cycle)
     709           0 :                                else if ( state.IterDone > 0 &&
     710           0 :                                          fabs(state.PeakResidualNoMask-state.PrevPeakResidualNoMask)/fabs(state.PrevPeakResidualNoMask)  > 3.0) {
     711             :                                    //cout << "(5) Peak res (no mask) : " << state.PeakResidualNoMask
     712             :                                    //     << "  Dev from prev peak res " << state.PrevPeakResidualNoMask << endl;
     713           0 :                                    stopCode = 5;}
     714             : 
     715             :                                // divergence check, 3 times increase from the minimum peak residual so far (across all previous major cycles).
     716           0 :                                else if ( state.IterDone > 0 &&
     717           0 :                               (fabs(state.PeakResidualNoMask)-state.MinPeakResidualNoMask)/state.MinPeakResidualNoMask  > 3.0 )
     718             :                       {
     719             :                         //cout << "(6) Peak res (no mask): " << state.PeakResidualNoMask
     720             :                         //    <<  "    Dev from min peak res " << state.MinPeakResidualNoMask << endl;
     721           0 :                         stopCode = 6;
     722             :                       }
     723             : 
     724             :                   }
     725             : 
     726           0 :                 if (stopCode == 0 && reachedMajorLimit) {
     727           0 :                   stopCode = 9;
     728             :                 }
     729             : 
     730             :                 /*
     731             :                 if( lastcyclecheck==False)
     732             :                   {
     733             :                     cout << "*****" << endl;
     734             :                     cout << "Peak residual : " << state.PeakResidual << "  No Mask : " << state.PeakResidualNoMask << endl;
     735             :                     cout << "Prev Peak residual : " << state.PrevPeakResidual << "  No Mask : " << state.PrevPeakResidualNoMask << endl;
     736             :                     cout << "Min Peak residual : " << state.MinPeakResidual << "  No Mask : " << state.MinPeakResidualNoMask << endl;
     737             :                   }
     738             :                 */
     739             : 
     740             :                 //              os << "Peak residual : " << state.PeakResidual << " and " << state.IterDone << " iterations."<< LogIO::POST;
     741             :                 //cout << "cleancomp : stopcode : " << stopCode << endl;
     742             : 
     743             :                 //cout << "peak res : " << state.PeakResidual << "   state.minPR : " << state.MinPeakResidual << endl;
     744             : 
     745           0 :                 if( lastcyclecheck==False)
     746             :                   {
     747           0 :                     if( fabs(state.PeakResidual) < state.MinPeakResidual )
     748           0 :                       {state.MinPeakResidual = fabs(state.PeakResidual);}
     749             : 
     750           0 :                     state.PrevPeakResidual = state.PeakResidual;
     751             : 
     752             : 
     753           0 :                     if( fabs(state.PeakResidualNoMask) < state.MinPeakResidualNoMask )
     754           0 :                       {state.MinPeakResidualNoMask = fabs(state.PeakResidualNoMask);}
     755             : 
     756           0 :                     state.PrevPeakResidualNoMask = state.PeakResidualNoMask;
     757             : 
     758           0 :                     state.PrevMajorCycleCount = state.MajorDone;
     759             : 
     760             :                   }
     761             : 
     762           0 :                 state.StopCode=stopCode;
     763           0 :                 return stopCode; } ) );
     764           0 :         }
     765             : 
     766           0 :         void grpcInteractiveCleanManager::resetMinorCycleInitInfo( grpcInteractiveCleanState &state ) {
     767             :                 /* Get ready to do the minor cycle */
     768           0 :                 state.PeakResidual = 0;
     769           0 :                 state.PeakResidualNoMask = 0;
     770           0 :                 state.MaxPsfSidelobe = 0;
     771           0 :                 state.MaxCycleIterDone = 0;
     772           0 :                 state.MaskSum = -1.0;
     773           0 :         }
     774             : 
     775           0 :     void grpcInteractiveCleanManager::resetMinorCycleInitInfo( ) {
     776           0 :         access( (void*) 0,
     777           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
     778           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
     779           0 :                            resetMinorCycleInitInfo(state);
     780           0 :                            return dummy; } ) );
     781           0 :     }
     782             : 
     783           0 :         void grpcInteractiveCleanManager::incrementMajorCycleCount( ) {
     784             : 
     785           0 :         access( (void*) 0,
     786           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
     787           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
     788           0 :                            state.PrevMajorCycleCount = state.MajorDone;
     789           0 :                            state.MajorDone++;
     790             : 
     791             :                            /* Interactive iteractions update */
     792           0 :                            state.InteractiveIterDone += state.MaxCycleIterDone;
     793             : 
     794           0 :                            resetMinorCycleInitInfo(state);
     795           0 :                            return dummy; } ) );
     796           0 :         }
     797             : 
     798           0 :         void grpcInteractiveCleanManager::mergeCycleInitializationRecord( Record &initRecord ){
     799           0 :         LogIO os( LogOrigin("grpcInteractiveCleanManager",__FUNCTION__,WHERE) );
     800             : 
     801           0 :         access( (void*) 0,
     802           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
     803           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
     804             : 
     805           0 :                            state.PeakResidual = max(state.PeakResidual, initRecord.asFloat(RecordFieldId("peakresidual")));
     806           0 :                            state.MaxPsfSidelobe =  max(state.MaxPsfSidelobe, initRecord.asFloat(RecordFieldId("maxpsfsidelobe")));
     807             : 
     808           0 :                            state.PeakResidualNoMask = max( state.PeakResidualNoMask, initRecord.asFloat(RecordFieldId("peakresidualnomask")));
     809           0 :                            state.MadRMS = max(state.MadRMS, initRecord.asFloat(RecordFieldId("madrms")));
     810           0 :                            state.NsigmaThreshold = initRecord.asFloat(RecordFieldId("nsigmathreshold"));
     811             : 
     812             :                            /*
     813             :                              It has been reset to -1.0.
     814             :                              If no masks have changed, it should remain at -1.0
     815             :                              If any mask has changed, the sum will come in, and should be added to this.
     816             :                            */
     817           0 :                            float thismasksum = initRecord.asFloat(RecordFieldId("masksum"));
     818           0 :                            if( thismasksum != -1.0 ) {
     819           0 :                                if ( state.MaskSum == -1.0 ) state.MaskSum = thismasksum;
     820           0 :                                else state.MaskSum += thismasksum;
     821             :                            }
     822             : 
     823           0 :                            if ( state.PrevPeakResidual == -1.0 ) state.PrevPeakResidual = state.PeakResidual;
     824           0 :                            if ( state.PrevPeakResidualNoMask == -1.0 ) state.PrevPeakResidualNoMask = state.PeakResidualNoMask;
     825           0 :                            if( state.IsCycleThresholdAuto == true ) updateCycleThreshold(state);
     826             : 
     827           0 :                            return dummy; } ) );
     828           0 :         }
     829             : 
     830             : 
     831           0 :         void grpcInteractiveCleanManager::mergeMinorCycleSummary( const Array<Double> &summary, grpcInteractiveCleanState &state, Int immod ){
     832           0 :                 IPosition cShp = state.SummaryMinor.shape();
     833           0 :                 IPosition nShp = summary.shape();
     834             : 
     835             :                 //bool uss = SIMinorCycleController::useSmallSummaryminor(); // temporary CAS-13683 workaround
     836             :         //int nSummaryFields = uss ? 6 : SIMinorCycleController::nSummaryFields;
     837           0 :                 int nSummaryFields = !state.FullSummary ? 6 : SIMinorCycleController::nSummaryFields;
     838             :         
     839           0 :                 if( cShp.nelements() != 2 || cShp[0] != nSummaryFields ||
     840           0 :                     nShp.nelements() != 2 || nShp[0] != nSummaryFields )
     841           0 :                         throw(AipsError("Internal error in shape of global minor-cycle summary record"));
     842             : 
     843           0 :                 state.SummaryMinor.resize( IPosition( 2, nSummaryFields, cShp[1]+nShp[1] ) ,true );
     844             : 
     845           0 :                 for (unsigned int row = 0; row < nShp[1]; row++) {
     846             :                         // iterations done
     847           0 :                         state.SummaryMinor( IPosition(2,0,cShp[1]+row) ) = summary(IPosition(2,0,row));
     848             :                        //if (state.FullSummary){
     849             :                        //    state.SummaryMinor( IPosition(2,0,cShp[1]+row) ) = state.IterDone + summary(IPosition(2,0,row));
     850             :                        //}
     851             :                        //else{
     852             :                        //    state.SummaryMinor( IPosition(2,0,cShp[1]+row) ) = summary(IPosition(2,0,row));
     853             :                        //}
     854             :                    //state.SummaryMinor( IPosition(2,0,cShp[1]+row) ) = summary(IPosition(2,0,row));
     855             :                         // peak residual
     856           0 :                         state.SummaryMinor( IPosition(2,1,cShp[1]+row) ) = summary(IPosition(2,1,row));
     857             :                         // model flux
     858           0 :                         state.SummaryMinor( IPosition(2,2,cShp[1]+row) ) = summary(IPosition(2,2,row));
     859             :                         // cycle threshold
     860           0 :                         state.SummaryMinor( IPosition(2,3,cShp[1]+row) ) = summary(IPosition(2,3,row));
     861             :                         //if (uss) { // temporary CAS-13683 workaround
     862           0 :                         if (!state.FullSummary) { // temporary CAS-13683 workaround
     863             :                                 // swap out mapper id with multifield id
     864           0 :                                 state.SummaryMinor( IPosition(2,4,cShp[1]+row) ) = immod;
     865             :                                 // chunk id (channel/stokes)
     866           0 :                                 state.SummaryMinor( IPosition(2,5,cShp[1]+row) ) = summary(IPosition(2,5,row));
     867             :                         } else {
     868             :                                 // mapper id
     869           0 :                                 state.SummaryMinor( IPosition(2,4,cShp[1]+row) ) = summary(IPosition(2,4,row));
     870             :                                 // channel id
     871           0 :                                 state.SummaryMinor( IPosition(2,5,cShp[1]+row) ) = summary(IPosition(2,5,row));
     872             :                                 // polarity id
     873           0 :                                 state.SummaryMinor( IPosition(2,6,cShp[1]+row) ) = summary(IPosition(2,6,row));
     874             :                                 // cycle start iterations done
     875           0 :                                 state.SummaryMinor( IPosition(2,7,cShp[1]+row) ) = state.IterDone + summary(IPosition(2,7,row));
     876             :                                 // starting iterations done
     877           0 :                                 state.SummaryMinor( IPosition(2,8,cShp[1]+row) ) = state.IterDone + summary(IPosition(2,8,row));
     878             :                                 // starting peak residual
     879           0 :                                 state.SummaryMinor( IPosition(2,9,cShp[1]+row) ) = summary(IPosition(2,9,row));
     880             :                                 // starting model flux
     881           0 :                                 state.SummaryMinor( IPosition(2,10,cShp[1]+row) ) = summary(IPosition(2,10,row));
     882             :                                 // starting peak residual, not limited to the user's mask
     883           0 :                                 state.SummaryMinor( IPosition(2,11,cShp[1]+row) ) = summary(IPosition(2,11,row));
     884             :                                 // peak residual, not limited to the user's mask
     885           0 :                                 state.SummaryMinor( IPosition(2,12,cShp[1]+row) ) = summary(IPosition(2,12,row));
     886             :                                 // number of pixels in the mask
     887           0 :                                 state.SummaryMinor( IPosition(2,13,cShp[1]+row) ) = summary(IPosition(2,13,row));
     888             :                                 // mpi server
     889           0 :                                 state.SummaryMinor( IPosition(2,14,cShp[1]+row) ) = summary(IPosition(2,14,row));
     890             :                                 // outlier field id
     891           0 :                                 state.SummaryMinor( IPosition(2,15,cShp[1]+row) ) = immod;
     892             :                                 // stopcode
     893           0 :                                 state.SummaryMinor( IPosition(2,16,cShp[1]+row) ) = summary(IPosition(2,16,row));
     894             :                         }
     895             : 
     896             :                 }
     897           0 :         }
     898             : 
     899           0 :         void grpcInteractiveCleanManager::mergeCycleExecutionRecord( Record& execRecord, Int immod ){
     900           0 :         LogIO os( LogOrigin("grpcInteractiveCleanManager",__FUNCTION__,WHERE) );
     901             : 
     902           0 :         access( (void*) 0,
     903           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
     904           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
     905           0 :                            mergeMinorCycleSummary( execRecord.asArrayDouble( RecordFieldId("summaryminor")), state, immod );
     906             : 
     907           0 :                            state.IterDone += execRecord.asInt(RecordFieldId("iterdone"));
     908             : 
     909           0 :                            state.MaxCycleIterDone = max( state.MaxCycleIterDone, execRecord.asInt(RecordFieldId("maxcycleiterdone")) );
     910             : 
     911           0 :                            state.MinorCyclePeakResidual = max( state.PeakResidual, execRecord.asFloat(RecordFieldId("peakresidual")) );
     912             : 
     913           0 :                            state.UpdatedModelFlag |=execRecord.asBool( RecordFieldId("updatedmodelflag") );
     914             : 
     915           0 :                            os << "Completed " << state.IterDone << " iterations." << LogIO::POST;
     916             :                            //with peak residual "<< state.PeakResidual << LogIO::POST;
     917           0 :                            return dummy; } ) );
     918           0 :         }
     919             : 
     920           0 :         void grpcInteractiveCleanManager::changeStopFlag( bool stopEnabled ) {
     921           0 :         access( (void*) 0,
     922           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
     923           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
     924           0 :                            state.StopFlag = stopEnabled;
     925           0 :                            return dummy;
     926             :                        } ) );
     927           0 :         }
     928             : 
     929             :     //====================================================================================================
     930             : 
     931           0 :     static bool isdir( const char *path ) {
     932             :         struct stat statbuf;
     933           0 :         int err = stat(path, &statbuf);
     934           0 :         if ( err == -1 ) return false;
     935           0 :         if ( S_ISDIR(statbuf.st_mode) ) return true;
     936           0 :         return false;
     937             :     }
     938             : 
     939           0 :     static std::string trim_trailing_slash( const char *str ) {
     940           0 :         char *temp = strdup(str);
     941           0 :         for ( int off = strlen(str) - 1; off >= 0; --off ) {
     942           0 :             if ( temp[off] == '/' ) temp[off] = '\0';
     943           0 :             else break;
     944             :         }
     945           0 :         std::string result = temp;
     946           0 :         free(temp);
     947           0 :         return result;
     948             :     }
     949             : 
     950           0 :     grpcInteractiveCleanGui::grpcInteractiveCleanGui( ) : viewer_pid(0), viewer_started(false) { }
     951           0 :     grpcInteractiveCleanGui::~grpcInteractiveCleanGui( ) {
     952           0 :         static const auto debug = getenv("GRPC_DEBUG");
     953             : 
     954           0 :         if ( ! viewer_started ) {
     955           0 :             if ( debug ) {
     956           0 :                 std::cerr << "viewer shutdown required (" << viewer_uri << ")" << 
     957           0 :                     " (process " << getpid( ) << ", thread " << 
     958           0 :                     std::this_thread::get_id() << ")" << std::endl;
     959           0 :                 fflush(stderr);
     960             :             }
     961             :         } else {
     962           0 :             if ( debug ) {
     963           0 :                 std::cerr << "sending shutdown message to viewer (" << viewer_uri << ")" << 
     964           0 :                     " (process " << getpid( ) << ", thread " << 
     965           0 :                     std::this_thread::get_id() << ")" << std::endl;
     966           0 :                 fflush(stderr);
     967             :             }
     968             : 
     969           0 :             bool stopped = stop_viewer( );
     970             : 
     971           0 :             if ( debug ) {
     972           0 :                 if ( stopped ) {
     973           0 :                     std::cerr << "viewer shutdown successful (" << viewer_uri << ")" << 
     974           0 :                         " (process " << getpid( ) << ", thread " << 
     975           0 :                         std::this_thread::get_id() << ")" << std::endl;
     976             :                 } else {
     977           0 :                     std::cerr << "viewer shutdown failed (" << viewer_uri << ")" << 
     978           0 :                         " (process " << getpid( ) << ", thread " << 
     979           0 :                         std::this_thread::get_id() << ")" << std::endl;
     980             :                 }
     981           0 :                 fflush(stderr);
     982             :             }
     983             :         }
     984           0 :     }
     985             : 
     986             : 
     987           0 :     bool grpcInteractiveCleanGui::alive( ) {
     988           0 :         static const auto debug = getenv("GRPC_DEBUG");
     989           0 :         if ( debug ) {
     990           0 :             std::cerr << "pinging viewer (" << viewer_uri << ")" << 
     991           0 :                 " (process " << getpid( ) << ", thread " << 
     992           0 :                 std::this_thread::get_id() << ")" << std::endl;
     993           0 :             fflush(stderr);
     994             :         }
     995           0 :         grpc::ClientContext context;
     996           0 :         ::google::protobuf::Empty resp;
     997           0 :         ::google::protobuf::Empty msg;
     998           0 :         auto ping = casatools::rpc::Ping::NewStub( grpc::CreateChannel( viewer_uri, grpc::InsecureChannelCredentials( ) ) );
     999           0 :         ::grpc::Status status = ping->now( &context, msg, &resp );
    1000           0 :         bool ping_result = status.ok( );
    1001           0 :         if ( debug ) {
    1002             :             std::cerr << "ping result: " << (ping_result ? "OK" : "FAIL")<< 
    1003           0 :                 " (process " << getpid( ) << ", thread " << 
    1004           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1005           0 :             fflush(stderr);
    1006             :         }
    1007           0 :         if ( ping_result == false ) {
    1008             :             int proc_status;
    1009           0 :             waitpid( viewer_pid, &proc_status, WUNTRACED | WCONTINUED | WNOHANG );
    1010           0 :             viewer_pid = 0;
    1011           0 :             viewer_proxy.release( );
    1012           0 :             viewer_started = false;
    1013           0 :             if ( debug ) {
    1014             :                 std::cerr << "ping failed resetting state" << 
    1015           0 :                     " (process " << getpid( ) << ", thread " << 
    1016           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1017           0 :                 fflush(stderr);
    1018             :             }
    1019             :         }
    1020           0 :         return ping_result;
    1021           0 :     }
    1022             : 
    1023           0 :     bool grpcInteractiveCleanGui::launch( ) {
    1024           0 :         static const auto debug = getenv("GRPC_DEBUG");
    1025           0 :         if ( viewer_started == false ) {
    1026             :             // start the viewer process if it is not already running...
    1027           0 :             if ( debug ) {
    1028             :                 std::cerr << "spawning viewer process" <<
    1029           0 :                     " (process " << getpid( ) << ", thread " <<
    1030           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1031           0 :                 fflush(stderr);
    1032             :             }
    1033           0 :             return spawn_viewer( );
    1034             :         } else {
    1035           0 :             if ( alive( ) ) {
    1036           0 :               if ( debug ) {
    1037             :                 std::cerr << "viewer process available" <<
    1038           0 :                     " (process " << getpid( ) << ", thread " <<
    1039           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1040           0 :                 fflush(stderr);
    1041             :               }
    1042           0 :               return true;
    1043             :             } else {
    1044           0 :               if ( debug ) {
    1045             :                 std::cerr << "re-spawning viewer process" <<
    1046           0 :                     " (process " << getpid( ) << ", thread " <<
    1047           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1048           0 :                 fflush(stderr);
    1049             :               }
    1050           0 :               return launch( );
    1051             :             }
    1052             :         }
    1053             :         return false;
    1054             :     }
    1055             : 
    1056           0 :     void grpcInteractiveCleanGui::close_panel( int id ) {
    1057           0 :         static const auto debug = getenv("GRPC_DEBUG");
    1058           0 :         if ( debug ) {
    1059           0 :             std::cerr << "close_panel(" << id << ")" <<
    1060           0 :                 " (process " << getpid( ) << ", thread " << 
    1061           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1062           0 :             fflush(stderr);
    1063             :         }
    1064           0 :         if ( id != -1 && alive( ) ) {
    1065           0 :             if ( debug ) {
    1066           0 :                 std::cerr << "close_panel(" << id << ") -- closing panel" <<
    1067           0 :                     " (process " << getpid( ) << ", thread " << 
    1068           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1069           0 :                 fflush(stderr);
    1070             :             }
    1071             :             {
    1072             :                 // unload panel's images
    1073           0 :                 rpc::img::Id panel;
    1074           0 :                 grpc::ClientContext context;
    1075           0 :                 ::google::protobuf::Empty resp;
    1076           0 :                 panel.set_id(id);
    1077           0 :                 viewer_proxy->unload( &context, panel, &resp );
    1078           0 :             }
    1079             :             {
    1080             :                 // close panel
    1081           0 :                 rpc::img::Id panel;
    1082           0 :                 grpc::ClientContext context;
    1083           0 :                 ::google::protobuf::Empty resp;
    1084           0 :                 panel.set_id(id);
    1085           0 :                 viewer_proxy->close( &context, panel, &resp );
    1086           0 :             }
    1087             :         }
    1088           0 :     }
    1089             : 
    1090           0 :     int grpcInteractiveCleanGui::open_panel( std::list<std::tuple<std::string,bool,bool,int>> images ) {
    1091           0 :         static const auto debug = getenv("GRPC_DEBUG");
    1092           0 :         if ( viewer_started == false ) {
    1093           0 :             if ( launch( ) == false ) return -1;
    1094             :         }
    1095           0 :         if ( debug ) {
    1096             :             std::cerr << "opening viewer panel" <<
    1097           0 :                 " (process " << getpid( ) << ", thread " << 
    1098           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1099           0 :             fflush(stderr);
    1100             :         }
    1101           0 :         grpc::ClientContext context;
    1102           0 :         ::rpc::img::NewPanel np;
    1103           0 :         rpc::img::Id resp;
    1104           0 :         np.set_type("clean2");
    1105           0 :         np.set_hidden(false);
    1106           0 :         viewer_proxy->panel( &context, np, &resp );
    1107           0 :         int result = resp.id( );
    1108             : 
    1109           0 :         if ( debug ) {
    1110           0 :             std::cerr << "opened viewer panel " << result <<
    1111           0 :                 " (process " << getpid( ) << ", thread " << 
    1112           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1113           0 :             fflush(stderr);
    1114             :         }
    1115             : 
    1116             :         // state for interactive masking in the new viewer panel
    1117           0 :         clean_state.insert( std::pair<int,CleanState>(result, CleanState( )) );
    1118             : 
    1119           0 :         if ( debug ) {
    1120           0 :             std::cerr << "created panel " << result <<
    1121           0 :                 " (process " << getpid( ) << ", thread " << 
    1122           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1123           0 :             fflush(stderr);
    1124             :         }
    1125           0 :         return result;
    1126           0 :     }
    1127             : 
    1128           0 :     void grpcInteractiveCleanGui::unload( int id ) {
    1129           0 :         grpc::ClientContext context;
    1130           0 :         ::rpc::img::Id data;
    1131           0 :         ::google::protobuf::Empty resp;
    1132           0 :         data.set_id(id);
    1133           0 :         viewer_proxy->unload( &context, data, &resp );
    1134           0 :     }
    1135             : 
    1136           0 :     bool grpcInteractiveCleanGui::clone( const std::string &imageName, const std::string &newImageName ) {
    1137           0 :         LogIO os(LogOrigin("grpcInteractiveCleanGui", __FUNCTION__, WHERE));
    1138             : 
    1139             :         try {
    1140           0 :             PagedImage<Float> oldImage( imageName );
    1141           0 :             PagedImage<Float> newImage( TiledShape( oldImage.shape(), oldImage.niceCursorShape()),
    1142           0 :                                         oldImage.coordinates(), newImageName );
    1143           0 :             newImage.set(0.0);
    1144           0 :             newImage.table().flush(true, true);
    1145           0 :         } catch (AipsError x) {
    1146           0 :             os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
    1147           0 :             return false;
    1148           0 :         } 
    1149           0 :         return true;
    1150           0 :     }
    1151             : 
    1152           0 :     float grpcInteractiveCleanGui::maskSum(const std::string &maskname) {
    1153             : 
    1154           0 :         PagedImage<Float> mask( maskname );
    1155             : 
    1156           0 :         LatticeExprNode msum( sum( mask ) );
    1157           0 :         float maskSum = msum.getFloat( );
    1158             : 
    1159           0 :         mask.unlock();
    1160           0 :         mask.tempClose();
    1161             : 
    1162           0 :         return maskSum;
    1163           0 :     }
    1164             : 
    1165           0 :     int grpcInteractiveCleanGui::interactivemask( int panel, const std::string &image, const std::string &mask,
    1166             :                                                   int &niter, int &cycleniter, std::string &thresh,
    1167             :                                                   std::string &cyclethresh, const bool forceReload ) {
    1168             : 
    1169           0 :         static const auto debug = getenv("GRPC_DEBUG");
    1170           0 :         LogIO os( LogOrigin("grpcInteractiveCleanGui",__FUNCTION__,WHERE) );
    1171             : 
    1172           0 :         if ( debug ) {
    1173           0 :             std::cerr << "starting interactivemask( " <<
    1174           0 :                                  panel << ", " << image << ", " << mask << ", " <<
    1175           0 :                                  niter << ", " << cycleniter << ", " << thresh << ", " <<
    1176             :                                  cyclethresh << ", " << (forceReload ? "true" : "false") << ")" << 
    1177           0 :                                  " (process " << getpid( ) << ", thread " << 
    1178           0 :                                  std::this_thread::get_id() << ")" << std::endl;
    1179           0 :             fflush(stderr);
    1180             :         }
    1181             : 
    1182           0 :         if ( viewer_started == false ) {
    1183             :             // viewer should be started before calling interactivemask(...)
    1184           0 :             os << LogIO::WARN << "Viewer GUI Not Available" << LogIO::POST;
    1185           0 :             return 0;
    1186             :         }
    1187             : 
    1188           0 :         auto state = clean_state.find(panel);
    1189           0 :         if ( state == clean_state.end( ) ) {
    1190           0 :             os << LogIO::WARN << "Invalid clean panel id used for interactive masking" << LogIO::POST;
    1191           0 :             return 0;
    1192             :         }
    1193             : 
    1194           0 :         if( Table::isReadable(mask) ) {
    1195           0 :             if ( ! Table::isWritable(mask) ) {
    1196           0 :                 os << LogIO::WARN << "Mask image is not modifiable " << LogIO::POST;
    1197           0 :                 return 0;
    1198             :             }
    1199             :             // we should regrid here if image and mask do not match
    1200             :         } else {
    1201           0 :             clone(image, mask);
    1202             :         }
    1203             :     
    1204           0 :         double startmask = maskSum(mask);
    1205             : 
    1206           0 :         if ( state->second.image_id == 0 || state->second.mask_id == 0 || forceReload ) {
    1207             : 
    1208             :             //Make sure image left after a "no more" is pressed is cleared
    1209           0 :             if ( forceReload && state->second.image_id !=0 )
    1210           0 :                 state->second.prev_image_id = state->second.image_id;
    1211           0 :             if ( forceReload && state->second.mask_id !=0 )
    1212           0 :                 state->second.prev_mask_id = state->second.mask_id;
    1213             : 
    1214           0 :             if ( state->second.prev_image_id ){
    1215           0 :                 if ( debug ) {
    1216           0 :                     std::cerr << "preparing to unload prev_image_id " <<
    1217           0 :                         state->second.prev_image_id << " (panel " << panel << ")" <<
    1218           0 :                         " (process " << getpid( ) << ", thread " << 
    1219           0 :                         std::this_thread::get_id() << ")" << std::endl;
    1220           0 :                     fflush(stderr);
    1221             :                 }
    1222           0 :                 unload( state->second.prev_image_id );
    1223             :             }
    1224           0 :             if ( state->second.prev_mask_id ) {
    1225           0 :                 if ( debug ) {
    1226           0 :                     std::cerr << "preparing to unload prev_mask_id " <<
    1227           0 :                         state->second.prev_mask_id << " (panel " << panel << ")" <<
    1228           0 :                         " (process " << getpid( ) << ", thread " << 
    1229           0 :                         std::this_thread::get_id() << ")" << std::endl;
    1230           0 :                     fflush(stderr);
    1231             :                 }
    1232           0 :                 unload( state->second.prev_mask_id );
    1233             :             }
    1234             : 
    1235           0 :             state->second.prev_image_id = 0;
    1236           0 :             state->second.prev_mask_id = 0;
    1237             : 
    1238             :             {
    1239           0 :                 grpc::ClientContext context;
    1240           0 :                 ::rpc::img::NewData nd;
    1241           0 :                 rpc::img::Id resp;
    1242           0 :                 nd.mutable_panel( )->set_id(panel);
    1243           0 :                 nd.set_path(image);
    1244           0 :                 nd.set_type("raster");
    1245           0 :                 nd.set_scale(0);
    1246           0 :                 viewer_proxy->load( &context, nd, &resp );
    1247           0 :                 state->second.image_id = resp.id( );
    1248           0 :             }
    1249             :             {
    1250           0 :                 grpc::ClientContext context;
    1251           0 :                 ::rpc::img::NewData nd;
    1252           0 :                 rpc::img::Id resp;
    1253           0 :                 nd.mutable_panel( )->set_id(panel);
    1254           0 :                 nd.set_path(mask);
    1255           0 :                 nd.set_type("contour");
    1256           0 :                 nd.set_scale(0);
    1257           0 :                 viewer_proxy->load( &context, nd, &resp );
    1258           0 :                 state->second.mask_id = resp.id( );
    1259           0 :             }
    1260             : 
    1261             :         } else {
    1262           0 :             grpc::ClientContext context;
    1263           0 :             ::rpc::img::Id id;
    1264           0 :             ::google::protobuf::Empty resp;
    1265           0 :             id.set_id(state->second.image_id);
    1266           0 :             viewer_proxy->reload( &context, id, &resp );
    1267           0 :             id.set_id(state->second.mask_id);
    1268           0 :             viewer_proxy->reload( &context, id, &resp );
    1269           0 :         }
    1270             : 
    1271           0 :         grpc::ClientContext context;
    1272           0 :         ::rpc::img::InteractiveMaskOptions options;
    1273           0 :         options.mutable_panel( )->set_id(state->first);
    1274           0 :         options.set_niter(niter);
    1275           0 :         options.set_cycleniter(cycleniter);
    1276           0 :         options.set_threshold(thresh);
    1277           0 :         options.set_cyclethreshold(cyclethresh);
    1278           0 :         ::rpc::img::InteractiveMaskResult imresult;
    1279           0 :         ::grpc::Status s = viewer_proxy->interactivemask( &context, options, &imresult );
    1280             : 
    1281           0 :         if ( ! s.ok( ) ) {
    1282           0 :             std::cerr << "interactive mask failed: " << s.error_details( ) << std::endl;
    1283           0 :             fflush(stderr);
    1284             :         }
    1285             : 
    1286           0 :         niter = imresult.state( ).niter( );
    1287           0 :         cycleniter = imresult.state( ).cycleniter( );
    1288           0 :         thresh = imresult.state( ).threshold( );
    1289           0 :         cyclethresh = imresult.state( ).cyclethreshold( );
    1290           0 :         int result = 1;
    1291           0 :         std::string action = imresult.action( );
    1292             : 
    1293           0 :         if ( debug ) {
    1294           0 :             std::cerr << "-------------------------------------------" << std::endl;
    1295           0 :             std::cerr << "  gui state from interactive masking" << std::endl;
    1296           0 :             std::cerr << "-------------------------------------------" << std::endl;
    1297           0 :             std::cerr << "           action: " << action << std::endl;
    1298           0 :             std::cerr << "            niter: " << niter << std::endl;
    1299           0 :             std::cerr << "      cycle niter: " << cycleniter << std::endl;
    1300           0 :             std::cerr << "        threshold: " << thresh << std::endl;
    1301           0 :             std::cerr << "  cycle threshold: " << cyclethresh << std::endl;
    1302           0 :             std::cerr << "-------------------------------------------" << std::endl;
    1303             :         }
    1304             : 
    1305           0 :         if ( action == "stop" ) result = 3;
    1306           0 :         else if ( action == "no more" ) result = 2;
    1307           0 :         else if ( action == "continue" ) result = 1;
    1308             :         else {
    1309           0 :             os << "ill-formed action result (" << action << ")" << LogIO::WARN << LogIO::POST;
    1310           0 :             return 0;
    1311             :         }
    1312             : 
    1313           0 :         state->second.prev_image_id = state->second.image_id;
    1314           0 :         state->second.prev_mask_id = state->second.mask_id;
    1315             :     
    1316           0 :         state->second.image_id = 0;
    1317           0 :         state->second.mask_id = 0;
    1318             : 
    1319           0 :         if ( debug ) {
    1320           0 :             std::cerr << "set prev_image_id to " << state->second.prev_image_id << " (panel " << panel << ")" <<
    1321           0 :                 " (process " << getpid( ) << ", thread " << 
    1322           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1323           0 :             std::cerr << "set prev_mask_id to " << state->second.prev_mask_id << " (panel " << panel << ")" <<
    1324           0 :                 " (process " << getpid( ) << ", thread " << 
    1325           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1326           0 :             fflush(stderr);
    1327             :         }
    1328             : 
    1329           0 :         double endmask = maskSum(mask);
    1330             : 
    1331           0 :         if( startmask != endmask ) {
    1332           0 :             result = -1 * result;
    1333           0 :             LogIO os( LogOrigin("grpcInteractiveCleanGui",__FUNCTION__,WHERE) );
    1334           0 :             os << "[" << mask << "] Mask modified from " << startmask << " pixels to " << endmask << " pixels " << LogIO::POST;
    1335           0 :         }
    1336             : 
    1337           0 :         return result;
    1338           0 :     }
    1339             : 
    1340           0 :     bool grpcInteractiveCleanGui::stop_viewer( ) {
    1341             :         // viewer is not running...
    1342           0 :         if ( ! viewer_started ) return false;
    1343           0 :         static const auto debug = getenv("GRPC_DEBUG");
    1344           0 :         if ( debug ) {
    1345           0 :             std::cerr << "sending shutdown message to viewer (" << viewer_uri << ")" << 
    1346           0 :                 " (process " << getpid( ) << ", thread " << 
    1347           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1348           0 :             fflush(stderr);
    1349             :         }
    1350             : 
    1351             :         // send shutdown message to viewer...
    1352           0 :         grpc::ClientContext context;
    1353           0 :         ::google::protobuf::Empty req;
    1354           0 :         ::google::protobuf::Empty resp;
    1355           0 :         auto shutdown = casatools::rpc::Shutdown::NewStub( grpc::CreateChannel( viewer_uri,
    1356           0 :                                                                                 grpc::InsecureChannelCredentials( ) ) );
    1357           0 :         shutdown->now( &context, req, &resp );
    1358             : 
    1359             :         // wait on viewer (appimage) to exit...
    1360             :         int status;
    1361           0 :         pid_t w = waitpid( viewer_pid, &status, WUNTRACED | WCONTINUED );
    1362           0 :         if ( w == -1 ){
    1363           0 :             if ( debug ) {
    1364             :                 std::cerr << "viewer process waitpid failed " <<
    1365           0 :                     " (process " << getpid( ) << ", thread " << 
    1366           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1367           0 :                 fflush(stderr);
    1368             :             }
    1369             :             // waitpid failed
    1370           0 :             return false;
    1371           0 :         } else if ( w == 0 ) {
    1372           0 :             if ( debug ) {
    1373             :                 std::cerr << "viewer process not found " <<
    1374           0 :                     " (process " << getpid( ) << ", thread " << 
    1375           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1376           0 :                 fflush(stderr);
    1377             :             }
    1378           0 :             return false;
    1379             :         } else {
    1380           0 :             if ( debug ) {
    1381             :                 std::cerr << "viewer process exited, status fetched " <<
    1382           0 :                     " (process " << getpid( ) << ", thread " << 
    1383           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1384           0 :                 fflush(stderr);
    1385             :             }
    1386           0 :             return true;
    1387             :         }
    1388             : 
    1389             :         viewer_pid = 0;
    1390             :         viewer_proxy.release( );
    1391             :         viewer_started = false;
    1392             :         return true;
    1393           0 :     }
    1394             : 
    1395           0 :     bool grpcInteractiveCleanGui::spawn_viewer( ) {
    1396           0 :         static const auto debug = getenv("GRPC_DEBUG");
    1397             : 
    1398           0 :         std::string viewer_path = get_viewer_path( );
    1399           0 :         if ( viewer_path.size( ) == 0 ) return false;
    1400             : 
    1401             :         // To minimize package size for distribution via pypi.org, the
    1402             :         // data repo has been moved out of the viewer appImage/app and
    1403             :         // into a separate package. The path to this needs to be specified
    1404             :         // when starting the viewer now...
    1405           0 :         std::string distro_data_path_arg = get_distro_data_path( );
    1406             : 
    1407             :         // sanity check on viewer path...
    1408             :         struct stat statbuf;
    1409           0 :         if ( stat( viewer_path.c_str( ), &statbuf ) < 0 ) {
    1410             :             // file (or dir) does not exist... e.g.
    1411             :             //   >>>>>>registry available at 0.0.0.0:40939
    1412             :             //   stopping registry<<<<<<
    1413           0 :             return false;
    1414             :         }
    1415             : 
    1416           0 :         std::string fifo = get_fifo( );
    1417           0 :         if ( fifo.size( ) == 0 ) return false;
    1418             : 
    1419             :         // here we start the viewer in a very basic manner... we do not bother
    1420             :         // with all of the theatrics needed to daemonize the launched process
    1421             :         // (see https://stackoverflow.com/questions/17954432/creating-a-daemon-in-linux)
    1422             :         // it could be that this should be done in the future, but for now we
    1423             :         // will adopt the simple...
    1424             : 
    1425           0 :         const int maxargc = 5;
    1426             :         char *arguments[maxargc];
    1427           0 :         for (int i = 0; i < maxargc; i++) { arguments[i] = (char*)""; };
    1428             : 
    1429           0 :         arguments[0] = strdup(viewer_path.c_str( ));
    1430           0 :         arguments[1] = (char*) malloc(sizeof(char) * (fifo.size( ) + 12));
    1431           0 :         sprintf( arguments[1], "--server=%s", fifo.c_str( ) );
    1432           0 :         arguments[2] = strdup("--oldregions");
    1433           0 :         int argc =3;
    1434           0 :         if ( distro_data_path_arg.size( ) > 0 ) {
    1435           0 :             distro_data_path_arg = std::string("--datapath=") + distro_data_path_arg;
    1436           0 :             arguments[argc] = strdup(distro_data_path_arg.c_str( ));
    1437           0 :             argc++;
    1438             :         }
    1439           0 :         std::string log_path = casatools::get_state( ).logPath( );
    1440           0 :         if ( log_path.size( ) > 0 ) {
    1441           0 :             arguments[argc] = (char*) malloc(sizeof(char) * (log_path.size( ) + 17));
    1442           0 :             sprintf( arguments[argc], "--casalogfile=%s", log_path.c_str( ) );
    1443           0 :             argc++;
    1444             :         }
    1445             : 
    1446           0 :         if ( debug ) {
    1447           0 :             std::cerr << "forking viewer process: ";
    1448           0 :             for (int i=0; i < argc; ++i) std::cout << arguments[i] << " ";
    1449           0 :             std::cerr << " (process " << getpid( ) << ", thread " << 
    1450           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1451           0 :             fflush(stderr);
    1452             :         }
    1453           0 :         pid_t pid = fork( );
    1454             : 
    1455           0 :             if ( pid == 0 ) {
    1456           0 :             if ( debug ) {
    1457           0 :                 std::cerr << "execing viewer process: ";
    1458           0 :                 for (int i=0; i < argc; ++i) std::cout << arguments[i] << " ";
    1459           0 :                 std::cerr << " (process " << getpid( ) << ", thread " << 
    1460           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1461           0 :                 fflush(stderr);
    1462             :             }
    1463           0 :             char **envp = getenv_sansmpi(); // bugfix: run the viewer without MPI CAS-13252
    1464           0 :             execle( arguments[0], arguments[0], arguments[1], arguments[2], arguments[3], arguments[4], NULL, envp );
    1465           0 :             perror( "grpcInteractiveCleanGui::launch(...) child process exec failed" );
    1466           0 :             exit(1);
    1467             :             }
    1468             : 
    1469           0 :             for ( int i=0; i < argc; ++i ) free(arguments[i]);
    1470             : 
    1471           0 :         if ( pid == -1 ) {
    1472           0 :             perror( "grpcInteractiveCleanGui::launch(...) child process fork failed" );
    1473           0 :             return false;
    1474             :         }
    1475             : 
    1476             :         // perform a health check, after a delay...
    1477             :         int status;
    1478           0 :         sleep(2);
    1479           0 :         pid_t w = waitpid( pid, &status, WUNTRACED | WCONTINUED | WNOHANG );
    1480           0 :         if ( w == -1 ){
    1481           0 :             if ( debug ) {
    1482             :                 std::cerr << "viewer process failed " <<
    1483           0 :                     " (process " << getpid( ) << ", thread " << 
    1484           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1485           0 :                 fflush(stderr);
    1486             :             }
    1487             :             // waitpid failed
    1488           0 :             return false;
    1489           0 :         } else if ( w != 0 ) {
    1490           0 :             if ( debug ) {
    1491             :                 std::cerr << "viewer process died " <<
    1492           0 :                     " (process " << getpid( ) << ", thread " << 
    1493           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1494           0 :                 fflush(stderr);
    1495             :             }
    1496             :             // process exited
    1497           0 :             if ( WIFEXITED(status) ) {
    1498           0 :                 printf("exited, status=%d\n", WEXITSTATUS(status));
    1499           0 :             } else if (WIFSIGNALED(status)) {
    1500           0 :                 printf("killed by signal %d\n", WTERMSIG(status));
    1501           0 :             } else if (WIFSTOPPED(status)) {
    1502           0 :                 printf("stopped by signal %d\n", WSTOPSIG(status));
    1503             :             }
    1504           0 :             return false;
    1505             :         }
    1506             : 
    1507           0 :         if ( debug ) {
    1508             :             std::cerr << "fetching viewer uri from " << fifo <<
    1509           0 :                 " (process " << getpid( ) << ", thread " << 
    1510           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1511           0 :             fflush(stderr);
    1512             :         }
    1513             :         char buffer[512];
    1514           0 :         std::string uri_buffer;
    1515           0 :         FILE *fp = fopen(fifo.c_str( ), "r");
    1516           0 :         while ( fgets( buffer, sizeof(buffer), fp ) ) { uri_buffer = uri_buffer + buffer; }
    1517           0 :         fclose(fp);
    1518           0 :         trim(uri_buffer);
    1519             : 
    1520             :         // validate viewer uri...
    1521           0 :         if ( ! std::regex_match( uri_buffer, std::regex("^([0-9]+\\.){3}[0-9]+:[0-9]+$") ) ) {
    1522             :             //rework of regex required for IPv6...
    1523           0 :             if ( debug ) {
    1524             :                 std::cerr << "bad viewer uri " << uri_buffer <<
    1525           0 :                     " (process " << getpid( ) << ", thread " << 
    1526           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1527           0 :                 fflush(stderr);
    1528             :             }
    1529           0 :             return false;
    1530             :         }
    1531             : 
    1532           0 :         if ( debug ) {
    1533             :             std::cerr << "received viewer uri: " << uri_buffer <<
    1534           0 :                 " (process " << getpid( ) << ", thread " << 
    1535           0 :                 std::this_thread::get_id() << ")" << std::endl;
    1536           0 :             fflush(stderr);
    1537             :         }
    1538             : 
    1539           0 :         viewer_uri = uri_buffer;
    1540           0 :         viewer_pid = pid;
    1541           0 :         viewer_proxy = rpc::img::view::NewStub( grpc::CreateChannel( viewer_uri,
    1542           0 :                                                                      grpc::InsecureChannelCredentials( ) ) );
    1543           0 :         viewer_started = true;
    1544             : 
    1545           0 :         return true;
    1546           0 :     }
    1547             : 
    1548           0 :     std::string grpcInteractiveCleanGui::get_python_path( ) {
    1549           0 :         std::string ret = casatools::get_state( ).pythonPath( );
    1550           0 :         return ret;
    1551             :     }
    1552             : 
    1553           0 :     std::string grpcInteractiveCleanGui::get_distro_data_path( ) {
    1554             :         static bool initialized = false;
    1555           0 :         static std::string result;
    1556           0 :         if ( initialized == false ) {
    1557           0 :             initialized = true;
    1558           0 :             result = casatools::get_state( ).distroDataPath( );
    1559             :             struct stat statbuf;
    1560           0 :             if ( stat( result.c_str( ), &statbuf ) < 0 ) {
    1561             :                 // file (or dir) does not exist...
    1562           0 :                 result = "";
    1563             :             }
    1564             :         }
    1565           0 :         return result;
    1566             :     }
    1567             : 
    1568           0 :     std::string grpcInteractiveCleanGui::get_viewer_path( ) {
    1569             :         // Get the path to the casaviewer Qt application, to be called in spawn_viewer()
    1570           0 :         std::string python_path = get_python_path( );
    1571           0 :         if ( python_path.size( ) == 0 ) return std::string( );
    1572             : 
    1573             :         //*** python3 -m casaviewer --app-path
    1574             :         char buffer[1024];
    1575           0 :         std::string result;
    1576           0 :         char **envp = getenv_sansmpi(); // bugfix: run the viewer without MPI CAS-13252
    1577           0 :         char *python_args[] = { (char*)python_path.c_str(), (char*)"-m", (char*)"casaviewer", (char*)"--app-path", NULL };
    1578           0 :         execve_getstdout((char*)python_path.c_str(), python_args, envp, buffer, 1024);
    1579           0 :         result = buffer;
    1580           0 :         free(envp);
    1581             : 
    1582           0 :         trim(result);
    1583           0 :         if ( result.size( ) == 0 ) return std::string( );
    1584           0 :         return result;
    1585           0 :     }
    1586             : 
    1587           0 :     std::string grpcInteractiveCleanGui::get_fifo( ) {
    1588           0 :         static const char *env_tmpdir = getenv("TMPDIR");
    1589           0 :         static std::string fifo_template = trim_trailing_slash(env_tmpdir && isdir(env_tmpdir) ? env_tmpdir : P_tmpdir) + "/vwr-XXXXXXXXXX";
    1590           0 :         static int fifo_template_size = fifo_template.size( );
    1591           0 :         char fifo_path[fifo_template_size+1];
    1592           0 :         strncpy( fifo_path, fifo_template.c_str( ), fifo_template_size );
    1593           0 :         fifo_path[fifo_template_size] = '\0';
    1594           0 :         int fd = mkstemp(fifo_path);
    1595           0 :         if ( fd == -1 ) throw std::runtime_error("mkstemp failed...");
    1596           0 :         close( fd );
    1597           0 :         unlink(fifo_path);
    1598           0 :         mkfifo( fifo_path, 0666 );
    1599           0 :         return fifo_path;
    1600           0 :     }
    1601             : 
    1602           0 :     casacore::Record grpcInteractiveCleanManager::pauseForUserInteraction( ) {
    1603           0 :                 LogIO os( LogOrigin("grpcInteractiveCleanManager",__FUNCTION__,WHERE) );
    1604           0 :         static const auto debug = getenv("GRPC_DEBUG");
    1605             : 
    1606           0 :         if ( clean_images.size( ) == 0 ) {
    1607             :             // cannot open clean panel in viewer if not images are available...
    1608           0 :             if ( debug ) {
    1609             :                 std::cerr << "no clean images available" <<
    1610           0 :                     " (process " << getpid( ) << ", thread " << 
    1611           0 :                     std::this_thread::get_id() << ")" << std::endl;
    1612           0 :                 fflush(stderr);
    1613             :             }
    1614           0 :             return Record( );
    1615             :         }
    1616             : 
    1617           0 :         if ( clean_panel_id == -1 || ! gui.alive( ) ) {
    1618             :             // open panel if it is not already open...
    1619           0 :             clean_panel_id = gui.open_panel( clean_images );
    1620             :         }
    1621             :         
    1622           0 :         int niter=0,cycleniter=0,iterdone;
    1623           0 :         float threshold=0.0, cyclethreshold=0.0;
    1624           0 :         access( (void*) 0,
    1625           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
    1626           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
    1627           0 :                            niter = state.Niter;
    1628           0 :                            cycleniter = state.CycleNiter;
    1629           0 :                            threshold = state.Threshold;
    1630           0 :                            cyclethreshold = state.CycleThreshold;
    1631           0 :                            iterdone = state.IterDone;
    1632           0 :                            return dummy;
    1633             :                        } ) );
    1634             :                            
    1635           0 :         std::string strthresh = std::to_string(threshold)+"Jy";
    1636           0 :         std::string strcycthresh = std::to_string(cyclethreshold)+"Jy";
    1637             :                   
    1638           0 :         int iterleft = niter - iterdone;
    1639           0 :         if( iterleft<0 ) iterleft=0;
    1640             : 
    1641           0 :         casacore::Vector<int> itsActionCodes(clean_images.size( ));
    1642           0 :         itsActionCodes = 1.0;
    1643             : 
    1644           0 :         unsigned ind = 0;
    1645           0 :         for ( auto it = clean_images.begin( ); it != clean_images.end( ); ++it, ++ind ) {
    1646           0 :             if ( std::get<2>(*it) ) {
    1647           0 :                 itsActionCodes[ind] = std::get<3>(*it);
    1648           0 :                 continue;
    1649             :             }
    1650           0 :             if ( fabs(itsActionCodes[ind]) == 1.0 ) {
    1651           0 :                 std::string imageName = std::get<0>(*it) + ".residual" + ( std::get<1>(*it) ? ".tt0" : "" );
    1652           0 :                 std::string maskName = std::get<0>(*it) + ".mask";
    1653           0 :                 std::string last_strcycthresh = strcycthresh;
    1654           0 :                 itsActionCodes[ind] = gui.interactivemask( clean_panel_id, imageName, maskName, iterleft,
    1655             :                                                            cycleniter, strthresh, strcycthresh );
    1656             : 
    1657           0 :                 if ( strcycthresh != last_strcycthresh ) {
    1658           0 :                     access( (void*) 0,
    1659           0 :                             std::function< void* ( void*, grpcInteractiveCleanState& )>(
    1660           0 :                                 [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
    1661             :                                     // if this is not set to false, the users cyclethreshold
    1662             :                                     // change are recomputed...
    1663           0 :                                     state.IsCycleThresholdAuto = false;
    1664           0 :                                     return dummy;
    1665             :                                 } ) );
    1666             :                 }
    1667             : 
    1668           0 :                 if( itsActionCodes[ind] < 0 ) os << "[" << std::get<0>(*it) <<"] Mask changed interactively." << LogIO::POST;
    1669           0 :                 if( fabs(itsActionCodes[ind])==3 || fabs(itsActionCodes[ind])==2 ) {
    1670             :                     // fabs(itsActionCodes[ind])==3   -->   stop
    1671             :                     // fabs(itsActionCodes[ind])==2   -->   no more
    1672           0 :                     std::get<2>(*it) = true;
    1673           0 :                     std::get<3>(*it) = fabs(itsActionCodes[ind]);
    1674             :                 }
    1675           0 :             }
    1676             :         }
    1677             : 
    1678             : 
    1679           0 :         Quantity qa;
    1680           0 :         casacore::Quantity::read(qa,strthresh);
    1681           0 :         threshold = qa.getValue(Unit("Jy"));
    1682             : 
    1683             : 
    1684           0 :         float oldcyclethreshold = cyclethreshold;
    1685           0 :         Quantity qb;
    1686           0 :         casacore::Quantity::read(qb,strcycthresh);
    1687           0 :         cyclethreshold = qb.getValue(Unit("Jy"));
    1688             : 
    1689           0 :         access( (void*) 0,
    1690           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
    1691           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
    1692           0 :                            if ( debug ) {
    1693           0 :                                std::cerr << "-------------------------------------------" << std::endl;
    1694           0 :                                std::cerr << "  exporting gui state: " << std::endl;
    1695           0 :                                std::cerr << "-------------------------------------------" << std::endl;
    1696           0 :                                std::cerr << "    Niter            " << state.Niter <<
    1697           0 :                                    " ---> " << iterdone+iterleft << std::endl;
    1698           0 :                                std::cerr << "    CycleNiter       " << state.CycleNiter <<
    1699           0 :                                    " ---> " << cycleniter << std::endl;
    1700           0 :                                std::cerr << "    Threshold        " << state.Threshold <<
    1701           0 :                                    " ---> " << threshold << std::endl;
    1702           0 :                                std::cerr << "    CycleThreshold   " << oldcyclethreshold <<
    1703           0 :                                    ( fabs( cyclethreshold - oldcyclethreshold ) > 1e-06 &&
    1704           0 :                                      cyclethreshold != 0 && oldcyclethreshold != 0 ?
    1705           0 :                                     " ---> " : " -x-> ") << cyclethreshold << std::endl;
    1706           0 :                                std::cerr << "-------------------------------------------" << std::endl;
    1707             :                            }
    1708             : 
    1709           0 :                            state.Niter = iterdone+iterleft;
    1710           0 :                            state.CycleNiter = cycleniter;
    1711           0 :                            state.Threshold = threshold;
    1712           0 :                            if ( cyclethreshold != 0 && oldcyclethreshold != 0 &&
    1713           0 :                                 fabs( cyclethreshold - oldcyclethreshold ) > 1e-06 )
    1714           0 :                                state.CycleThreshold = cyclethreshold;
    1715             : 
    1716           0 :                            return dummy;
    1717             :                        } ) );
    1718             : 
    1719           0 :                   Bool alldone=true;
    1720           0 :                   for ( ind = 0; ind < clean_images.size( ); ++ind ) {
    1721           0 :                       alldone = alldone & ( fabs(itsActionCodes[ind])==3 );
    1722             :                   }
    1723           0 :                   if( alldone==true ) changeStopFlag( true );
    1724             : 
    1725           0 :           Record returnRec;
    1726           0 :           for( ind = 0; ind < clean_images.size( ); ind++ ){
    1727           0 :               returnRec.define( RecordFieldId( String::toString(ind)), itsActionCodes[ind] );
    1728             :           }
    1729             : 
    1730           0 :           return returnRec;
    1731           0 :         }
    1732             : 
    1733           0 :     void grpcInteractiveCleanManager::closePanel( ) {
    1734           0 :         gui.close_panel(clean_panel_id);
    1735           0 :         clean_panel_id = -1;
    1736           0 :         clean_images.clear( );
    1737           0 :         access( (void*) 0,
    1738           0 :                 std::function< void* ( void*, grpcInteractiveCleanState& )>(
    1739           0 :                        [&]( void *dummy, grpcInteractiveCleanState &state ) -> void* {
    1740           0 :                            state.reset( );
    1741           0 :                            return dummy; } ) );
    1742           0 :     }
    1743             : 
    1744             : } //# NAMESPACE CASA - END

Generated by: LCOV version 1.16