2 /* $OpenBSD: sftp-common.c,v 1.20 2006/08/03 03:34:42 deraadt Exp $ */
4 * Copyright (c) 2001 Markus Friedl. All rights reserved.
5 * Copyright (c) 2001 Damien Miller. All rights reserved.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 __RCSID("$NetBSD: sftp-common.c,v 1.12 2006/09/28 21:22:15 christos Exp $");
30 #include <sys/types.h>
32 #include <sys/param.h>
47 #include "sftp-common.h"
49 /* Clear contents of attributes structure */
51 attrib_clear(Attrib
*a
)
62 /* Convert from struct stat to filexfer attribs */
64 stat_to_attrib(const struct stat
*st
, Attrib
*a
)
68 a
->flags
|= SSH2_FILEXFER_ATTR_SIZE
;
69 a
->size
= st
->st_size
;
70 a
->flags
|= SSH2_FILEXFER_ATTR_UIDGID
;
73 a
->flags
|= SSH2_FILEXFER_ATTR_PERMISSIONS
;
74 a
->perm
= st
->st_mode
;
75 a
->flags
|= SSH2_FILEXFER_ATTR_ACMODTIME
;
76 a
->atime
= st
->st_atime
;
77 a
->mtime
= st
->st_mtime
;
80 /* Convert from filexfer attribs to struct stat */
82 attrib_to_stat(const Attrib
*a
, struct stat
*st
)
84 memset(st
, 0, sizeof(*st
));
86 if (a
->flags
& SSH2_FILEXFER_ATTR_SIZE
)
87 st
->st_size
= a
->size
;
88 if (a
->flags
& SSH2_FILEXFER_ATTR_UIDGID
) {
92 if (a
->flags
& SSH2_FILEXFER_ATTR_PERMISSIONS
)
93 st
->st_mode
= a
->perm
;
94 if (a
->flags
& SSH2_FILEXFER_ATTR_ACMODTIME
) {
95 st
->st_atime
= a
->atime
;
96 st
->st_mtime
= a
->mtime
;
100 /* Decode attributes in buffer */
102 decode_attrib(Buffer
*b
)
107 a
.flags
= buffer_get_int(b
);
108 if (a
.flags
& SSH2_FILEXFER_ATTR_SIZE
)
109 a
.size
= buffer_get_int64(b
);
110 if (a
.flags
& SSH2_FILEXFER_ATTR_UIDGID
) {
111 a
.uid
= buffer_get_int(b
);
112 a
.gid
= buffer_get_int(b
);
114 if (a
.flags
& SSH2_FILEXFER_ATTR_PERMISSIONS
)
115 a
.perm
= buffer_get_int(b
);
116 if (a
.flags
& SSH2_FILEXFER_ATTR_ACMODTIME
) {
117 a
.atime
= buffer_get_int(b
);
118 a
.mtime
= buffer_get_int(b
);
120 /* vendor-specific extensions */
121 if (a
.flags
& SSH2_FILEXFER_ATTR_EXTENDED
) {
125 count
= buffer_get_int(b
);
126 for (i
= 0; i
< count
; i
++) {
127 type
= buffer_get_string(b
, NULL
);
128 data
= buffer_get_string(b
, NULL
);
129 debug3("Got file attribute \"%s\"", type
);
137 /* Encode attributes to buffer */
139 encode_attrib(Buffer
*b
, const Attrib
*a
)
141 buffer_put_int(b
, a
->flags
);
142 if (a
->flags
& SSH2_FILEXFER_ATTR_SIZE
)
143 buffer_put_int64(b
, a
->size
);
144 if (a
->flags
& SSH2_FILEXFER_ATTR_UIDGID
) {
145 buffer_put_int(b
, a
->uid
);
146 buffer_put_int(b
, a
->gid
);
148 if (a
->flags
& SSH2_FILEXFER_ATTR_PERMISSIONS
)
149 buffer_put_int(b
, a
->perm
);
150 if (a
->flags
& SSH2_FILEXFER_ATTR_ACMODTIME
) {
151 buffer_put_int(b
, a
->atime
);
152 buffer_put_int(b
, a
->mtime
);
156 /* Convert from SSH2_FX_ status to text error message */
164 return("End of file");
165 case SSH2_FX_NO_SUCH_FILE
:
166 return("No such file or directory");
167 case SSH2_FX_PERMISSION_DENIED
:
168 return("Permission denied");
169 case SSH2_FX_FAILURE
:
171 case SSH2_FX_BAD_MESSAGE
:
172 return("Bad message");
173 case SSH2_FX_NO_CONNECTION
:
174 return("No connection");
175 case SSH2_FX_CONNECTION_LOST
:
176 return("Connection lost");
177 case SSH2_FX_OP_UNSUPPORTED
:
178 return("Operation unsupported");
180 return("Unknown status");
186 * drwxr-xr-x 5 markus markus 1024 Jan 13 18:39 .ssh
189 ls_file(const char *name
, const struct stat
*st
, int remote
)
191 int ulen
, glen
, sz
= 0;
194 struct tm
*ltime
= localtime(&st
->st_mtime
);
196 char buf
[1024], mode
[11+1], tbuf
[12+1], ubuf
[11+1], gbuf
[11+1];
198 strmode(st
->st_mode
, mode
);
199 if (!remote
&& (pw
= getpwuid(st
->st_uid
)) != NULL
) {
202 snprintf(ubuf
, sizeof ubuf
, "%u", (u_int
)st
->st_uid
);
205 if (!remote
&& (gr
= getgrgid(st
->st_gid
)) != NULL
) {
208 snprintf(gbuf
, sizeof gbuf
, "%u", (u_int
)st
->st_gid
);
212 if (time(NULL
) - st
->st_mtime
< (365*24*60*60)/2)
213 sz
= strftime(tbuf
, sizeof tbuf
, "%b %e %H:%M", ltime
);
215 sz
= strftime(tbuf
, sizeof tbuf
, "%b %e %Y", ltime
);
219 ulen
= MAX(strlen(user
), 8);
220 glen
= MAX(strlen(group
), 8);
221 snprintf(buf
, sizeof buf
, "%s %3u %-*s %-*s %8llu %s %s", mode
,
222 (u_int
)st
->st_nlink
, ulen
, user
, glen
, group
,
223 (unsigned long long)st
->st_size
, tbuf
, name
);