Blame src/FileSetOutput.cc

Packit Service a2489d
/*
Packit Service a2489d
 * lftp - file transfer program
Packit Service a2489d
 *
Packit Service a2489d
 * Copyright (c) 1996-2017 by Alexander V. Lukyanov (lav@yars.free.net)
Packit Service a2489d
 *
Packit Service a2489d
 * This program is free software; you can redistribute it and/or modify
Packit Service a2489d
 * it under the terms of the GNU General Public License as published by
Packit Service a2489d
 * the Free Software Foundation; either version 3 of the License, or
Packit Service a2489d
 * (at your option) any later version.
Packit Service a2489d
 *
Packit Service a2489d
 * This program is distributed in the hope that it will be useful,
Packit Service a2489d
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service a2489d
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service a2489d
 * GNU General Public License for more details.
Packit Service a2489d
 *
Packit Service a2489d
 * You should have received a copy of the GNU General Public License
Packit Service a2489d
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Packit Service a2489d
 */
Packit Service a2489d
Packit Service a2489d
#include <config.h>
Packit Service a2489d
#include "FileSet.h"
Packit Service a2489d
Packit Service a2489d
#include <sys/types.h>
Packit Service a2489d
#include <sys/stat.h>
Packit Service a2489d
#include <utime.h>
Packit Service a2489d
#include <stdlib.h>
Packit Service a2489d
#include <unistd.h>
Packit Service a2489d
#include <sys/stat.h>
Packit Service a2489d
#include <fnmatch.h>
Packit Service a2489d
#include <locale.h>
Packit Service a2489d
#include <mbswidth.h>
Packit Service a2489d
Packit Service a2489d
CDECL_BEGIN
Packit Service a2489d
#include <filemode.h>
Packit Service a2489d
#include "human.h"
Packit Service a2489d
CDECL_END
Packit Service a2489d
Packit Service a2489d
#include "misc.h"
Packit Service a2489d
#include "ResMgr.h"
Packit Service a2489d
Packit Service a2489d
#include "FileSetOutput.h"
Packit Service a2489d
#include "ArgV.h"
Packit Service a2489d
#include "ColumnOutput.h"
Packit Service a2489d
#include "DirColors.h"
Packit Service a2489d
#include "FileGlob.h"
Packit Service a2489d
#include "CopyJob.h"
Packit Service a2489d
Packit Service a2489d
Packit Service a2489d
ResDecl	res_default_cls         ("cmd:cls-default",  "-F", FileSetOutput::ValidateArgv,ResMgr::NoClosure),
Packit Service a2489d
	res_default_comp_cls    ("cmd:cls-completion-default", "-FBa",FileSetOutput::ValidateArgv,ResMgr::NoClosure);
Packit Service a2489d
Packit Service a2489d
ResDecl res_time_style	("cmd:time-style", "%b %e  %Y|%b %e %H:%M", 0, ResMgr::NoClosure);
Packit Service a2489d
Packit Service a2489d
/* note: this sorts (add a nosort if necessary) */
Packit Service a2489d
void FileSetOutput::print(FileSet &fs, const JobRef<OutputJob>& o) const
Packit Service a2489d
{
Packit Service a2489d
   fs.Sort(sort, sort_casefold, sort_reverse);
Packit Service a2489d
   if(sort_dirs_first) fs.Sort(FileSet::DIRSFIRST, false, sort_reverse);
Packit Service a2489d
Packit Service a2489d
   ColumnOutput c;
Packit Service a2489d
Packit Service a2489d
   DirColors &col=*DirColors::GetInstance();
Packit Service a2489d
Packit Service a2489d
   const char *suffix_color = "";
Packit Service a2489d
Packit Service a2489d
   /* Most fields are only printed if at least one file has that
Packit Service a2489d
    * information; if no files have perm information, for example,
Packit Service a2489d
    * discard the entire field. */
Packit Service a2489d
   int have = fs.Have();
Packit Service a2489d
Packit Service a2489d
   for(int i = 0; fs[i]; i++) {
Packit Service a2489d
      const FileInfo *f = fs[i];
Packit Service a2489d
      if(!showdots && !list_directories &&
Packit Service a2489d
	    (!strcmp(basename_ptr(f->name),".") || !strcmp(basename_ptr(f->name),"..")))
Packit Service a2489d
	 continue;
Packit Service a2489d
Packit Service a2489d
      if(pat && *pat &&
Packit Service a2489d
	    fnmatch(pat, f->name, patterns_casefold? FNM_CASEFOLD:0))
Packit Service a2489d
	 continue;
Packit Service a2489d
Packit Service a2489d
      c.append();
Packit Service a2489d
Packit Service a2489d
      if((mode & PERMS) && (f->defined&FileInfo::MODE)) {
Packit Service a2489d
	 char mode[16];
Packit Service a2489d
	 memset(mode, 0, sizeof(mode));
Packit Service a2489d
	 strmode(f->mode, mode);
Packit Service a2489d
	 /* FIXME: f->mode doesn't have type info; it wouldn't
Packit Service a2489d
	  * be hard to fix that */
Packit Service a2489d
	 if(f->filetype == FileInfo::DIRECTORY) mode[0] = 'd';
Packit Service a2489d
	 else if(f->filetype == FileInfo::SYMLINK) mode[0] = 'l';
Packit Service a2489d
	 else mode[0] = '-';
Packit Service a2489d
Packit Service a2489d
	 c.add(mode, "");
Packit Service a2489d
      } else if(have & FileInfo::MODE) {
Packit Service a2489d
	 c.add("           ", "");
Packit Service a2489d
      }
Packit Service a2489d
Packit Service a2489d
      if((have & FileInfo::NLINKS) && (mode & NLINKS)) {
Packit Service a2489d
	 if(f->defined&f->NLINKS)
Packit Service a2489d
	    c.addf("%4i ", "", f->nlinks);
Packit Service a2489d
	 else
Packit Service a2489d
	    c.addf("%4i ", "", "");
Packit Service a2489d
      }
Packit Service a2489d
Packit Service a2489d
      if((have & FileInfo::USER) && (mode & USER)) {
Packit Service a2489d
	 c.addf("%-8.8s ", "", (f->defined&f->USER)? f->user: "");
Packit Service a2489d
      }
Packit Service a2489d
Packit Service a2489d
      if((have & FileInfo::GROUP) && (mode & GROUP)) {
Packit Service a2489d
	 c.addf("%-8.8s ", "", (f->defined&f->GROUP)? f->group: "");
Packit Service a2489d
      }
Packit Service a2489d
Packit Service a2489d
      if((mode & SIZE) && (have&FileInfo::SIZE)) {
Packit Service a2489d
	 char sz[LONGEST_HUMAN_READABLE + 2];
Packit Service a2489d
	 if((f->filetype == FileInfo::NORMAL || !size_filesonly)
Packit Service a2489d
	 && (f->defined&f->SIZE)) {
Packit Service a2489d
	    char buffer[LONGEST_HUMAN_READABLE + 1];
Packit Service a2489d
	    snprintf(sz, sizeof(sz), "%8s ",
Packit Service a2489d
	       human_readable (f->size, buffer, human_opts, 1,
Packit Service a2489d
		  output_block_size? output_block_size:1024));
Packit Service a2489d
	 } else {
Packit Service a2489d
	    snprintf(sz, sizeof(sz), "%8s ", ""); /* pad */
Packit Service a2489d
	 }
Packit Service a2489d
	 c.add(sz, "");
Packit Service a2489d
      }
Packit Service a2489d
Packit Service a2489d
      /* We use unprec dates; doing MDTMs for each file in ls is far too
Packit Service a2489d
       * slow.  If someone actually wants that (to get dates on servers with
Packit Service a2489d
       * unparsable dates, or more accurate dates), it wouldn't be
Packit Service a2489d
       * difficult.  If we did this, we could also support --full-time. */
Packit Service a2489d
      if((mode & DATE) && (have & f->DATE)) {
Packit Service a2489d
	 /* Consider a time to be recent if it is within the past six
Packit Service a2489d
	  * months.  A Gregorian year has 365.2425 * 24 * 60 * 60 ==
Packit Service a2489d
	  * 31556952 seconds on the average.  Write this value as an
Packit Service a2489d
	  * integer constant to avoid floating point hassles.  */
Packit Service a2489d
	 const int six_months_ago = SMTask::now.UnixTime() - 31556952 / 2;
Packit Service a2489d
	 bool recent = six_months_ago <= f->date;
Packit Service a2489d
Packit Service a2489d
	 const char *use_fmt=time_fmt;
Packit Service a2489d
	 if(!use_fmt)
Packit Service a2489d
	    use_fmt=ResMgr::Query("cmd:time-style",0);
Packit Service a2489d
	 if(!use_fmt || !*use_fmt)
Packit Service a2489d
	    use_fmt="%b %e  %Y\n%b %e %H:%M";
Packit Service a2489d
Packit Service a2489d
	 xstring_ca dt_mem(xstrftime(use_fmt, localtime (&f->date.ts)));
Packit Service a2489d
	 char *dt=strtok(dt_mem.get_non_const(),"\n|");
Packit Service a2489d
	 if(recent) {
Packit Service a2489d
	    char *dt1=strtok(NULL,"\n|");
Packit Service a2489d
	    if(dt1)
Packit Service a2489d
	       dt=dt1;
Packit Service a2489d
	 }
Packit Service a2489d
	 if (!(f->defined&f->DATE)) {
Packit Service a2489d
	    /* put an empty field; make sure it's the same width */
Packit Service a2489d
	    int wid = mbswidth(dt, 0);
Packit Service a2489d
	    dt = string_alloca(wid+1);
Packit Service a2489d
	    memset(dt, ' ', wid);
Packit Service a2489d
	    dt[wid] = 0;
Packit Service a2489d
	 }
Packit Service a2489d
	 c.addf("%s ", "", dt);
Packit Service a2489d
      }
Packit Service a2489d
Packit Service a2489d
      const char *nm = f->name;
Packit Service a2489d
      if(basenames) nm = basename_ptr(nm);
Packit Service a2489d
      c.add(nm, col.GetColor(f));
Packit Service a2489d
Packit Service a2489d
      if(classify)
Packit Service a2489d
	 c.add(FileInfoSuffix(*f), suffix_color);
Packit Service a2489d
Packit Service a2489d
      if((mode & LINKS) &&
Packit Service a2489d
	 f->filetype == FileInfo::SYMLINK &&
Packit Service a2489d
	 f->symlink) {
Packit Service a2489d
	 c.add(" -> ", "");
Packit Service a2489d
Packit Service a2489d
	 /* see if we have a file entry for the symlink */
Packit Service a2489d
	 FileInfo tmpfi;
Packit Service a2489d
	 FileInfo *lfi = fs.FindByName(f->symlink);
Packit Service a2489d
Packit Service a2489d
	 if(!lfi) {
Packit Service a2489d
	    /* create a temporary one */
Packit Service a2489d
	    tmpfi.SetName(f->symlink);
Packit Service a2489d
	    lfi = &tmpfi;
Packit Service a2489d
	 }
Packit Service a2489d
Packit Service a2489d
	 c.add(lfi->name, col.GetColor(lfi));
Packit Service a2489d
	 if(classify)
Packit Service a2489d
	    c.add(FileInfoSuffix(*lfi), suffix_color);
Packit Service a2489d
      }
Packit Service a2489d
   }
Packit Service a2489d
Packit Service a2489d
   c.print(o, single_column? 0:width, color);
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
const char *FileSetOutput::FileInfoSuffix(const FileInfo &fi) const
Packit Service a2489d
{
Packit Service a2489d
   if(!(fi.defined&fi.TYPE))
Packit Service a2489d
      return "";
Packit Service a2489d
   if(fi.filetype == FileInfo::DIRECTORY)
Packit Service a2489d
      return "/";
Packit Service a2489d
   else if(fi.filetype == FileInfo::SYMLINK)
Packit Service a2489d
      return "@";
Packit Service a2489d
   return "";
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
void FileSetOutput::config(const OutputJob *o)
Packit Service a2489d
{
Packit Service a2489d
   width = o->GetWidth();
Packit Service a2489d
   if(width == -1)
Packit Service a2489d
      width = 80;
Packit Service a2489d
   color = ResMgr::QueryTriBool("color:use-color", 0, o->IsTTY());
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
void FileSetOutput::long_list()
Packit Service a2489d
{
Packit Service a2489d
   single_column = true;
Packit Service a2489d
   mode = ALL;
Packit Service a2489d
   /* -l's default size is 1; otherwise 1024 */
Packit Service a2489d
   if(!output_block_size)
Packit Service a2489d
      output_block_size = 1;
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
const char *FileSetOutput::parse_res(const char *res)
Packit Service a2489d
{
Packit Service a2489d
   Ref<ArgV> arg(new ArgV("",res));
Packit Service a2489d
   const char *error=parse_argv(arg);
Packit Service a2489d
   if(error)
Packit Service a2489d
      return error;
Packit Service a2489d
Packit Service a2489d
   /* shouldn't be any non-option arguments */
Packit Service a2489d
   if(arg->count() > 1)
Packit Service a2489d
      return _("non-option arguments found");
Packit Service a2489d
Packit Service a2489d
   return 0;
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
const char *FileSetOutput::ValidateArgv(xstring_c *s)
Packit Service a2489d
{
Packit Service a2489d
   if(!*s) return NULL;
Packit Service a2489d
Packit Service a2489d
   FileSetOutput tmp;
Packit Service a2489d
Packit Service a2489d
   const char *ret = tmp.parse_res(*s);
Packit Service a2489d
   if(ret) return ret;
Packit Service a2489d
Packit Service a2489d
   return NULL;
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
int FileSetOutput::Need() const
Packit Service a2489d
{
Packit Service a2489d
   int need=FileInfo::NAME;
Packit Service a2489d
   if(mode & PERMS)
Packit Service a2489d
      need|=FileInfo::MODE;
Packit Service a2489d
//   if(mode & SIZE) /* this must be optional */
Packit Service a2489d
//      need|=FileInfo::SIZE;
Packit Service a2489d
//   if(mode & DATE) /* this too */
Packit Service a2489d
//      need|=FileInfo::DATE;
Packit Service a2489d
   if(mode & LINKS)
Packit Service a2489d
      need|=FileInfo::SYMLINK_DEF;
Packit Service a2489d
   if(mode & USER)
Packit Service a2489d
      need|=FileInfo::USER;
Packit Service a2489d
   if(mode & GROUP)
Packit Service a2489d
      need|=FileInfo::GROUP;
Packit Service a2489d
   if(need_exact_time)
Packit Service a2489d
      need|=FileInfo::DATE;
Packit Service a2489d
Packit Service a2489d
   return need;
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
#undef super
Packit Service a2489d
#define super SessionJob
Packit Service a2489d
Packit Service a2489d
clsJob::clsJob(FA *s, ArgV *a, FileSetOutput *_opts, OutputJob *_output):
Packit Service a2489d
   SessionJob(s),
Packit Service a2489d
   fso(_opts),
Packit Service a2489d
   args(a),
Packit Service a2489d
   done(0),
Packit Service a2489d
   use_cache(true),
Packit Service a2489d
   error(false),
Packit Service a2489d
   state(INIT)
Packit Service a2489d
{
Packit Service a2489d
   list_info=0;
Packit Service a2489d
Packit Service a2489d
   if(args->count() == 1)
Packit Service a2489d
      args->Add("");
Packit Service a2489d
Packit Service a2489d
   output=_output;
Packit Service a2489d
   AddWaiting(output);
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
int clsJob::Done()
Packit Service a2489d
{
Packit Service a2489d
   return done && output->Done();
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
int clsJob::Do()
Packit Service a2489d
{
Packit Service a2489d
   int m=STALL;
Packit Service a2489d
Packit Service a2489d
   if(output->Done())
Packit Service a2489d
      state=DONE;
Packit Service a2489d
Packit Service a2489d
   switch(state)
Packit Service a2489d
   {
Packit Service a2489d
   case INIT:
Packit Service a2489d
      state=START_LISTING;
Packit Service a2489d
      m=MOVED;
Packit Service a2489d
Packit Service a2489d
   case START_LISTING:
Packit Service a2489d
   {
Packit Service a2489d
      list_info=0;
Packit Service a2489d
Packit Service a2489d
      /* next: */
Packit Service a2489d
      mask.set(0);
Packit Service a2489d
      dir.set(args->getnext());
Packit Service a2489d
      if(!dir) {
Packit Service a2489d
	 /* done */
Packit Service a2489d
	 state=DONE;
Packit Service a2489d
	 return MOVED;
Packit Service a2489d
      }
Packit Service a2489d
Packit Service a2489d
      /* If the basename contains wildcards, set up the mask. */
Packit Service a2489d
      const char *bn = basename_ptr(dir);
Packit Service a2489d
      if(Glob::HasWildcards(bn)) {
Packit Service a2489d
	 /* The mask is the whole argument, not just the basename; this is
Packit Service a2489d
	  * because the whole relative paths will end up in the FileSet, and
Packit Service a2489d
	  * that's what this pattern will be matched against. */
Packit Service a2489d
	 mask.set(dir);
Packit Service a2489d
	 // leave the final / on the path, to prevent the dirname of
Packit Service a2489d
	 // "file/*" from being treated as a file
Packit Service a2489d
	 dir.truncate(bn-dir); // this can result in dir eq ""
Packit Service a2489d
      } else {
Packit Service a2489d
	 // no need to glob, just unquote metacharacters.
Packit Service a2489d
	 Glob::UnquoteWildcards(const_cast<char*>(bn));
Packit Service a2489d
      }
Packit Service a2489d
Packit Service a2489d
      list_info=new GetFileInfo(session, dir, fso->list_directories);
Packit Service a2489d
      list_info->UseCache(use_cache);
Packit Service a2489d
      list_info->Need(fso->Need());
Packit Service a2489d
Packit Service a2489d
      state=GETTING_LIST_INFO;
Packit Service a2489d
      m=MOVED;
Packit Service a2489d
   }
Packit Service a2489d
   case GETTING_LIST_INFO:
Packit Service a2489d
   {
Packit Service a2489d
      if(!list_info->Done())
Packit Service a2489d
	 return m;
Packit Service a2489d
Packit Service a2489d
      if(list_info->Error()) {
Packit Service a2489d
	 eprintf("%s\n", list_info->ErrorText());
Packit Service a2489d
	 error=true;
Packit Service a2489d
	 state=START_LISTING;
Packit Service a2489d
	 return MOVED;
Packit Service a2489d
      }
Packit Service a2489d
Packit Service a2489d
      /* one just finished */
Packit Service a2489d
      fso->pat.move_here(mask);
Packit Service a2489d
      FileSet *res = list_info->GetResult();
Packit Service a2489d
Packit Service a2489d
      if(res)
Packit Service a2489d
	 fso->print(*res, output);
Packit Service a2489d
Packit Service a2489d
      fso->pat.set(0);
Packit Service a2489d
      delete res;
Packit Service a2489d
Packit Service a2489d
      state=START_LISTING;
Packit Service a2489d
      return MOVED;
Packit Service a2489d
   }
Packit Service a2489d
Packit Service a2489d
   case DONE:
Packit Service a2489d
      if(!done)
Packit Service a2489d
      {
Packit Service a2489d
	 output->PutEOF();
Packit Service a2489d
	 done=true;
Packit Service a2489d
	 m=MOVED;
Packit Service a2489d
      }
Packit Service a2489d
      break;
Packit Service a2489d
   }
Packit Service a2489d
   return m;
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
void clsJob::SuspendInternal()
Packit Service a2489d
{
Packit Service a2489d
   super::SuspendInternal();
Packit Service a2489d
   if(list_info)
Packit Service a2489d
      list_info->SuspendSlave();
Packit Service a2489d
   session->SuspendSlave();
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
void clsJob::ResumeInternal()
Packit Service a2489d
{
Packit Service a2489d
   if(list_info)
Packit Service a2489d
      list_info->ResumeSlave();
Packit Service a2489d
   session->ResumeSlave();
Packit Service a2489d
   super::ResumeInternal();
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
void clsJob::ShowRunStatus(const SMTaskRef<StatusLine>& s)
Packit Service a2489d
{
Packit Service a2489d
   if(fso->quiet)
Packit Service a2489d
      return;
Packit Service a2489d
Packit Service a2489d
   if(!output->ShowStatusLine(s))
Packit Service a2489d
      return;
Packit Service a2489d
Packit Service a2489d
   if(list_info && !list_info->Done())
Packit Service a2489d
   {
Packit Service a2489d
      const char *curr = args->getcurr();
Packit Service a2489d
      if(!*curr)
Packit Service a2489d
	 curr = ".";
Packit Service a2489d
      const char *stat = list_info->Status();
Packit Service a2489d
      if(*stat)
Packit Service a2489d
	 s->Show("`%s' %s %s", curr, stat, output->Status(s));
Packit Service a2489d
   }
Packit Service a2489d
   else
Packit Service a2489d
	 s->Show("%s", output->Status(s));
Packit Service a2489d
}
Packit Service a2489d
Packit Service a2489d
xstring& clsJob::FormatStatus(xstring& s,int v,const char *prefix)
Packit Service a2489d
{
Packit Service a2489d
   Job::FormatStatus(s,v,prefix);
Packit Service a2489d
Packit Service a2489d
   if(list_info)
Packit Service a2489d
   {
Packit Service a2489d
      const char *curr = args->getcurr();
Packit Service a2489d
      if(!*curr)
Packit Service a2489d
	 curr = ".";
Packit Service a2489d
      const char *stat = list_info->Status();
Packit Service a2489d
      if(*stat)
Packit Service a2489d
	 s.appendf("%s`%s' %s\n", prefix, curr, stat);
Packit Service a2489d
   }
Packit Service a2489d
   return s;
Packit Service a2489d
}