1 /* $OpenBSD: sftp-glob.c,v 1.33 2023/09/10 23:12:32 djm Exp $ */
3 * Copyright (c) 2001-2004 Damien Miller <djm@openbsd.org>
5 * Permission to use, copy, modify, and distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
9 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
20 #include <sys/types.h>
21 #ifdef HAVE_SYS_STAT_H
22 # include <sys/stat.h>
32 #include "sftp-common.h"
33 #include "sftp-client.h"
35 int sftp_glob(struct sftp_conn
*, const char *, int,
36 int (*)(const char *, int), glob_t
*);
44 struct sftp_conn
*conn
;
48 fudge_opendir(const char *path
)
50 struct SFTP_OPENDIR
*r
;
52 r
= xcalloc(1, sizeof(*r
));
54 if (sftp_readdir(cur
.conn
, path
, &r
->dir
)) {
64 static struct dirent
*
65 fudge_readdir(struct SFTP_OPENDIR
*od
)
67 /* Solaris needs sizeof(dirent) + path length (see below) */
68 static char buf
[sizeof(struct dirent
) + MAXPATHLEN
];
69 struct dirent
*ret
= (struct dirent
*)buf
;
70 #ifdef __GNU_LIBRARY__
72 #endif /* __GNU_LIBRARY__ */
74 if (od
->dir
[od
->offset
] == NULL
)
77 memset(buf
, 0, sizeof(buf
));
80 * Solaris defines dirent->d_name as a one byte array and expects
81 * you to hack around it.
83 #ifdef BROKEN_ONE_BYTE_DIRENT_D_NAME
84 strlcpy(ret
->d_name
, od
->dir
[od
->offset
++]->filename
, MAXPATHLEN
);
86 strlcpy(ret
->d_name
, od
->dir
[od
->offset
++]->filename
,
89 #ifdef __GNU_LIBRARY__
91 * Idiot glibc uses extensions to struct dirent for readdir with
92 * ALTDIRFUNCs. Not that this is documented anywhere but the
93 * source... Fake an inode number to appease it.
98 #endif /* __GNU_LIBRARY__ */
104 fudge_closedir(struct SFTP_OPENDIR
*od
)
106 sftp_free_dirents(od
->dir
);
111 fudge_lstat(const char *path
, struct stat
*st
)
115 if (sftp_lstat(cur
.conn
, path
, 1, &a
) != 0)
118 attrib_to_stat(&a
, st
);
124 fudge_stat(const char *path
, struct stat
*st
)
128 if (sftp_stat(cur
.conn
, path
, 1, &a
) != 0)
131 attrib_to_stat(&a
, st
);
137 sftp_glob(struct sftp_conn
*conn
, const char *pattern
, int flags
,
138 int (*errfunc
)(const char *, int), glob_t
*pglob
)
145 pglob
->gl_opendir
= fudge_opendir
;
146 pglob
->gl_readdir
= (struct dirent
*(*)(void *))fudge_readdir
;
147 pglob
->gl_closedir
= (void (*)(void *))fudge_closedir
;
148 pglob
->gl_lstat
= fudge_lstat
;
149 pglob
->gl_stat
= fudge_stat
;
151 memset(&cur
, 0, sizeof(cur
));
154 if ((r
= glob(pattern
, flags
| GLOB_ALTDIRFUNC
, errfunc
, pglob
)) != 0)
157 * When both GLOB_NOCHECK and GLOB_MARK are active, a single gl_pathv
158 * entry has been returned and that entry has not already been marked,
159 * then check whether it needs a '/' appended as a directory mark.
161 * This ensures that a NOCHECK result is annotated as a directory.
162 * The glob(3) spec doesn't promise to mark NOCHECK entries, but doing
163 * it simplifies our callers (sftp/scp) considerably.
165 * XXX doesn't try to handle gl_offs.
167 if ((flags
& (GLOB_NOCHECK
|GLOB_MARK
)) == (GLOB_NOCHECK
|GLOB_MARK
) &&
168 pglob
->gl_matchc
== 0 && pglob
->gl_offs
== 0 &&
169 pglob
->gl_pathc
== 1 && (s
= pglob
->gl_pathv
[0]) != NULL
&&
170 (l
= strlen(s
)) > 0 && s
[l
-1] != '/') {
171 if (fudge_stat(s
, &sb
) == 0 && S_ISDIR(sb
.st_mode
)) {
172 /* NOCHECK on a directory; annotate */
173 if ((s
= realloc(s
, l
+ 2)) != NULL
) {
174 memcpy(s
+ l
, "/", 2);
175 pglob
->gl_pathv
[0] = s
;