XRootD
XrdCpFile.cc
Go to the documentation of this file.
1 /******************************************************************************/
2 /* */
3 /* X r d C p F i l e . c c */
4 /* */
5 /* (c) 2012 by the Board of Trustees of the Leland Stanford, Jr., University */
6 /* All Rights Reserved */
7 /* Produced by Andrew Hanushevsky for Stanford University under contract */
8 /* DE-AC02-76-SFO0515 with the Department of Energy */
9 /* */
10 /* This file is part of the XRootD software suite. */
11 /* */
12 /* XRootD is free software: you can redistribute it and/or modify it under */
13 /* the terms of the GNU Lesser General Public License as published by the */
14 /* Free Software Foundation, either version 3 of the License, or (at your */
15 /* option) any later version. */
16 /* */
17 /* XRootD is distributed in the hope that it will be useful, but WITHOUT */
18 /* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or */
19 /* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public */
20 /* License for more details. */
21 /* */
22 /* You should have received a copy of the GNU Lesser General Public License */
23 /* along with XRootD in a file called COPYING.LESSER (LGPL license) and file */
24 /* COPYING (GPL license). If not, see <http://www.gnu.org/licenses/>. */
25 /* */
26 /* The copyright holder's institutional names and contributor's names may not */
27 /* be used to endorse or promote products derived from this software without */
28 /* specific prior written permission of the institution or contributor. */
29 /******************************************************************************/
30 
31 #include <cerrno>
32 #include <fcntl.h>
33 #include <cstring>
34 #include <strings.h>
35 #include <sys/types.h>
36 #include <sys/stat.h>
37 
38 #include "XrdApps/XrdCpFile.hh"
39 #include "XrdOuc/XrdOucNSWalk.hh"
40 
41 /******************************************************************************/
42 /* S t a t i c M e m b e r s */
43 /******************************************************************************/
44 
45 const char *XrdCpFile::mPfx = 0;
46 
47 /******************************************************************************/
48 /* C o n s t r u c t o r */
49 /******************************************************************************/
50 
51 XrdCpFile::XrdCpFile(const char *FSpec, int &badURL)
52 {
53  static struct proto {const char *pHdr; int pHsz; PType pVal;}
54  pTab[] = {{"xroot://", 8, isXroot},
55  {"xroots://", 9, isXroots},
56  {"root://", 7, isXroot},
57  {"roots://", 8, isXroots},
58  {"http://", 7, isHttp},
59  {"pelican://", 10, isPelican},
60  {"s3://", 5, isS3},
61  {"https://", 8, isHttps}
62  };
63  static int pTnum = sizeof(pTab)/sizeof(struct proto);
64  const char *Slash;
65  int i;
66 
67 // Do some common initialization
68 //
69  Doff = 0;
70  Dlen = 0;
71  Next = 0;
72  fSize = 0;
73  badURL= 0;
74  memset(ProtName, 0, sizeof(ProtName));
75 
76 // Copy out the path and remove trailing slashes (except the last one)
77 //
78  Path = strdup(FSpec);
79  i = strlen(Path);
80  while(i) if (Path[i-1] != '/' || (i > 1 && Path[i-2] != '/')) break;
81  else Path[--i] = 0;
82 
83 // Check for stdin stdout spec
84 //
85  if (!strcmp(Path, "-"))
86  {Protocol = isStdIO;
87  return;
88  }
89 
90 // Dtermine protocol of the incoming spec
91 //
92  for (i = 0; i < pTnum; i++)
93  {if (!strncmp(FSpec, pTab[i].pHdr, pTab[i].pHsz))
94  {Protocol = pTab[i].pVal;
95  memcpy(ProtName, pTab[i].pHdr, pTab[i].pHsz-3);
96  return;
97  }
98  }
99 
100 // See if this is a file
101 //
102  Protocol = isFile;
103  if (!strncmp(Path, "file://", 7))
104  {char *pP = Path + 7;
105  if (!strncmp(pP, "localhost", 9)) memmove( Path, pP + 9, strlen( pP + 9 ) + 1 );
106  else if (*pP == '/') memmove( Path, pP, strlen( pP ) + 1 );
107  else {Protocol = isOther;
108  strcpy(ProtName, "remote");
109  return;
110  }
111  }
112 
113 // Set the default Doff and Dlen assuming non-recursive copy
114 //
115  if ((Slash = rindex(Path, '/'))) Dlen = Doff = Slash - Path + 1;
116 }
117 
118 /******************************************************************************/
119 
120 XrdCpFile::XrdCpFile(char *FSpec, struct stat &Stat, short doff, short dlen)
121  : Next(0), Path(FSpec), Doff(doff), Dlen(dlen),
122  Protocol(isFile), fSize(Stat.st_size)
123  {strcpy(ProtName, "file");}
124 
125 /******************************************************************************/
126 /* E x t e n d */
127 /******************************************************************************/
128 
129 int XrdCpFile::Extend(XrdCpFile **pLast, int &nFile, long long &nBytes)
130 {
132  XrdOucNSWalk::NSEnt *nP, *nnP;
133  XrdCpFile *fP, *pP = this;
134  int rc;
135  short dlen, doff = strlen(Path);
136 
137  nsObj.setMsgOn(mPfx);
138 
139  while((nP = nsObj.Index(rc)) && rc == 0)
140  {do {dlen = nP->Plen - doff;
141  fP = new XrdCpFile(nP->Path, nP->Stat, doff, dlen);
142  nFile++; nBytes += nP->Stat.st_size; nP->Path = 0;
143  pP->Next = fP; pP = fP;
144  nnP = nP->Next; delete nP;
145  } while((nP = nnP));
146  }
147 
148  if (pLast) *pLast = pP;
149  return rc;
150 }
151 
152 /******************************************************************************/
153 /* R e s o l v e */
154 /******************************************************************************/
155 
157 {
158  struct stat Stat;
159 
160 // Ignore this call if this is not a file
161 //
162  if (Protocol != isFile) return 0;
163 
164 // This should exist but it might not, the caller will determine what to do
165 //
166  char *cgibeg = strchr( Path, '?' );
167  if( cgibeg ) *cgibeg = '\0';
168  if (stat(Path, &Stat)) return errno;
169  if( cgibeg ) *cgibeg = '?';
170 
171 // Find out what this really is
172 //
173  if (S_ISREG(Stat.st_mode)) fSize = Stat.st_size;
174  else if (S_ISDIR(Stat.st_mode)) Protocol = isDir;
175  else if (!strcmp(Path, "/dev/null")) Protocol = isDevNull;
176  else if (!strcmp(Path, "/dev/zero")) Protocol = isDevZero;
177  else return ENOTSUP;
178 
179 // All is well
180 //
181  return 0;
182 }
struct stat Stat
Definition: XrdCks.cc:49
#define stat(a, b)
Definition: XrdPosix.hh:101
XrdOucString Path
short Doff
Definition: XrdCpFile.hh:46
PType Protocol
Definition: XrdCpFile.hh:49
int Resolve()
Definition: XrdCpFile.cc:156
int Extend(XrdCpFile **pLast, int &nFile, long long &nBytes)
Definition: XrdCpFile.cc:129
long long fSize
Definition: XrdCpFile.hh:51
char * Path
Definition: XrdCpFile.hh:45
char ProtName[8]
Definition: XrdCpFile.hh:50
XrdCpFile * Next
Definition: XrdCpFile.hh:44
short Dlen
Definition: XrdCpFile.hh:47
static const int retFile
void setMsgOn(const char *pfx)
NSEnt * Index(int &rc, const char **dPath=0)
Definition: XrdOucNSWalk.cc:93
static const int Recurse
ProtocolImpl< false > Protocol
struct NSEnt * Next
Definition: XrdOucNSWalk.hh:48