Blame src/QueueFeeder.cc

Packit 8f70b4
/*
Packit 8f70b4
 * lftp - file transfer program
Packit 8f70b4
 *
Packit 8f70b4
 * Copyright (c) 1996-2012 by Alexander V. Lukyanov (lav@yars.free.net)
Packit 8f70b4
 *
Packit 8f70b4
 * This program is free software; you can redistribute it and/or modify
Packit 8f70b4
 * it under the terms of the GNU General Public License as published by
Packit 8f70b4
 * the Free Software Foundation; either version 3 of the License, or
Packit 8f70b4
 * (at your option) any later version.
Packit 8f70b4
 *
Packit 8f70b4
 * This program is distributed in the hope that it will be useful,
Packit 8f70b4
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 8f70b4
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 8f70b4
 * GNU General Public License for more details.
Packit 8f70b4
 *
Packit 8f70b4
 * You should have received a copy of the GNU General Public License
Packit 8f70b4
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Packit 8f70b4
 */
Packit 8f70b4
Packit 8f70b4
/* All indexes in this function start at 0; -1 is used contextually to
Packit 8f70b4
 * indicate the last job or moving to the end of the list. */
Packit 8f70b4
#include <config.h>
Packit 8f70b4
#include <unistd.h>
Packit 8f70b4
#include <assert.h>
Packit 8f70b4
#include <fnmatch.h>
Packit 8f70b4
#include <stddef.h>
Packit 8f70b4
Packit 8f70b4
#include "QueueFeeder.h"
Packit 8f70b4
#include "plural.h"
Packit 8f70b4
#include "misc.h"
Packit 8f70b4
Packit 8f70b4
const char *QueueFeeder::NextCmd(CmdExec *exec, const char *)
Packit 8f70b4
{
Packit 8f70b4
   if(jobs == NULL) return NULL;
Packit 8f70b4
Packit 8f70b4
   /* denext the first job */
Packit 8f70b4
   QueueJob *job = grab_job(0);
Packit 8f70b4
Packit 8f70b4
   buffer.truncate(0);
Packit 8f70b4
Packit 8f70b4
   if(xstrcmp(cur_pwd, job->pwd)) {
Packit 8f70b4
      buffer.append("cd ").append_quoted(job->pwd).append("; ");
Packit 8f70b4
      cur_pwd.set(job->pwd);
Packit 8f70b4
   }
Packit 8f70b4
Packit 8f70b4
   if(xstrcmp(cur_lpwd, job->lpwd)) {
Packit 8f70b4
      buffer.append("lcd ").append_quoted(job->lpwd).append("; ");
Packit 8f70b4
      cur_lpwd.set(job->lpwd);
Packit 8f70b4
   }
Packit 8f70b4
Packit 8f70b4
   buffer.append(job->cmd.get()).append('\n');
Packit 8f70b4
   delete job;
Packit 8f70b4
   return buffer;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
void QueueFeeder::QueueCmd(const char *cmd, const char *pwd, const char *lpwd, int pos, int v)
Packit 8f70b4
{
Packit 8f70b4
   QueueJob *job = new QueueJob;
Packit 8f70b4
   job->cmd.set(cmd);
Packit 8f70b4
   job->pwd.set(pwd);
Packit 8f70b4
   job->lpwd.set(lpwd);
Packit 8f70b4
Packit 8f70b4
   /* we never want a newline at the end: */
Packit 8f70b4
   if(last_char(job->cmd) == '\n')
Packit 8f70b4
      job->cmd.truncate(strlen(job->cmd)-1);
Packit 8f70b4
Packit 8f70b4
   insert_jobs(job, jobs, lastjob, pos != -1? get_job(pos): NULL);
Packit 8f70b4
   PrintJobs(job, v, _("Added job$|s$"));
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
int QueueFeeder::JobCount(const QueueJob *j)
Packit 8f70b4
{
Packit 8f70b4
   int job_count=0;
Packit 8f70b4
   for(; j; j=j->next)
Packit 8f70b4
      job_count++;
Packit 8f70b4
   return job_count;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
/* verbose:
Packit 8f70b4
 * 0, quiet
Packit 8f70b4
 * 1, interactive
Packit 8f70b4
 * 2, verbose (print changes of pwd and lpwd)
Packit 8f70b4
 * PrintRequeue, output to requeue
Packit 8f70b4
 */
Packit 8f70b4
xstring& QueueFeeder::FormatJobs(xstring& s,const QueueJob *job, int v, const char *plur) const
Packit 8f70b4
{
Packit 8f70b4
   if(v < 1)
Packit 8f70b4
      return s;
Packit 8f70b4
Packit 8f70b4
   const char *pwd = 0, *lpwd = 0;
Packit 8f70b4
   if(v == PrintRequeue)
Packit 8f70b4
   {
Packit 8f70b4
      for(const QueueJob *j = job; j; j=j->next)
Packit 8f70b4
      {
Packit 8f70b4
	 if(xstrcmp(pwd, job->pwd))
Packit 8f70b4
	 {
Packit 8f70b4
	    s.append("cd ").append_quoted(job->pwd).append(" &\n");
Packit 8f70b4
	    pwd = job->pwd;
Packit 8f70b4
	 }
Packit 8f70b4
Packit 8f70b4
	 if(xstrcmp(lpwd, job->lpwd))
Packit 8f70b4
	 {
Packit 8f70b4
	    s.append("lcd ").append_quoted(job->lpwd).append(" &\n");
Packit 8f70b4
	    lpwd = job->lpwd;
Packit 8f70b4
	 }
Packit 8f70b4
Packit 8f70b4
	 s.append("queue ").append_quoted(job->cmd).append('\n');
Packit 8f70b4
      }
Packit 8f70b4
      return s;
Packit 8f70b4
   }
Packit 8f70b4
Packit 8f70b4
   int job_count=JobCount(job);
Packit 8f70b4
   if(job_count>1)
Packit 8f70b4
      s.appendf("%s:\n", plural(plur,job_count));
Packit 8f70b4
Packit 8f70b4
   pwd = cur_pwd;
Packit 8f70b4
   lpwd = cur_lpwd;
Packit 8f70b4
Packit 8f70b4
   int n = 1;
Packit 8f70b4
   for(const QueueJob *j = job; j; j=j->next)
Packit 8f70b4
   {
Packit 8f70b4
      /* Print pwd/lpwd changes when v >= 2.  (This only happens when there's
Packit 8f70b4
       * more than one.) */
Packit 8f70b4
      if(xstrcmp(pwd, job->pwd))
Packit 8f70b4
      {
Packit 8f70b4
	 if(v > 2) {
Packit 8f70b4
	    s.append("\tcd ").append_quoted(job->pwd).append('\n');
Packit 8f70b4
	 }
Packit 8f70b4
	 pwd = job->pwd;
Packit 8f70b4
      }
Packit 8f70b4
Packit 8f70b4
      if(xstrcmp(lpwd, job->lpwd))
Packit 8f70b4
      {
Packit 8f70b4
	 if(v > 2) {
Packit 8f70b4
	    s.append("\tlcd ").append_quoted(job->lpwd).append('\n');
Packit 8f70b4
	 }
Packit 8f70b4
	 lpwd = job->lpwd;
Packit 8f70b4
      }
Packit 8f70b4
Packit 8f70b4
      if(job_count==1)
Packit 8f70b4
	 s.appendf("%s: ", plural(plur,job_count));
Packit 8f70b4
      else
Packit 8f70b4
	 s.appendf("\t%2d. ",n++);
Packit 8f70b4
Packit 8f70b4
      s.append(j->cmd.get()).append('\n');
Packit 8f70b4
   }
Packit 8f70b4
   return s;
Packit 8f70b4
}
Packit 8f70b4
void QueueFeeder::PrintJobs(const QueueJob *job, int v, const char *plur) const
Packit 8f70b4
{
Packit 8f70b4
   xstring buf("");
Packit 8f70b4
   FormatJobs(buf,job,v,plur);
Packit 8f70b4
   printf("%s",buf.get());
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
bool QueueFeeder::DelJob(int from, int v)
Packit 8f70b4
{
Packit 8f70b4
   QueueJob *job = grab_job(from);
Packit 8f70b4
   if(!job)
Packit 8f70b4
   {
Packit 8f70b4
      if(v > 0)
Packit 8f70b4
      {
Packit 8f70b4
	 if(from == -1 || !jobs)
Packit 8f70b4
	    printf(_("No queued jobs.\n"));
Packit 8f70b4
	 else
Packit 8f70b4
	    printf(_("No queued job #%i.\n"), from+1);
Packit 8f70b4
      }
Packit 8f70b4
      return false;
Packit 8f70b4
   }
Packit 8f70b4
Packit 8f70b4
   PrintJobs(job, v, _("Deleted job$|s$"));
Packit 8f70b4
Packit 8f70b4
   FreeList(job);
Packit 8f70b4
   return true;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
bool QueueFeeder::DelJob(const char *cmd, int v)
Packit 8f70b4
{
Packit 8f70b4
   QueueJob *job = grab_job(cmd);
Packit 8f70b4
   if(!job)
Packit 8f70b4
   {
Packit 8f70b4
      if(v > 0)
Packit 8f70b4
      {
Packit 8f70b4
	 if(!jobs)
Packit 8f70b4
	    printf(_("No queued jobs.\n"));
Packit 8f70b4
	 else
Packit 8f70b4
	    printf(_("No queued jobs match \"%s\".\n"), cmd);
Packit 8f70b4
      }
Packit 8f70b4
      return false;
Packit 8f70b4
   }
Packit 8f70b4
Packit 8f70b4
   PrintJobs(job, v, _("Deleted job$|s$"));
Packit 8f70b4
Packit 8f70b4
   FreeList(job);
Packit 8f70b4
   return true;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
/* When moving, grab the insertion pointer *before* pulling out things to
Packit 8f70b4
 * move, since doing so will change offsets.  (Note that "to == -1" means
Packit 8f70b4
 * "move to the end", not "before the last entry".)
Packit 8f70b4
 */
Packit 8f70b4
bool QueueFeeder::MoveJob(int from, int to, int v)
Packit 8f70b4
{
Packit 8f70b4
   /* Safety: make sure we don't try to move an item before itself. */
Packit 8f70b4
   if(from == to) return false;
Packit 8f70b4
Packit 8f70b4
   QueueJob *before = to != -1? get_job(to): NULL;
Packit 8f70b4
Packit 8f70b4
   QueueJob *job = grab_job(from);
Packit 8f70b4
   if(job == NULL) return false;
Packit 8f70b4
Packit 8f70b4
   PrintJobs(job, v, _("Moved job$|s$"));
Packit 8f70b4
   assert(job != before);
Packit 8f70b4
Packit 8f70b4
   insert_jobs(job, jobs, lastjob, before);
Packit 8f70b4
   return true;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
bool QueueFeeder::MoveJob(const char *cmd, int to, int v)
Packit 8f70b4
{
Packit 8f70b4
   QueueJob *before = to != -1? get_job(to): NULL;
Packit 8f70b4
Packit 8f70b4
   /* Mild hack: we need to make sure the "before" job isn't one that's
Packit 8f70b4
    * going to be moved, so move it upward until it isn't. */
Packit 8f70b4
   while(before && !fnmatch(cmd, before->cmd,FNM_CASEFOLD)) before=before->next;
Packit 8f70b4
Packit 8f70b4
   QueueJob *job = grab_job(cmd);
Packit 8f70b4
   if(job == NULL) return false;
Packit 8f70b4
Packit 8f70b4
   PrintJobs(job, v, _("Moved job$|s$"));
Packit 8f70b4
Packit 8f70b4
   insert_jobs(job, jobs, lastjob, before);
Packit 8f70b4
   return true;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
/* remove the given job from the list */
Packit 8f70b4
void QueueFeeder::unlink_job(QueueJob *job)
Packit 8f70b4
{
Packit 8f70b4
   /* update head/tail */
Packit 8f70b4
   if(!job->prev) jobs = jobs->next;
Packit 8f70b4
   if(!job->next) lastjob = lastjob->prev;
Packit 8f70b4
Packit 8f70b4
   /* linked list stuff */
Packit 8f70b4
   if(job->prev) job->prev->next = job->next;
Packit 8f70b4
   if(job->next) job->next->prev = job->prev;
Packit 8f70b4
   job->prev = job->next = 0;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
QueueFeeder::QueueJob *QueueFeeder::get_job(int n)
Packit 8f70b4
{
Packit 8f70b4
   QueueJob *j;
Packit 8f70b4
   if(n == -1) {
Packit 8f70b4
      j = lastjob;
Packit 8f70b4
   } else {
Packit 8f70b4
      j = jobs;
Packit 8f70b4
      while(j && n--) j=j->next;
Packit 8f70b4
   }
Packit 8f70b4
Packit 8f70b4
   return j;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
/* get the n'th job, removed from the list; returns NULL (an empty list)
Packit 8f70b4
 * if there aren't that many jobs: */
Packit 8f70b4
QueueFeeder::QueueJob *QueueFeeder::grab_job(int n)
Packit 8f70b4
{
Packit 8f70b4
   QueueJob *j = get_job(n);
Packit 8f70b4
   if(j)
Packit 8f70b4
      unlink_job(j);
Packit 8f70b4
   return j;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
QueueFeeder::QueueJob *QueueFeeder::grab_job(const char *cmd)
Packit 8f70b4
{
Packit 8f70b4
   QueueJob *j = jobs, *head = NULL, *tail = NULL;
Packit 8f70b4
Packit 8f70b4
   while(j) {
Packit 8f70b4
      QueueJob *match = get_next_match(cmd, j);
Packit 8f70b4
      if(!match) break;
Packit 8f70b4
      j = match->next;
Packit 8f70b4
Packit 8f70b4
      /* matches */
Packit 8f70b4
      unlink_job(match);
Packit 8f70b4
      insert_jobs(match, head, tail, NULL);
Packit 8f70b4
   }
Packit 8f70b4
Packit 8f70b4
   return head;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
QueueFeeder::QueueJob *QueueFeeder::get_next_match(const char *cmd, QueueJob *j)
Packit 8f70b4
{
Packit 8f70b4
   while(j) {
Packit 8f70b4
      if(!fnmatch(cmd, j->cmd,FNM_CASEFOLD))
Packit 8f70b4
	 return j;
Packit 8f70b4
Packit 8f70b4
      j = j->next;
Packit 8f70b4
   }
Packit 8f70b4
   return 0;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
/* insert a list of jobs before "before", or at the end if before is NULL.
Packit 8f70b4
 * If before is not NULL, it must be contained between lst_head and lst_tail. */
Packit 8f70b4
void QueueFeeder::insert_jobs(QueueJob *job,
Packit 8f70b4
      			      QueueJob *&lst_head,
Packit 8f70b4
      			      QueueJob *&lst_tail,
Packit 8f70b4
			      QueueJob *before)
Packit 8f70b4
{
Packit 8f70b4
   assert(!job->prev); /* this should be an independant, clean list head */
Packit 8f70b4
Packit 8f70b4
   /* Find the last entry in the new list.  (This is a bit inefficient, as
Packit 8f70b4
    * we usually know this somewhere else, but passing around both head
Packit 8f70b4
    * and tail pointers of the new job list is too klugy.) */
Packit 8f70b4
   QueueJob *tail = job;
Packit 8f70b4
   while(tail->next) tail=tail->next;
Packit 8f70b4
Packit 8f70b4
   if(!before) {
Packit 8f70b4
      /* end */
Packit 8f70b4
      job->prev = lst_tail;
Packit 8f70b4
      tail->next = 0; /* superfluous; here for clarity */
Packit 8f70b4
   } else {
Packit 8f70b4
      tail->next = before;
Packit 8f70b4
      job->prev = before->prev;
Packit 8f70b4
   }
Packit 8f70b4
Packit 8f70b4
   if(job->prev) job->prev->next = job;
Packit 8f70b4
   if(tail->next) tail->next->prev = tail;
Packit 8f70b4
   if(!tail->next) lst_tail = tail;
Packit 8f70b4
   if(!job->prev) lst_head = job;
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
/* Free a list of jobs (forward only; j should be a head pointer.) */
Packit 8f70b4
void QueueFeeder::FreeList(QueueJob *j)
Packit 8f70b4
{
Packit 8f70b4
   while(j) {
Packit 8f70b4
      QueueJob *job = j;
Packit 8f70b4
      j = j->next;
Packit 8f70b4
      delete job;
Packit 8f70b4
   }
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
QueueFeeder::~QueueFeeder()
Packit 8f70b4
{
Packit 8f70b4
   FreeList(jobs);
Packit 8f70b4
}
Packit 8f70b4
Packit 8f70b4
Packit 8f70b4
xstring& QueueFeeder::FormatStatus(xstring& s,int v,const char *prefix) const
Packit 8f70b4
{
Packit 8f70b4
   if(jobs == NULL)
Packit 8f70b4
      return s;
Packit 8f70b4
Packit 8f70b4
   if(v == PrintRequeue)
Packit 8f70b4
      return FormatJobs(s, jobs, v, "");
Packit 8f70b4
Packit 8f70b4
   s.append(prefix).append(_("Commands queued:")).append('\n');
Packit 8f70b4
Packit 8f70b4
   int n = 1;
Packit 8f70b4
Packit 8f70b4
   const char *pwd = cur_pwd, *lpwd = cur_lpwd;
Packit 8f70b4
   for(const QueueJob *job = jobs; job; job = job->next) {
Packit 8f70b4
      if(v<2 && n>4 && job->next)
Packit 8f70b4
      {
Packit 8f70b4
	 s.appendf("%s%2d. ...\n",prefix,n);
Packit 8f70b4
	 break;
Packit 8f70b4
      }
Packit 8f70b4
      /* Print pwd/lpwd changes when v >= 2. */
Packit 8f70b4
      if(v >= 2 && (xstrcmp(pwd, job->pwd)))
Packit 8f70b4
	 s.appendf("%s    cd %s\n",prefix,job->pwd.get());
Packit 8f70b4
      if(v >= 2 && (xstrcmp(lpwd, job->lpwd)))
Packit 8f70b4
	 s.appendf("%s    lcd %s\n",prefix,job->lpwd.get());
Packit 8f70b4
      pwd = job->pwd;
Packit 8f70b4
      lpwd = job->lpwd;
Packit 8f70b4
Packit 8f70b4
      s.appendf("%s%2d. %s\n",prefix,n++,job->cmd.get());
Packit 8f70b4
   }
Packit 8f70b4
   return s;
Packit 8f70b4
}