python: Add missing word to comment
[Samba.git] / source3 / libsmb / libsmb_path.c
blob615de1c398bc482942849846af96ac0f76f98210
1 /*
2 Unix SMB/Netbios implementation.
3 SMB client library implementation
4 Copyright (C) Andrew Tridgell 1998
5 Copyright (C) Richard Sharpe 2000, 2002
6 Copyright (C) John Terpstra 2000
7 Copyright (C) Tom Jansen (Ninja ISD) 2002
8 Copyright (C) Derrell Lipman 2003-2008
9 Copyright (C) Jeremy Allison 2007, 2008
11 This program is free software; you can redistribute it and/or modify
12 it under the terms of the GNU General Public License as published by
13 the Free Software Foundation; either version 3 of the License, or
14 (at your option) any later version.
16 This program is distributed in the hope that it will be useful,
17 but WITHOUT ANY WARRANTY; without even the implied warranty of
18 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 GNU General Public License for more details.
21 You should have received a copy of the GNU General Public License
22 along with this program. If not, see <http://www.gnu.org/licenses/>.
25 #include "includes.h"
26 #include "libsmbclient.h"
27 #include "libsmb_internal.h"
31 * smbc_urldecode()
32 * and urldecode_talloc() (internal fn.)
34 * Convert strings of %xx to their single character equivalent. Each 'x' must
35 * be a valid hexadecimal digit, or that % sequence is left undecoded.
37 * dest may, but need not be, the same pointer as src.
39 * Returns the number of % sequences which could not be converted due to lack
40 * of two following hexadecimal digits.
42 static int
43 urldecode_talloc(TALLOC_CTX *ctx, char **pp_dest, const char *src)
45 int old_length = strlen(src);
46 int i = 0;
47 int err_count = 0;
48 size_t newlen = 1;
49 char *p, *dest;
51 if (old_length == 0) {
52 return 0;
55 *pp_dest = NULL;
56 for (i = 0; i < old_length; ) {
57 unsigned char character = src[i++];
59 if (character == '%') {
60 uint8_t v;
61 bool ok = hex_byte(&src[i], &v);
62 if (ok) {
63 character = v;
64 if (character == '\0') {
65 break; /* Stop at %00 */
67 i += 2;
68 } else {
69 err_count++;
72 newlen++;
75 dest = talloc_array(ctx, char, newlen);
76 if (!dest) {
77 return err_count;
80 err_count = 0;
81 for (p = dest, i = 0; i < old_length; ) {
82 unsigned char character = src[i++];
84 if (character == '%') {
85 uint8_t v;
86 bool ok = hex_byte(&src[i], &v);
87 if (ok) {
88 character = v;
89 if (character == '\0') {
90 break; /* Stop at %00 */
92 i += 2;
93 } else {
94 err_count++;
97 *p++ = character;
100 *p = '\0';
101 *pp_dest = dest;
102 return err_count;
106 smbc_urldecode(char *dest,
107 char *src,
108 size_t max_dest_len)
110 TALLOC_CTX *frame = talloc_stackframe();
111 char *pdest;
112 int ret = urldecode_talloc(frame, &pdest, src);
114 if (pdest) {
115 strlcpy(dest, pdest, max_dest_len);
117 TALLOC_FREE(frame);
118 return ret;
122 * smbc_urlencode()
124 * Convert any characters not specifically allowed in a URL into their %xx
125 * equivalent.
127 * Returns the remaining buffer length.
130 smbc_urlencode(char *dest,
131 char *src,
132 int max_dest_len)
134 char hex[] = "0123456789ABCDEF";
136 for (; *src != '\0' && max_dest_len >= 3; src++) {
138 if ((*src < '0' &&
139 *src != '-' &&
140 *src != '.') ||
141 (*src > '9' &&
142 *src < 'A') ||
143 (*src > 'Z' &&
144 *src < 'a' &&
145 *src != '_') ||
146 (*src > 'z')) {
147 *dest++ = '%';
148 *dest++ = hex[(*src >> 4) & 0x0f];
149 *dest++ = hex[*src & 0x0f];
150 max_dest_len -= 3;
151 } else {
152 *dest++ = *src;
153 max_dest_len--;
157 if (max_dest_len <= 0) {
158 /* Ensure we return -1 if no null termination. */
159 return -1;
162 *dest++ = '\0';
163 max_dest_len--;
165 return max_dest_len;
169 * Function to parse a path and turn it into components
171 * The general format of an SMB URI is explain in Christopher Hertel's CIFS
172 * book, at http://ubiqx.org/cifs/Appendix-D.html. We accept a subset of the
173 * general format ("smb:" only; we do not look for "cifs:").
176 * We accept:
177 * smb://[[[domain;]user[:password]@]server[:port][/share[/path[/file]]]]
178 * [?options]
180 * Meaning of URLs:
182 * smb:// Show all workgroups.
184 * The method of locating the list of workgroups varies
185 * depending upon the setting of the context variable
186 * context->options.browse_max_lmb_count. This value
187 * determines the maximum number of local master browsers to
188 * query for the list of workgroups. In order to ensure that
189 * a complete list of workgroups is obtained, all master
190 * browsers must be queried, but if there are many
191 * workgroups, the time spent querying can begin to add up.
192 * For small networks (not many workgroups), it is suggested
193 * that this variable be set to 0, indicating query all local
194 * master browsers. When the network has many workgroups, a
195 * reasonable setting for this variable might be around 3.
197 * smb://name/ if name<1D> or name<1B> exists, list servers in
198 * workgroup, else, if name<20> exists, list all shares
199 * for server ...
201 * If "options" are provided, this function returns the entire option list as a
202 * string, for later parsing by the caller. Note that currently, no options
203 * are supported.
206 #define SMBC_PREFIX "smb:"
209 SMBC_parse_path(TALLOC_CTX *ctx,
210 SMBCCTX *context,
211 const char *fname,
212 char **pp_workgroup,
213 char **pp_server,
214 uint16_t *p_port,
215 char **pp_share,
216 char **pp_path,
217 char **pp_user,
218 char **pp_password,
219 char **pp_options)
221 char *s;
222 const char *p;
223 char *q, *r;
224 char *workgroup = NULL;
225 int len;
227 /* Ensure these returns are at least valid pointers. */
228 *pp_server = talloc_strdup(ctx, "");
229 *p_port = smbc_getPort(context);
230 *pp_share = talloc_strdup(ctx, "");
231 *pp_path = talloc_strdup(ctx, "");
232 *pp_user = talloc_strdup(ctx, "");
233 *pp_password = talloc_strdup(ctx, "");
235 if (!*pp_server || !*pp_share || !*pp_path ||
236 !*pp_user || !*pp_password) {
237 return -1;
241 * Assume we won't find an authentication domain to parse, so default
242 * to the workgroup in the provided context.
244 if (pp_workgroup != NULL) {
245 *pp_workgroup =
246 talloc_strdup(ctx, smbc_getWorkgroup(context));
249 if (pp_options) {
250 *pp_options = talloc_strdup(ctx, "");
252 s = talloc_strdup(ctx, fname);
254 /* see if it has the right prefix */
255 len = strlen(SMBC_PREFIX);
256 if (strncmp(s,SMBC_PREFIX,len) || (s[len] != '/' && s[len] != 0)) {
257 return -1; /* What about no smb: ? */
260 p = s + len;
262 /* Watch the test below, we are testing to see if we should exit */
264 if (strncmp(p, "//", 2) && strncmp(p, "\\\\", 2)) {
265 DEBUG(1, ("Invalid path (does not begin with smb://)\n"));
266 return -1;
269 p += 2; /* Skip the double slash */
271 /* See if any options were specified */
272 if ((q = strrchr(p, '?')) != NULL ) {
273 /* There are options. Null terminate here and point to them */
274 *q++ = '\0';
276 DEBUG(4, ("Found options '%s'\n", q));
278 /* Copy the options */
279 if (pp_options && *pp_options != NULL) {
280 TALLOC_FREE(*pp_options);
281 *pp_options = talloc_strdup(ctx, q);
285 if (*p == '\0') {
286 goto decoding;
289 if (*p == '/') {
290 *pp_server = talloc_strndup(
291 ctx, smbc_getWorkgroup(context), 16);
292 if (!*pp_server) {
293 return -1;
295 return 0;
299 * ok, its for us. Now parse out the server, share etc.
301 * However, we want to parse out [[domain;]user[:password]@] if it
302 * exists ...
305 /* check that '@' occurs before '/', if '/' exists at all */
306 q = strchr_m(p, '@');
307 r = strchr_m(p, '/');
308 if (q && (!r || q < r)) {
309 char *userinfo = NULL;
310 const char *u;
312 next_token_no_ltrim_talloc(ctx, &p, &userinfo, "@");
313 if (!userinfo) {
314 return -1;
316 u = userinfo;
318 if (strchr_m(u, ';')) {
319 next_token_no_ltrim_talloc(ctx, &u, &workgroup, ";");
320 if (!workgroup) {
321 return -1;
323 if (pp_workgroup) {
324 *pp_workgroup = workgroup;
328 if (strchr_m(u, ':')) {
329 next_token_no_ltrim_talloc(ctx, &u, pp_user, ":");
330 if (!*pp_user) {
331 return -1;
333 *pp_password = talloc_strdup(ctx, u);
334 if (!*pp_password) {
335 return -1;
337 } else {
338 *pp_user = talloc_strdup(ctx, u);
339 if (!*pp_user) {
340 return -1;
345 if (!next_token_talloc(ctx, &p, pp_server, "/")) {
346 return -1;
350 * Does *pp_server contain a ':' ? If so
351 * this denotes the port.
353 q = strchr_m(*pp_server, ':');
354 if (q != NULL) {
355 long int port;
356 char *endptr = NULL;
357 *q = '\0';
358 q++;
359 if (*q == '\0') {
360 /* Bad port. */
361 return -1;
363 port = strtol(q, &endptr, 10);
364 if (*endptr != '\0') {
365 /* Bad port. */
366 return -1;
368 *p_port = (uint16_t)port;
371 if (*p == (char)0) {
372 goto decoding; /* That's it ... */
375 if (!next_token_talloc(ctx, &p, pp_share, "/")) {
376 return -1;
380 * Prepend a leading slash if there's a file path, as required by
381 * NetApp filers.
383 if (*p != '\0') {
384 *pp_path = talloc_asprintf(ctx,
385 "\\%s",
387 } else {
388 *pp_path = talloc_strdup(ctx, "");
390 if (!*pp_path) {
391 return -1;
393 string_replace(*pp_path, '/', '\\');
395 decoding:
396 (void) urldecode_talloc(ctx, pp_path, *pp_path);
397 (void) urldecode_talloc(ctx, pp_server, *pp_server);
398 (void) urldecode_talloc(ctx, pp_share, *pp_share);
399 (void) urldecode_talloc(ctx, pp_user, *pp_user);
400 (void) urldecode_talloc(ctx, pp_password, *pp_password);
402 if (!workgroup) {
403 workgroup = talloc_strdup(ctx, smbc_getWorkgroup(context));
405 if (!workgroup) {
406 return -1;
409 /* set the credentials to make DFS work */
410 smbc_set_credentials_with_fallback(context,
411 workgroup,
412 *pp_user,
413 *pp_password);
414 return 0;