]> git.ipfire.org Git - thirdparty/samba.git/commitdiff
netapi: add netapi testsuite.
authorGünther Deschner <gd@samba.org>
Fri, 18 Jul 2008 17:12:42 +0000 (19:12 +0200)
committerGünther Deschner <gd@samba.org>
Wed, 30 Jul 2008 14:35:38 +0000 (16:35 +0200)
Guenther

source/lib/netapi/tests/Makefile.in [new file with mode: 0644]
source/lib/netapi/tests/common.c [new file with mode: 0644]
source/lib/netapi/tests/common.h [new file with mode: 0644]
source/lib/netapi/tests/netapitest.c [new file with mode: 0644]
source/lib/netapi/tests/netdisplay.c [new file with mode: 0644]
source/lib/netapi/tests/netgroup.c [new file with mode: 0644]
source/lib/netapi/tests/netlocalgroup.c [new file with mode: 0644]
source/lib/netapi/tests/netuser.c [new file with mode: 0644]

diff --git a/source/lib/netapi/tests/Makefile.in b/source/lib/netapi/tests/Makefile.in
new file mode 100644 (file)
index 0000000..f13281e
--- /dev/null
@@ -0,0 +1,57 @@
+KRB5LIBS=@KRB5_LIBS@
+LDAP_LIBS=@LDAP_LIBS@
+LIBS=@LIBS@ -lnetapi -ltdb -ltalloc
+DEVELOPER_CFLAGS=@DEVELOPER_CFLAGS@
+FLAGS=-I../ -L../../../bin @CFLAGS@ $(GTK_FLAGS)
+CC=@CC@
+PICFLAG=@PICFLAG@
+LDFLAGS=@PIE_LDFLAGS@ @LDFLAGS@
+DYNEXP=@DYNEXP@
+NETAPI_LIBS=$(LIBS) $(KRB5LIBS) $(LDAP_LIBS)
+CMDLINE_LIBS=$(NETAPI_LIBS) @POPTLIBS@
+
+# Compile a source file.
+COMPILE_CC = $(CC) -I. $(FLAGS) $(PICFLAG) -c $< -o $@
+COMPILE = $(COMPILE_CC)
+
+PROGS = bin/netapitest@EXEEXT@
+
+all: $(PROGS)
+
+MAKEDIR = || exec false; \
+         if test -d "$$dir"; then :; else \
+         echo mkdir "$$dir"; \
+         mkdir -p "$$dir" >/dev/null 2>&1 || \
+         test -d "$$dir" || \
+         mkdir "$$dir" || \
+         exec false; fi || exec false
+
+BINARY_PREREQS = bin/.dummy
+
+bin/.dummy:
+       @if (: >> $@ || : > $@) >/dev/null 2>&1; then :; else \
+         dir=bin $(MAKEDIR); fi
+       @: >> $@ || : > $@ # what a fancy emoticon!
+
+.c.o:
+       @if (: >> $@ || : > $@) >/dev/null 2>&1; then rm -f $@; else \
+        dir=`echo $@ | sed 's,/[^/]*$$,,;s,^$$,.,'` $(MAKEDIR); fi
+       @echo Compiling $*.c
+       @$(COMPILE) && exit 0;\
+               echo "The following command failed:" 1>&2;\
+               echo "$(COMPILE_CC)" 1>&2;\
+               $(COMPILE_CC) >/dev/null 2>&1
+
+CMDLINE_OBJ = common.o
+NETAPIBUFFER_OBJ = netapibuffer.o
+NETAPITEST_OBJ = netapitest.o netlocalgroup.o netuser.o netgroup.o netdisplay.o $(CMDLINE_OBJ)
+
+bin/netapitest@EXEEXT@: $(BINARY_PREREQS) $(NETAPITEST_OBJ)
+       @echo Linking $@
+       @$(CC) $(FLAGS) -o $@ $(NETAPITEST_OBJ) $(LDFLAGS) $(DYNEXP) $(CMDLINE_LIBS)
+
+clean:
+       -rm -f $(PROGS)
+       -rm -f core */*~ *~ \
+               */*.o */*/*.o */*/*/*.o
+
diff --git a/source/lib/netapi/tests/common.c b/source/lib/netapi/tests/common.c
new file mode 100644 (file)
index 0000000..22175af
--- /dev/null
@@ -0,0 +1,86 @@
+/*
+ *  Unix SMB/CIFS implementation.
+ *  NetApi testsuite
+ *  Copyright (C) Guenther Deschner 2008
+ *
+ *  This program is free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 3 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This program is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <stdlib.h>
+#include <string.h>
+#include <sys/types.h>
+#include <inttypes.h>
+
+#include <popt.h>
+#include <netapi.h>
+
+#include "common.h"
+
+void popt_common_callback(poptContext con,
+                        enum poptCallbackReason reason,
+                        const struct poptOption *opt,
+                        const char *arg, const void *data)
+{
+       struct libnetapi_ctx *ctx = NULL;
+
+       libnetapi_getctx(&ctx);
+
+       if (reason == POPT_CALLBACK_REASON_PRE) {
+       }
+
+       if (reason == POPT_CALLBACK_REASON_POST) {
+       }
+
+       if (!opt) {
+               return;
+       }
+       switch (opt->val) {
+               case 'U': {
+                       char *puser = strdup(arg);
+                       char *p = NULL;
+
+                       if ((p = strchr(puser,'%'))) {
+                               size_t len;
+                               *p = 0;
+                               libnetapi_set_username(ctx, puser);
+                               libnetapi_set_password(ctx, p+1);
+                               len = strlen(p+1);
+                               memset(strchr(arg,'%')+1,'X',len);
+                       } else {
+                               libnetapi_set_username(ctx, puser);
+                       }
+                       free(puser);
+                       break;
+               }
+               case 'd':
+                       libnetapi_set_debuglevel(ctx, arg);
+                       break;
+               case 'p':
+                       libnetapi_set_password(ctx, arg);
+                       break;
+               case 'k':
+                       libnetapi_set_use_kerberos(ctx);
+                       break;
+       }
+}
+
+struct poptOption popt_common_netapi_examples[] = {
+       { NULL, 0, POPT_ARG_CALLBACK|POPT_CBFLAG_PRE|POPT_CBFLAG_POST, (void *)popt_common_callback },
+       { "user", 'U', POPT_ARG_STRING, NULL, 'U', "Username used for connection", "USERNAME" },
+       { "password", 'p', POPT_ARG_STRING, NULL, 'p', "Password used for connection", "PASSWORD" },
+       { "debuglevel", 'd', POPT_ARG_STRING, NULL, 'd', "Debuglevel", "DEBUGLEVEL" },
+       { "kerberos", 'k', POPT_ARG_NONE, NULL, 'k', "Use Kerberos", NULL },
+       POPT_TABLEEND
+};
+
diff --git a/source/lib/netapi/tests/common.h b/source/lib/netapi/tests/common.h
new file mode 100644 (file)
index 0000000..ed073c0
--- /dev/null
@@ -0,0 +1,55 @@
+/*
+ *  Unix SMB/CIFS implementation.
+ *  NetApi testsuite
+ *  Copyright (C) Guenther Deschner 2008
+ *
+ *  This program is free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 3 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This program is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <popt.h>
+
+void popt_common_callback(poptContext con,
+                        enum poptCallbackReason reason,
+                        const struct poptOption *opt,
+                        const char *arg, const void *data);
+
+extern struct poptOption popt_common_netapi_examples[];
+
+#define POPT_COMMON_LIBNETAPI_EXAMPLES { NULL, 0, POPT_ARG_INCLUDE_TABLE, popt_common_netapi_examples, 0, "Common samba netapi example options:", NULL },
+
+NET_API_STATUS test_netuseradd(const char *hostname,
+                              const char *username);
+
+NET_API_STATUS netapitest_localgroup(struct libnetapi_ctx *ctx,
+                                    const char *hostname);
+NET_API_STATUS netapitest_user(struct libnetapi_ctx *ctx,
+                              const char *hostname);
+NET_API_STATUS netapitest_group(struct libnetapi_ctx *ctx,
+                               const char *hostname);
+NET_API_STATUS netapitest_display(struct libnetapi_ctx *ctx,
+                                 const char *hostname);
+
+#ifndef ARRAY_SIZE
+#define ARRAY_SIZE(a) (sizeof(a)/sizeof(a[0]))
+#endif
+
+#define NETAPI_STATUS(x,y,fn) \
+       printf("FAILURE: line %d: %s failed with status: %s (%d)\n", \
+               __LINE__, fn, libnetapi_get_error_string(x,y), y);
+
+#define NETAPI_STATUS_MSG(x,y,fn,z) \
+       printf("FAILURE: line %d: %s failed with status: %s (%d), %s\n", \
+               __LINE__, fn, libnetapi_get_error_string(x,y), y, z);
+
+#define ZERO_STRUCT(x) memset((char *)&(x), 0, sizeof(x))
diff --git a/source/lib/netapi/tests/netapitest.c b/source/lib/netapi/tests/netapitest.c
new file mode 100644 (file)
index 0000000..de81f5e
--- /dev/null
@@ -0,0 +1,92 @@
+/*
+ *  Unix SMB/CIFS implementation.
+ *  NetApi testsuite
+ *  Copyright (C) Guenther Deschner 2008
+ *
+ *  This program is free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 3 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This program is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <sys/types.h>
+#include <inttypes.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include <netapi.h>
+
+#include "common.h"
+
+int main(int argc, const char **argv)
+{
+       NET_API_STATUS status = 0;
+       struct libnetapi_ctx *ctx = NULL;
+       const char *hostname = NULL;
+
+       poptContext pc;
+       int opt;
+
+       struct poptOption long_options[] = {
+               POPT_AUTOHELP
+               POPT_COMMON_LIBNETAPI_EXAMPLES
+               POPT_TABLEEND
+       };
+
+       status = libnetapi_init(&ctx);
+       if (status != 0) {
+               return status;
+       }
+
+       pc = poptGetContext("netapitest", argc, argv, long_options, 0);
+
+       poptSetOtherOptionHelp(pc, "hostname");
+       while((opt = poptGetNextOpt(pc)) != -1) {
+       }
+
+       if (!poptPeekArg(pc)) {
+               poptPrintHelp(pc, stderr, 0);
+               goto out;
+       }
+       hostname = poptGetArg(pc);
+
+       status = netapitest_localgroup(ctx, hostname);
+       if (status) {
+               goto out;
+       }
+
+       status = netapitest_user(ctx, hostname);
+       if (status) {
+               goto out;
+       }
+
+       status = netapitest_group(ctx, hostname);
+       if (status) {
+               goto out;
+       }
+
+       status = netapitest_display(ctx, hostname);
+       if (status) {
+               goto out;
+       }
+
+ out:
+       if (status != 0) {
+               printf("testsuite failed with: %s\n",
+                       libnetapi_get_error_string(ctx, status));
+       }
+
+       libnetapi_free(ctx);
+       poptFreeContext(pc);
+
+       return status;
+}
diff --git a/source/lib/netapi/tests/netdisplay.c b/source/lib/netapi/tests/netdisplay.c
new file mode 100644 (file)
index 0000000..090792c
--- /dev/null
@@ -0,0 +1,150 @@
+/*
+ *  Unix SMB/CIFS implementation.
+ *  NetGroup testsuite
+ *  Copyright (C) Guenther Deschner 2008
+ *
+ *  This program is free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 3 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This program is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <sys/types.h>
+#include <inttypes.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include <netapi.h>
+
+#include "common.h"
+
+static NET_API_STATUS test_netquerydisplayinformation(const char *hostname,
+                                                     uint32_t level,
+                                                     const char *name)
+{
+       NET_API_STATUS status;
+       uint32_t entries_read = 0;
+       int found_name = 0;
+       const char *current_name;
+       uint8_t *buffer = NULL;
+       uint32_t idx = 0;
+       int i;
+
+       struct NET_DISPLAY_USER *user;
+       struct NET_DISPLAY_GROUP *group;
+       struct NET_DISPLAY_MACHINE *machine;
+
+       printf("testing NetQueryDisplayInformation level %d\n", level);
+
+       do {
+               status = NetQueryDisplayInformation(hostname,
+                                                   level,
+                                                   idx,
+                                                   1000,
+                                                   (uint32_t)-1,
+                                                   &entries_read,
+                                                   (void **)&buffer);
+               if (status == 0 || status == ERROR_MORE_DATA) {
+                       switch (level) {
+                               case 1:
+                                       user = (struct NET_DISPLAY_USER *)buffer;
+                                       break;
+                               case 2:
+                                       machine = (struct NET_DISPLAY_MACHINE *)buffer;
+                                       break;
+                               case 3:
+                                       group = (struct NET_DISPLAY_GROUP *)buffer;
+                                       break;
+                               default:
+                                       return -1;
+                       }
+
+                       for (i=0; i<entries_read; i++) {
+
+                               switch (level) {
+                                       case 1:
+                                               current_name =  user->usri1_name;
+                                               break;
+                                       case 2:
+                                               current_name =  machine->usri2_name;
+                                               break;
+                                       case 3:
+                                               current_name =  group->grpi3_name;
+                                               break;
+                                       default:
+                                               break;
+                               }
+
+                               if (name && strcasecmp(current_name, name) == 0) {
+                                       found_name = 1;
+                               }
+
+                               switch (level) {
+                                       case 1:
+                                               user++;
+                                               break;
+                                       case 2:
+                                               machine++;
+                                               break;
+                                       case 3:
+                                               group++;
+                                               break;
+                               }
+                       }
+                       NetApiBufferFree(buffer);
+               }
+               idx += entries_read;
+       } while (status == ERROR_MORE_DATA);
+
+       if (status) {
+               return status;
+       }
+
+       if (name && !found_name) {
+               printf("failed to get name\n");
+               return -1;
+       }
+
+       return 0;
+}
+
+NET_API_STATUS netapitest_display(struct libnetapi_ctx *ctx,
+                                 const char *hostname)
+{
+       NET_API_STATUS status = 0;
+       uint32_t levels[] = { 1, 2, 3};
+       int i;
+
+       printf("NetDisplay tests\n");
+
+       /* test enum */
+
+       for (i=0; i<ARRAY_SIZE(levels); i++) {
+
+               status = test_netquerydisplayinformation(hostname, levels[i], NULL);
+               if (status) {
+                       NETAPI_STATUS(ctx, status, "NetQueryDisplayInformation");
+                       goto out;
+               }
+       }
+
+       status = 0;
+
+       printf("NetDisplay tests succeeded\n");
+ out:
+       if (status != 0) {
+               printf("NetDisplay testsuite failed with: %s\n",
+                       libnetapi_get_error_string(ctx, status));
+       }
+
+       return status;
+}
diff --git a/source/lib/netapi/tests/netgroup.c b/source/lib/netapi/tests/netgroup.c
new file mode 100644 (file)
index 0000000..a89a772
--- /dev/null
@@ -0,0 +1,286 @@
+/*
+ *  Unix SMB/CIFS implementation.
+ *  NetGroup testsuite
+ *  Copyright (C) Guenther Deschner 2008
+ *
+ *  This program is free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 3 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This program is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <sys/types.h>
+#include <inttypes.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include <netapi.h>
+
+#include "common.h"
+
+static NET_API_STATUS test_netgroupenum(const char *hostname,
+                                       uint32_t level,
+                                       const char *groupname)
+{
+       NET_API_STATUS status;
+       uint32_t entries_read = 0;
+       uint32_t total_entries = 0;
+       uint32_t resume_handle = 0;
+       int found_group = 0;
+       const char *current_name;
+       uint8_t *buffer = NULL;
+       int i;
+
+       struct GROUP_INFO_0 *info0;
+       struct GROUP_INFO_1 *info1;
+       struct GROUP_INFO_2 *info2;
+       struct GROUP_INFO_3 *info3;
+
+       printf("testing NetGroupEnum level %d\n", level);
+
+       do {
+               status = NetGroupEnum(hostname,
+                                     level,
+                                     &buffer,
+                                     120, //(uint32_t)-1,
+                                     &entries_read,
+                                     &total_entries,
+                                     &resume_handle);
+               if (status == 0 || status == ERROR_MORE_DATA) {
+                       switch (level) {
+                               case 0:
+                                       info0 = (struct GROUP_INFO_0 *)buffer;
+                                       break;
+                               case 1:
+                                       info1 = (struct GROUP_INFO_1 *)buffer;
+                                       break;
+                               case 2:
+                                       info2 = (struct GROUP_INFO_2 *)buffer;
+                                       break;
+                               case 3:
+                                       info3 = (struct GROUP_INFO_3 *)buffer;
+                                       break;
+                               default:
+                                       return -1;
+                       }
+
+                       for (i=0; i<entries_read; i++) {
+
+                               switch (level) {
+                                       case 0:
+                                               current_name =  info0->grpi0_name;
+                                               break;
+                                       case 1:
+                                               current_name =  info1->grpi1_name;
+                                               break;
+                                       case 2:
+                                               current_name =  info2->grpi2_name;
+                                               break;
+                                       case 3:
+                                               current_name =  info3->grpi3_name;
+                                               break;
+                                       default:
+                                               break;
+                               }
+
+                               if (strcasecmp(current_name, groupname) == 0) {
+                                       found_group = 1;
+                               }
+
+                               switch (level) {
+                                       case 0:
+                                               info0++;
+                                               break;
+                                       case 1:
+                                               info1++;
+                                               break;
+                                       case 2:
+                                               info2++;
+                                               break;
+                                       case 3:
+                                               info3++;
+                                               break;
+                               }
+                       }
+                       NetApiBufferFree(buffer);
+               }
+       } while (status == ERROR_MORE_DATA);
+
+       if (status) {
+               return status;
+       }
+
+       if (!found_group) {
+               printf("failed to get group\n");
+               return -1;
+       }
+
+       return 0;
+}
+
+NET_API_STATUS netapitest_group(struct libnetapi_ctx *ctx,
+                               const char *hostname)
+{
+       NET_API_STATUS status = 0;
+       const char *username, *groupname, *groupname2;
+       uint8_t *buffer = NULL;
+       struct GROUP_INFO_0 g0;
+       uint32_t parm_err = 0;
+       uint32_t levels[] = { 0, 1, 2, 3};
+       uint32_t enum_levels[] = { 0, 1, 2, 3};
+       int i;
+
+       printf("NetGroup tests\n");
+
+       username = "torture_test_user";
+       groupname = "torture_test_group";
+       groupname2 = "torture_test_group2";
+
+       /* cleanup */
+       NetGroupDel(hostname, groupname);
+       NetGroupDel(hostname, groupname2);
+       NetUserDel(hostname, username);
+
+       /* add a group */
+
+       g0.grpi0_name = groupname;
+
+       printf("testing NetGroupAdd\n");
+
+       status = NetGroupAdd(hostname, 0, (uint8_t *)&g0, &parm_err);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetGroupAdd");
+               goto out;
+       }
+
+       /* 2nd add must fail */
+
+       status = NetGroupAdd(hostname, 0, (uint8_t *)&g0, &parm_err);
+       if (status == 0) {
+               NETAPI_STATUS(ctx, status, "NetGroupAdd");
+               goto out;
+       }
+
+       /* test enum */
+
+       for (i=0; i<ARRAY_SIZE(enum_levels); i++) {
+
+               status = test_netgroupenum(hostname, enum_levels[i], groupname);
+               if (status) {
+                       NETAPI_STATUS(ctx, status, "NetGroupEnum");
+                       goto out;
+               }
+       }
+
+       /* basic queries */
+
+       for (i=0; i<ARRAY_SIZE(levels); i++) {
+
+               printf("testing NetGroupGetInfo level %d\n", levels[i]);
+
+               status = NetGroupGetInfo(hostname, groupname, levels[i], &buffer);
+               if (status && status != 124) {
+                       NETAPI_STATUS(ctx, status, "NetGroupGetInfo");
+                       goto out;
+               }
+       }
+
+       /* group rename */
+
+       g0.grpi0_name = groupname2;
+
+       printf("testing NetGroupSetInfo level 0\n");
+
+       status = NetGroupSetInfo(hostname, groupname, 0, (uint8_t *)&g0, &parm_err);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetGroupSetInfo");
+               goto out;
+       }
+
+       /* should not exist anymore */
+
+       status = NetGroupDel(hostname, groupname);
+       if (status == 0) {
+               NETAPI_STATUS(ctx, status, "NetGroupDel");
+               goto out;
+       }
+
+       /* query info */
+
+       for (i=0; i<ARRAY_SIZE(levels); i++) {
+
+               status = NetGroupGetInfo(hostname, groupname2, levels[i], &buffer);
+               if (status && status != 124) {
+                       NETAPI_STATUS(ctx, status, "NetGroupGetInfo");
+                       goto out;
+               }
+       }
+
+       /* add user to group */
+
+       status = test_netuseradd(hostname, username);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetUserAdd");
+               goto out;
+       }
+
+       printf("testing NetGroupAddUser\n");
+
+       status = NetGroupAddUser(hostname, groupname2, username);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetGroupAddUser");
+               goto out;
+       }
+
+       printf("testing NetGroupDelUser\n");
+
+       status = NetGroupDelUser(hostname, groupname2, username);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetGroupDelUser");
+               goto out;
+       }
+
+       status = NetUserDel(hostname, username);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetUserDel");
+               goto out;
+       }
+
+       /* delete */
+
+       printf("testing NetGroupDel\n");
+
+       status = NetGroupDel(hostname, groupname2);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetGroupDel");
+               goto out;
+       };
+
+       /* should not exist anymore */
+
+       status = NetGroupGetInfo(hostname, groupname2, 0, &buffer);
+       if (status == 0) {
+               NETAPI_STATUS_MSG(ctx, status, "NetGroupGetInfo", "expected failure and error code");
+               goto out;
+       };
+
+       status = 0;
+
+       printf("NetGroup tests succeeded\n");
+ out:
+       if (status != 0) {
+               printf("NetGroup testsuite failed with: %s\n",
+                       libnetapi_get_error_string(ctx, status));
+       }
+
+       return status;
+}
diff --git a/source/lib/netapi/tests/netlocalgroup.c b/source/lib/netapi/tests/netlocalgroup.c
new file mode 100644 (file)
index 0000000..0d82059
--- /dev/null
@@ -0,0 +1,226 @@
+/*
+ *  Unix SMB/CIFS implementation.
+ *  NetLocalGroup testsuite
+ *  Copyright (C) Guenther Deschner 2008
+ *
+ *  This program is free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 3 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This program is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <sys/types.h>
+#include <inttypes.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include <netapi.h>
+
+#include "common.h"
+
+static NET_API_STATUS test_netlocalgroupenum(const char *hostname,
+                                            uint32_t level,
+                                            const char *groupname)
+{
+       NET_API_STATUS status;
+       uint32_t entries_read = 0;
+       uint32_t total_entries = 0;
+       uint32_t resume_handle = 0;
+       int found_group = 0;
+       const char *current_name;
+       uint8_t *buffer = NULL;
+       int i;
+
+       struct LOCALGROUP_INFO_0 *info0;
+       struct LOCALGROUP_INFO_1 *info1;
+
+       printf("testing NetLocalGroupEnum level %d\n", level);
+
+       do {
+               status = NetLocalGroupEnum(hostname,
+                                          level,
+                                          &buffer,
+                                          (uint32_t)-1,
+                                          &entries_read,
+                                          &total_entries,
+                                          &resume_handle);
+               if (status == 0 || status == ERROR_MORE_DATA) {
+                       switch (level) {
+                               case 0:
+                                       info0 = (struct LOCALGROUP_INFO_0 *)buffer;
+                                       break;
+                               case 1:
+                                       info1 = (struct LOCALGROUP_INFO_1 *)buffer;
+                                       break;
+                               default:
+                                       return -1;
+                       }
+
+                       for (i=0; i<entries_read; i++) {
+
+                               switch (level) {
+                                       case 0:
+                                               current_name =  info0->lgrpi0_name;
+                                               break;
+                                       case 1:
+                                               current_name =  info1->lgrpi1_name;
+                                               break;
+                                       default:
+                                               break;
+                               }
+
+                               if (strcasecmp(current_name, groupname) == 0) {
+                                       found_group = 1;
+                               }
+
+                               switch (level) {
+                                       case 0:
+                                               info0++;
+                                               break;
+                                       case 1:
+                                               info1++;
+                                               break;
+                               }
+                       }
+                       NetApiBufferFree(buffer);
+               }
+       } while (status == ERROR_MORE_DATA);
+
+       if (status) {
+               return status;
+       }
+
+       if (!found_group) {
+               printf("failed to get group\n");
+               return -1;
+       }
+
+       return 0;
+}
+
+NET_API_STATUS netapitest_localgroup(struct libnetapi_ctx *ctx,
+                                    const char *hostname)
+{
+       NET_API_STATUS status = 0;
+       const char *groupname, *groupname2;
+       uint8_t *buffer = NULL;
+       struct LOCALGROUP_INFO_0 g0;
+       uint32_t parm_err = 0;
+       uint32_t levels[] = { 0, 1, 1002 };
+       uint32_t enum_levels[] = { 0, 1 };
+       int i;
+
+       printf("NetLocalgroup tests\n");
+
+       groupname = "torture_test_localgroup";
+       groupname2 = "torture_test_localgroup2";
+
+       /* cleanup */
+       NetLocalGroupDel(hostname, groupname);
+       NetLocalGroupDel(hostname, groupname2);
+
+       /* add a localgroup */
+
+       printf("testing NetLocalGroupAdd\n");
+
+       g0.lgrpi0_name = groupname;
+
+       status = NetLocalGroupAdd(hostname, 0, (uint8_t *)&g0, &parm_err);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetLocalGroupAdd");
+               goto out;
+       };
+
+       /* test enum */
+
+       for (i=0; i<ARRAY_SIZE(enum_levels); i++) {
+
+               status = test_netlocalgroupenum(hostname, enum_levels[i], groupname);
+               if (status) {
+                       NETAPI_STATUS(ctx, status, "NetLocalGroupEnum");
+                       goto out;
+               }
+       }
+
+
+       /* basic queries */
+
+       for (i=0; i<ARRAY_SIZE(levels); i++) {
+
+               printf("testing NetLocalGroupGetInfo level %d\n", levels[i]);
+
+               status = NetLocalGroupGetInfo(hostname, groupname, levels[i], &buffer);
+               if (status && status != 124) {
+                       NETAPI_STATUS(ctx, status, "NetLocalGroupGetInfo");
+                       goto out;
+               };
+       }
+
+       /* alias rename */
+
+       printf("testing NetLocalGroupSetInfo level 0\n");
+
+       g0.lgrpi0_name = groupname2;
+
+       status = NetLocalGroupSetInfo(hostname, groupname, 0, (uint8_t *)&g0, &parm_err);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetLocalGroupSetInfo");
+               goto out;
+       };
+
+       /* should not exist anymore */
+
+       status = NetLocalGroupDel(hostname, groupname);
+       if (status == 0) {
+               NETAPI_STATUS(ctx, status, "NetLocalGroupDel");
+               goto out;
+       };
+
+       /* query info */
+
+       for (i=0; i<ARRAY_SIZE(levels); i++) {
+               status = NetLocalGroupGetInfo(hostname, groupname2, levels[i], &buffer);
+               if (status && status != 124) {
+                       NETAPI_STATUS(ctx, status, "NetLocalGroupGetInfo");
+                       goto out;
+               };
+       }
+
+       /* delete */
+
+       printf("testing NetLocalGroupDel\n");
+
+       status = NetLocalGroupDel(hostname, groupname2);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetLocalGroupDel");
+               goto out;
+       };
+
+       /* should not exist anymore */
+
+       status = NetLocalGroupGetInfo(hostname, groupname2, 0, &buffer);
+       if (status == 0) {
+               NETAPI_STATUS(ctx, status, "NetLocalGroupGetInfo");
+               goto out;
+       };
+
+       status = 0;
+
+       printf("NetLocalgroup tests succeeded\n");
+ out:
+       if (status != 0) {
+               printf("NetLocalGroup testsuite failed with: %s\n",
+                       libnetapi_get_error_string(ctx, status));
+       }
+
+       return status;
+}
diff --git a/source/lib/netapi/tests/netuser.c b/source/lib/netapi/tests/netuser.c
new file mode 100644 (file)
index 0000000..ca2dc1a
--- /dev/null
@@ -0,0 +1,259 @@
+/*
+ *  Unix SMB/CIFS implementation.
+ *  NetUser testsuite
+ *  Copyright (C) Guenther Deschner 2008
+ *
+ *  This program is free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 3 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This program is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <sys/types.h>
+#include <inttypes.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include <netapi.h>
+
+#include "common.h"
+
+static NET_API_STATUS test_netuserenum(const char *hostname,
+                                      uint32_t level,
+                                      const char *username)
+{
+       NET_API_STATUS status;
+       uint32_t entries_read = 0;
+       uint32_t total_entries = 0;
+       uint32_t resume_handle = 0;
+       const char *current_name;
+       int found_user = 0;
+       uint8_t *buffer = NULL;
+       int i;
+
+       struct USER_INFO_0 *info0;
+       struct USER_INFO_10 *info10;
+       struct USER_INFO_20 *info20;
+       struct USER_INFO_23 *info23;
+
+       printf("testing NetUserEnum level %d\n", level);
+
+       do {
+               status = NetUserEnum(hostname,
+                                    level,
+                                    FILTER_NORMAL_ACCOUNT,
+                                    &buffer,
+                                    120, //(uint32_t)-1,
+                                    &entries_read,
+                                    &total_entries,
+                                    &resume_handle);
+               if (status == 0 || status == ERROR_MORE_DATA) {
+                       switch (level) {
+                               case 0:
+                                       info0 = (struct USER_INFO_0 *)buffer;
+                                       break;
+                               case 10:
+                                       info10 = (struct USER_INFO_10 *)buffer;
+                                       break;
+                               case 20:
+                                       info20 = (struct USER_INFO_20 *)buffer;
+                                       break;
+                               case 23:
+                                       info23 = (struct USER_INFO_23 *)buffer;
+                                       break;
+                               default:
+                                       return -1;
+                       }
+
+                       for (i=0; i<entries_read; i++) {
+
+                               switch (level) {
+                                       case 0:
+                                               current_name = info0->usri0_name;
+                                               break;
+                                       case 10:
+                                               current_name = info10->usri10_name;
+                                               break;
+                                       case 20:
+                                               current_name = info20->usri20_name;
+                                               break;
+                                       case 23:
+                                               current_name = info23->usri23_name;
+                                               break;
+                                       default:
+                                               return -1;
+                               }
+
+                               if (strcasecmp(current_name, username) == 0) {
+                                       found_user = 1;
+                               }
+
+                               switch (level) {
+                                       case 0:
+                                               info0++;
+                                               break;
+                                       case 10:
+                                               info10++;
+                                               break;
+                                       case 20:
+                                               info20++;
+                                               break;
+                                       case 23:
+                                               info23++;
+                                               break;
+                                       default:
+                                               break;
+                               }
+                       }
+                       NetApiBufferFree(buffer);
+               }
+       } while (status == ERROR_MORE_DATA);
+
+       if (status) {
+               return status;
+       }
+
+       if (!found_user) {
+               printf("failed to get user\n");
+               return -1;
+       }
+
+       return 0;
+}
+
+NET_API_STATUS test_netuseradd(const char *hostname,
+                              const char *username)
+{
+       struct USER_INFO_1 u1;
+       uint32_t parm_err = 0;
+
+       ZERO_STRUCT(u1);
+
+       printf("testing NetUserAdd\n");
+
+       u1.usri1_name = username;
+       u1.usri1_password = "W297!832jD8J";
+       u1.usri1_password_age = 0;
+       u1.usri1_priv = 0;
+       u1.usri1_home_dir = NULL;
+       u1.usri1_comment = "User created using Samba NetApi Example code";
+       u1.usri1_flags = 0;
+       u1.usri1_script_path = NULL;
+
+       return NetUserAdd(hostname, 1, (uint8_t *)&u1, &parm_err);
+}
+
+NET_API_STATUS netapitest_user(struct libnetapi_ctx *ctx,
+                              const char *hostname)
+{
+       NET_API_STATUS status = 0;
+       const char *username, *username2;
+       uint8_t *buffer = NULL;
+       uint32_t levels[] = { 0, 10, 20, 23 };
+       uint32_t enum_levels[] = { 0, 10, 20, 23 };
+       int i;
+
+       struct USER_INFO_1007 u1007;
+       uint32_t parm_err = 0;
+
+       printf("NetUser tests\n");
+
+       username = "torture_test_user";
+       username2 = "torture_test_user2";
+
+       /* cleanup */
+       NetUserDel(hostname, username);
+       NetUserDel(hostname, username2);
+
+       /* add a user */
+
+       status = test_netuseradd(hostname, username);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetUserAdd");
+               goto out;
+       }
+
+       /* enum the new user */
+
+       for (i=0; i<ARRAY_SIZE(enum_levels); i++) {
+
+               status = test_netuserenum(hostname, enum_levels[i], username);
+               if (status) {
+                       NETAPI_STATUS(ctx, status, "NetUserEnum");
+                       goto out;
+               }
+       }
+
+       /* basic queries */
+
+       for (i=0; i<ARRAY_SIZE(levels); i++) {
+
+               printf("testing NetUserGetInfo level %d\n", levels[i]);
+
+               status = NetUserGetInfo(hostname, username, levels[i], &buffer);
+               if (status && status != 124) {
+                       NETAPI_STATUS(ctx, status, "NetUserGetInfo");
+                       goto out;
+               }
+       }
+
+       /* modify description */
+
+       printf("testing NetUserSetInfo level %d\n", 1007);
+
+       u1007.usri1007_comment = "NetApi modified user";
+
+       status = NetUserSetInfo(hostname, username, 1007, (uint8_t *)&u1007, &parm_err);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetUserSetInfo");
+               goto out;
+       }
+
+       /* query info */
+
+       for (i=0; i<ARRAY_SIZE(levels); i++) {
+               status = NetUserGetInfo(hostname, username, levels[i], &buffer);
+               if (status && status != 124) {
+                       NETAPI_STATUS(ctx, status, "NetUserGetInfo");
+                       goto out;
+               }
+       }
+
+       /* delete */
+
+       printf("testing NetUserDel\n");
+
+       status = NetUserDel(hostname, username);
+       if (status) {
+               NETAPI_STATUS(ctx, status, "NetUserDel");
+               goto out;
+       }
+
+       /* should not exist anymore */
+
+       status = NetUserGetInfo(hostname, username, 0, &buffer);
+       if (status == 0) {
+               NETAPI_STATUS(ctx, status, "NetUserGetInfo");
+               goto out;
+       }
+
+       status = 0;
+
+       printf("NetUser tests succeeded\n");
+ out:
+       if (status != 0) {
+               printf("NetUser testsuite failed with: %s\n",
+                       libnetapi_get_error_string(ctx, status));
+       }
+
+       return status;
+}