s3-talloc Change TALLOC_ZERO_P() to talloc_zero()
[nivanova/samba-autobuild/.git] / source3 / lib / netapi / netapi.c
index 1db745b5c7205460a52034822fa561cd25870e2c..14259864ae2bce4f7fb979b3c74e4bb5a19f2acc 100644 (file)
@@ -1,7 +1,7 @@
 /*
  *  Unix SMB/CIFS implementation.
  *  NetApi Support
- *  Copyright (C) Guenther Deschner 2007
+ *  Copyright (C) Guenther Deschner 2007-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
 
 #include "includes.h"
 #include "lib/netapi/netapi.h"
-
-extern bool AllowDebugChange;
+#include "lib/netapi/netapi_private.h"
+#include "secrets.h"
+#include "krb5_env.h"
 
 struct libnetapi_ctx *stat_ctx = NULL;
-TALLOC_CTX *frame = NULL;
 static bool libnetapi_initialized = false;
 
-NET_API_STATUS libnetapi_init(struct libnetapi_ctx **context)
+/****************************************************************
+****************************************************************/
+
+static NET_API_STATUS libnetapi_init_private_context(struct libnetapi_ctx *ctx)
 {
-       struct libnetapi_ctx *ctx = NULL;
+       struct libnetapi_private_ctx *priv;
 
-       if (stat_ctx && libnetapi_initialized) {
-               *context = stat_ctx;
-               return W_ERROR_V(WERR_OK);
+       if (!ctx) {
+               return W_ERROR_V(WERR_INVALID_PARAM);
        }
 
-       frame = talloc_stackframe();
-
-       ctx = talloc_zero(frame, struct libnetapi_ctx);
-       if (!ctx) {
-               TALLOC_FREE(frame);
+       priv = talloc_zero(ctx, struct libnetapi_private_ctx);
+       if (!priv) {
                return W_ERROR_V(WERR_NOMEM);
        }
 
-       DEBUGLEVEL = 0;
-       setup_logging("libnetapi", true);
+       ctx->private_data = priv;
 
-       dbf = x_stderr;
-       x_setbuf(x_stderr, NULL);
-       AllowDebugChange = false;
+       return NET_API_STATUS_SUCCESS;
+}
 
-       load_case_tables();
+/****************************************************************
+Create a libnetapi context, for use in non-Samba applications.  This
+loads the smb.conf file and sets the debug level to 0, so that
+applications are not flooded with debug logs at level 10, when they
+were not expecting it.
+****************************************************************/
 
-       if (!lp_load(get_dyn_CONFIGFILE(), true, false, false, false)) {
+NET_API_STATUS libnetapi_init(struct libnetapi_ctx **context)
+{
+       NET_API_STATUS ret;
+       TALLOC_CTX *frame;
+       if (stat_ctx && libnetapi_initialized) {
+               *context = stat_ctx;
+               return NET_API_STATUS_SUCCESS;
+       }
+
+#if 0
+       talloc_enable_leak_report();
+#endif
+       frame = talloc_stackframe();
+
+       /* Case tables must be loaded before any string comparisons occour */
+       load_case_tables_library();
+
+       /* When libnetapi is invoked from an application, it does not
+        * want to be swamped with level 10 debug messages, even if
+        * this has been set for the server in smb.conf */
+       lp_set_cmdline("log level", "0");
+       setup_logging("libnetapi", DEBUG_STDERR);
+
+       if (!lp_load(get_dyn_CONFIGFILE(), true, false, false, true)) {
                TALLOC_FREE(frame);
+               fprintf(stderr, "error loading %s\n", get_dyn_CONFIGFILE() );
                return W_ERROR_V(WERR_GENERAL_FAILURE);
        }
 
-       AllowDebugChange = true;
-
        init_names();
        load_interfaces();
        reopen_logs();
 
        BlockSignals(True, SIGPIPE);
 
+       ret = libnetapi_net_init(context);
+       TALLOC_FREE(frame);
+       return ret;
+}
+
+/****************************************************************
+Create a libnetapi context, for use inside the 'net' binary.
+
+As we know net has already loaded the smb.conf file, and set the debug
+level etc, this avoids doing so again (which causes trouble with -d on
+the command line).
+****************************************************************/
+
+NET_API_STATUS libnetapi_net_init(struct libnetapi_ctx **context)
+{
+       NET_API_STATUS status;
+       struct libnetapi_ctx *ctx = NULL;
+       char *krb5_cc_env = NULL;
+
+       TALLOC_CTX *frame = talloc_stackframe();
+
+       ctx = talloc_zero(frame, struct libnetapi_ctx);
+       if (!ctx) {
+               TALLOC_FREE(frame);
+               return W_ERROR_V(WERR_NOMEM);
+       }
+
+       BlockSignals(True, SIGPIPE);
+
+       krb5_cc_env = getenv(KRB5_ENV_CCNAME);
+       if (!krb5_cc_env || (strlen(krb5_cc_env) == 0)) {
+               ctx->krb5_cc_env = talloc_strdup(ctx, "MEMORY:libnetapi");
+               setenv(KRB5_ENV_CCNAME, ctx->krb5_cc_env, 1);
+       }
+
+       if (getenv("USER")) {
+               ctx->username = talloc_strdup(ctx, getenv("USER"));
+       } else {
+               ctx->username = talloc_strdup(ctx, "");
+       }
+       if (!ctx->username) {
+               TALLOC_FREE(frame);
+               fprintf(stderr, "libnetapi_init: out of memory\n");
+               return W_ERROR_V(WERR_NOMEM);
+       }
+
+       status = libnetapi_init_private_context(ctx);
+       if (status != 0) {
+               TALLOC_FREE(frame);
+               return status;
+       }
+
        libnetapi_initialized = true;
 
+       talloc_steal(NULL, ctx);
        *context = stat_ctx = ctx;
-
-       return W_ERROR_V(WERR_OK);
+       
+       TALLOC_FREE(frame);
+       return NET_API_STATUS_SUCCESS;
 }
 
+/****************************************************************
+ Return the static libnetapi context
+****************************************************************/
+
 NET_API_STATUS libnetapi_getctx(struct libnetapi_ctx **ctx)
 {
        if (stat_ctx) {
                *ctx = stat_ctx;
-               return W_ERROR_V(WERR_OK);
+               return NET_API_STATUS_SUCCESS;
        }
 
        return libnetapi_init(ctx);
 }
 
+/****************************************************************
+ Free the static libnetapi context
+****************************************************************/
+
 NET_API_STATUS libnetapi_free(struct libnetapi_ctx *ctx)
 {
+       if (!ctx) {
+               return NET_API_STATUS_SUCCESS;
+       }
+
+       libnetapi_samr_free(ctx);
+
+       libnetapi_shutdown_cm(ctx);
+
+       if (ctx->krb5_cc_env) {
+               char *env = getenv(KRB5_ENV_CCNAME);
+               if (env && (strequal(ctx->krb5_cc_env, env))) {
+                       unsetenv(KRB5_ENV_CCNAME);
+               }
+       }
+
        gfree_names();
        gfree_loadparm();
-       gfree_case_tables();
        gfree_charcnv();
        gfree_interfaces();
 
+       secrets_shutdown();
+
+       if (ctx == stat_ctx) {
+               stat_ctx = NULL;
+       }
        TALLOC_FREE(ctx);
-       TALLOC_FREE(frame);
 
        gfree_debugsyms();
 
-       return W_ERROR_V(WERR_OK);
+       return NET_API_STATUS_SUCCESS;
 }
 
+/****************************************************************
+ Override the current log level for libnetapi
+****************************************************************/
+
 NET_API_STATUS libnetapi_set_debuglevel(struct libnetapi_ctx *ctx,
                                        const char *debuglevel)
 {
-       AllowDebugChange = true;
-       ctx->debuglevel = debuglevel;
-       if (!debug_parse_levels(debuglevel)) {
+       TALLOC_CTX *frame = talloc_stackframe();
+       ctx->debuglevel = talloc_strdup(ctx, debuglevel);
+       
+       if (!lp_set_cmdline("log level", debuglevel)) {
+               TALLOC_FREE(frame);
                return W_ERROR_V(WERR_GENERAL_FAILURE);
        }
-       return W_ERROR_V(WERR_OK);
+       TALLOC_FREE(frame);
+       return NET_API_STATUS_SUCCESS;
 }
 
+/****************************************************************
+****************************************************************/
+
 NET_API_STATUS libnetapi_get_debuglevel(struct libnetapi_ctx *ctx,
-                                       const char **debuglevel)
+                                       char **debuglevel)
 {
        *debuglevel = ctx->debuglevel;
-       return W_ERROR_V(WERR_OK);
+       return NET_API_STATUS_SUCCESS;
 }
 
+/****************************************************************
+****************************************************************/
+
 NET_API_STATUS libnetapi_set_username(struct libnetapi_ctx *ctx,
                                      const char *username)
 {
        TALLOC_FREE(ctx->username);
-       ctx->username = talloc_strdup(ctx, username);
+       ctx->username = talloc_strdup(ctx, username ? username : "");
+
        if (!ctx->username) {
                return W_ERROR_V(WERR_NOMEM);
        }
-       return W_ERROR_V(WERR_OK);
+       return NET_API_STATUS_SUCCESS;
 }
 
 NET_API_STATUS libnetapi_set_password(struct libnetapi_ctx *ctx,
@@ -135,7 +253,7 @@ NET_API_STATUS libnetapi_set_password(struct libnetapi_ctx *ctx,
        if (!ctx->password) {
                return W_ERROR_V(WERR_NOMEM);
        }
-       return W_ERROR_V(WERR_OK);
+       return NET_API_STATUS_SUCCESS;
 }
 
 NET_API_STATUS libnetapi_set_workgroup(struct libnetapi_ctx *ctx,
@@ -146,58 +264,124 @@ NET_API_STATUS libnetapi_set_workgroup(struct libnetapi_ctx *ctx,
        if (!ctx->workgroup) {
                return W_ERROR_V(WERR_NOMEM);
        }
-       return W_ERROR_V(WERR_OK);
-}
-
-const char *libnetapi_errstr(struct libnetapi_ctx *ctx,
-                            NET_API_STATUS status)
-{
-       const char *err_str = NULL;
-
-       switch (status) {
-               case 0:
-                       err_str = "Success";
-                       break;
-               case 0x00000005: /* WERR_ACCESS_DENIED */
-                       err_str = "Access is denied";
-                       break;
-               case 0x00000057: /* WERR_INVALID_PARAM */
-                       err_str = "Invalid parameter";
-                       break;
-               case 0x0000052E: /* WERR_LOGON_FAILURE */
-                       err_str = "Invalid logon credentials";
-                       break;
-               case 0x00000995: /* WERR_DOMAIN_CONTROLLER_NOT_FOUND */
-                       err_str = "A domain controller could not be found";
-                       break;
-               case 0x00000a84: /* WERR_SETUP_NOT_JOINED */
-                       err_str = "Join failed";
-                       break;
-               case 0x00000a83: /* WERR_SETUP_ALREADY_JOINED */
-                       err_str = "Machine is already joined";
-                       break;
-               case 0x00000a85: /* WERR_SETUP_DOMAIN_CONTROLLER */
-                       err_str = "Machine is a Domain Controller";
-                       break;
-               case 0x00000032: /* WERR_NOT_SUPPORTED */
-                       err_str = "Not supported";
-                       break;
-               case 0x0000051f: /* WERR_NO_LOGON_SERVERS */
-                       err_str = "No logon servers found";
-                       break;
-               case 0x00000056: /* WERR_BAD_PASSWORD */
-                       err_str = "A bad password was supplied";
-                       break;
-               case 0x00000520: /* WERR_NO_SUCH_LOGON_SESSION */
-                       err_str = "No such logon session";
-                       break;
-               default:
-                       err_str = talloc_asprintf(ctx, "0x%08x", status);
-                       if (!err_str) {
-                               return NULL;
-                       }
-                       break;
-       }
-
-       return err_str;
+       return NET_API_STATUS_SUCCESS;
+}
+
+/****************************************************************
+****************************************************************/
+
+NET_API_STATUS libnetapi_set_use_kerberos(struct libnetapi_ctx *ctx)
+{
+       ctx->use_kerberos = true;
+       return NET_API_STATUS_SUCCESS;
+}
+
+NET_API_STATUS libnetapi_set_use_ccache(struct libnetapi_ctx *ctx)
+{
+       ctx->use_ccache = true;
+       return NET_API_STATUS_SUCCESS;
+}
+
+/****************************************************************
+Return a libnetapi error as a string, caller must free with NetApiBufferFree
+****************************************************************/
+
+char *libnetapi_errstr(NET_API_STATUS status)
+{
+       TALLOC_CTX *frame = talloc_stackframe();
+       char *ret;
+       if (status & 0xc0000000) {
+               ret = talloc_strdup(NULL, 
+                                    get_friendly_nt_error_msg(NT_STATUS(status)));
+       } else {
+               ret = talloc_strdup(NULL,
+                                   get_friendly_werror_msg(W_ERROR(status)));
+       }
+       TALLOC_FREE(frame);
+       return ret;
+}
+
+/****************************************************************
+****************************************************************/
+
+NET_API_STATUS libnetapi_set_error_string(struct libnetapi_ctx *ctx,
+                                         const char *format, ...)
+{
+       va_list args;
+
+       TALLOC_FREE(ctx->error_string);
+
+       va_start(args, format);
+       ctx->error_string = talloc_vasprintf(ctx, format, args);
+       va_end(args);
+
+       if (!ctx->error_string) {
+               return W_ERROR_V(WERR_NOMEM);
+       }
+       return NET_API_STATUS_SUCCESS;
+}
+
+/****************************************************************
+Return a libnetapi_errstr(), caller must free with NetApiBufferFree
+****************************************************************/
+
+char *libnetapi_get_error_string(struct libnetapi_ctx *ctx,
+                                      NET_API_STATUS status_in)
+{
+       NET_API_STATUS status;
+       struct libnetapi_ctx *tmp_ctx = ctx;
+
+       if (!tmp_ctx) {
+               status = libnetapi_getctx(&tmp_ctx);
+               if (status != 0) {
+                       return NULL;
+               }
+       }
+
+       if (tmp_ctx->error_string) {
+               return talloc_strdup(NULL, tmp_ctx->error_string);
+       }
+
+       return libnetapi_errstr(status_in);
+}
+
+/****************************************************************
+****************************************************************/
+
+NET_API_STATUS NetApiBufferAllocate(uint32_t byte_count,
+                                   void **buffer)
+{
+       void *buf = NULL;
+
+       if (!buffer) {
+               return W_ERROR_V(WERR_INSUFFICIENT_BUFFER);
+       }
+
+       if (byte_count == 0) {
+               goto done;
+       }
+
+       buf = talloc_size(NULL, byte_count);
+       if (!buf) {
+               return W_ERROR_V(WERR_NOMEM);
+       }
+
+ done:
+       *buffer = buf;
+
+       return NET_API_STATUS_SUCCESS;
+}
+
+/****************************************************************
+****************************************************************/
+
+NET_API_STATUS NetApiBufferFree(void *buffer)
+{
+       if (!buffer) {
+               return W_ERROR_V(WERR_INSUFFICIENT_BUFFER);
+       }
+
+       talloc_free(buffer);
+
+       return NET_API_STATUS_SUCCESS;
 }