2 * Alternate testbrowse utility provided by Mikhail Kshevetskiy.
3 * This version tests use of multiple contexts.
10 #include <libsmbclient.h>
13 const char *workgroup
= "NT";
14 const char *username
= "guest";
15 const char *password
= "";
17 typedef struct smbitem smbitem
;
18 typedef int(*qsort_cmp
)(const void *, const void *);
26 static void smbc_auth_fn(
29 char *wrkgrp
, int wrkgrplen
,
30 char *user
, int userlen
,
31 char *passwd
, int passwdlen
){
38 strncpy(wrkgrp
, workgroup
, wrkgrplen
- 1); wrkgrp
[wrkgrplen
- 1] = 0;
39 strncpy(user
, username
, userlen
- 1); user
[userlen
- 1] = 0;
40 strncpy(passwd
, password
, passwdlen
- 1); passwd
[passwdlen
- 1] = 0;
43 static SMBCCTX
* create_smbctx(void){
46 if ((ctx
= smbc_new_context()) == NULL
) return NULL
;
48 smbc_setDebug(ctx
, debuglevel
);
49 smbc_setFunctionAuthData(ctx
, smbc_auth_fn
);
51 if (smbc_init_context(ctx
) == NULL
){
52 smbc_free_context(ctx
, 1);
59 static void delete_smbctx(SMBCCTX
* ctx
){
60 smbc_getFunctionPurgeCachedServers(ctx
)(ctx
);
61 smbc_free_context(ctx
, 1);
64 static smbitem
* get_smbitem_list(SMBCCTX
*ctx
, char *smb_path
){
66 struct smbc_dirent
*dirent
;
67 smbitem
*list
= NULL
, *item
;
69 if ((fd
= smbc_getFunctionOpendir(ctx
)(ctx
, smb_path
)) == NULL
)
71 while((dirent
= smbc_getFunctionReaddir(ctx
)(ctx
, fd
)) != NULL
){
72 if (strcmp(dirent
->name
, "") == 0) continue;
73 if (strcmp(dirent
->name
, ".") == 0) continue;
74 if (strcmp(dirent
->name
, "..") == 0) continue;
76 if ((item
= malloc(sizeof(smbitem
) + strlen(dirent
->name
))) == NULL
)
80 item
->type
= dirent
->smbc_type
;
81 strcpy(item
->name
, dirent
->name
);
84 smbc_getFunctionClose(ctx
)(ctx
, fd
);
85 return /* smbitem_list_sort */ (list
);
89 static void print_smb_path(const char *group
, const char *path
){
90 if ((strlen(group
) == 0) && (strlen(path
) == 0)) printf("/\n");
91 else if (strlen(path
) == 0) printf("/%s\n", group
);
93 if (strlen(group
) == 0) group
= "(unknown_group)";
94 printf("/%s/%s\n", group
, path
);
98 static void recurse(SMBCCTX
*ctx
, const char *smb_group
, char *smb_path
, int maxlen
){
100 smbitem
*list
, *item
;
103 len
= strlen(smb_path
);
105 list
= get_smbitem_list(ctx
, smb_path
);
110 if (list
->type
== SMBC_WORKGROUP
){
111 print_smb_path(list
->name
, "");
112 smb_group
= list
->name
;
114 else print_smb_path(smb_group
, list
->name
);
116 if (maxlen
< 7 + strlen(list
->name
)) break;
117 strcpy(smb_path
+ 6, list
->name
);
118 if ((ctx1
= create_smbctx()) != NULL
){
119 recurse(ctx1
, smb_group
, smb_path
, maxlen
);
122 recurse(ctx
, smb_group
, smb_path
, maxlen
);
123 smbc_getFunctionPurgeCachedServers(ctx
)(ctx
);
126 case SMBC_FILE_SHARE
:
129 if (maxlen
< len
+ strlen(list
->name
) + 2) break;
132 strcpy(smb_path
+ len
+ 1, list
->name
);
133 print_smb_path(smb_group
, smb_path
+ 6);
134 if (list
->type
!= SMBC_FILE
){
135 recurse(ctx
, smb_group
, smb_path
, maxlen
);
136 if (list
->type
== SMBC_FILE_SHARE
)
137 smbc_getFunctionPurgeCachedServers(ctx
)(ctx
);
145 smb_path
[len
] = '\0';
148 int main(int argc
, char *argv
[]){
151 char smb_path
[32768] = "smb://";
153 if ((ctx
= create_smbctx()) == NULL
){
154 perror("Cant create samba context.");
158 if (argc
== 1) recurse(ctx
, "", smb_path
, sizeof(smb_path
));
159 else for(i
= 1; i
< argc
; i
++){
160 strncpy(smb_path
+ 6, argv
[i
], sizeof(smb_path
) - 7);
161 smb_path
[sizeof(smb_path
) - 1] = '\0';
162 recurse(ctx
, "", smb_path
, sizeof(smb_path
));