/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ /* ** File: env.c ** Description: Testing environment variable operations ** */ #include "prenv.h" #include "prmem.h" #include "plgetopt.h" #include #include #include PRIntn debug = 0; PRIntn verbose = 0; PRIntn secure = 0; PRBool failedAlready = PR_FALSE; #define ENVNAME "NSPR_ENVIRONMENT_TEST_VARIABLE" #define ENVVALUE "The expected result" #define ENVBUFSIZE 256 char *envBuf; /* buffer pointer. We leak memory here on purpose! */ static char * NewBuffer( size_t size ) { char *buf = malloc( size ); if ( NULL == buf ) { printf("env: NewBuffer() failed\n"); exit(1); } return(buf); } /* end NewBuffer() */ int main(int argc, char **argv) { char *value; PRStatus rc; { /* Get command line options */ PLOptStatus os; PLOptState *opt = PL_CreateOptState(argc, argv, "vds"); while (PL_OPT_EOL != (os = PL_GetNextOpt(opt))) { if (PL_OPT_BAD == os) { continue; } switch (opt->option) { case 'd': /* debug */ debug = 1; break; case 'v': /* verbose */ verbose = 1; break; case 's': /* secure / set[ug]id */ /* ** To test PR_GetEnvSecure, make this executable (or a ** copy of it) setuid / setgid / otherwise inherently ** privileged (e.g., file capabilities) and run it ** with this flag. */ secure = 1; break; default: break; } } PL_DestroyOptState(opt); } /* end block "Get command line options" */ #if 0 { /* ** This uses Windows native environment manipulation ** as an experiment. Note the separation of namespace! */ BOOL rv; DWORD size; rv = SetEnvironmentVariable( ENVNAME, ENVVALUE ); if ( rv == 0 ) { if (debug) { printf("env: Shit! SetEnvironmentVariable() failed\n"); } failedAlready = PR_TRUE; } if (verbose) { printf("env: SetEnvironmentVariable() worked\n"); } size = GetEnvironmentVariable( ENVNAME, envBuf, ENVBUFSIZE ); if ( size == 0 ) { if (debug) { printf("env: Shit! GetEnvironmentVariable() failed. Found: %s\n", envBuf ); } failedAlready = PR_TRUE; } if (verbose) { printf("env: GetEnvironmentVariable() worked. Found: %s\n", envBuf); } value = PR_GetEnv( ENVNAME ); if ( (NULL == value ) || (strcmp( value, ENVVALUE))) { if (debug) { printf( "env: PR_GetEnv() failed retrieving WinNative. Found: %s\n", value); } failedAlready = PR_TRUE; } if (verbose) { printf("env: PR_GetEnv() worked. Found: %s\n", value); } } #endif /* set an environment variable, read it back */ envBuf = NewBuffer( ENVBUFSIZE ); sprintf( envBuf, ENVNAME "=" ENVVALUE ); rc = PR_SetEnv( envBuf ); if ( PR_FAILURE == rc ) { if (debug) { printf( "env: PR_SetEnv() failed setting\n"); } failedAlready = PR_TRUE; } else { if (verbose) { printf("env: PR_SetEnv() worked.\n"); } } value = PR_GetEnv( ENVNAME ); if ( (NULL == value ) || (strcmp( value, ENVVALUE))) { if (debug) { printf( "env: PR_GetEnv() Failed after setting\n" ); } failedAlready = PR_TRUE; } else { if (verbose) { printf("env: PR_GetEnv() worked after setting it. Found: %s\n", value ); } } if ( secure ) { /* ** In this case we've been run with elevated privileges, so ** test that PR_GetEnvSecure *doesn't* find that env var. */ value = PR_GetEnvSecure( ENVNAME ); if ( NULL != value ) { if (debug) { printf( "env: PR_GetEnvSecure() failed; expected NULL, found \"%s\"\n", value ); } failedAlready = PR_TRUE; } else { if (verbose) { printf("env: PR_GetEnvSecure() worked\n" ); } } } else { /* ** In this case the program is being run normally, so do the ** same check for PR_GetEnvSecure as for PR_GetEnv. */ value = PR_GetEnvSecure( ENVNAME ); if ( (NULL == value ) || (strcmp( value, ENVVALUE))) { if (debug) { printf( "env: PR_GetEnvSecure() Failed after setting\n" ); } failedAlready = PR_TRUE; } else { if (verbose) { printf("env: PR_GetEnvSecure() worked after setting it. Found: %s\n", value ); } } } /* ---------------------------------------------------------------------- */ /* check that PR_DuplicateEnvironment() agrees with PR_GetEnv() */ { #if defined(XP_UNIX) && (!defined(DARWIN) || defined(HAVE_CRT_EXTERNS_H)) static const PRBool expect_failure = PR_FALSE; #else static const PRBool expect_failure = PR_TRUE; #endif char **i, **dupenv = PR_DuplicateEnvironment(); if ( NULL == dupenv ) { if (expect_failure) { if (verbose) printf("env: PR_DuplicateEnvironment failed, " "as expected on this platform.\n"); } else { if (debug) { printf("env: PR_DuplicateEnvironment() failed.\n"); } failedAlready = PR_TRUE; } } else { unsigned found = 0; if (expect_failure) { if (debug) printf("env: PR_DuplicateEnvironment() succeeded, " "but failure is expected on this platform.\n"); failedAlready = PR_TRUE; } else { if (verbose) { printf("env: PR_DuplicateEnvironment() succeeded.\n"); } } for (i = dupenv; *i; i++) { char *equals = strchr(*i, '='); if ( equals == NULL ) { if (debug) printf("env: PR_DuplicateEnvironment() returned a string" " with no '=': %s\n", *i); failedAlready = PR_TRUE; } else { /* We own this string, so we can temporarily alter it */ /* *i is the null-terminated name; equals + 1 is the value */ *equals = '\0'; if ( strcmp(*i, ENVNAME) == 0) { found++; if (verbose) printf("env: PR_DuplicateEnvironment() found " ENVNAME " (%u so far).\n", found); } /* Multiple values for the same name can't happen, according to POSIX. */ value = PR_GetEnv(*i); if ( value == NULL ) { if (debug) printf("env: PR_DuplicateEnvironment() returned a name" " which PR_GetEnv() failed to find: %s\n", *i); failedAlready = PR_TRUE; } else if ( strcmp(equals + 1, value) != 0) { if (debug) printf("env: PR_DuplicateEnvironment() returned the wrong" " value for %s: expected %s; found %s\n", *i, value, equals + 1); failedAlready = PR_TRUE; } else { if (verbose) printf("env: PR_DuplicateEnvironment() agreed with" " PR_GetEnv() about %s\n", *i); } } PR_Free(*i); } PR_Free(dupenv); if (found != 1) { if (debug) printf("env: PR_DuplicateEnvironment() found %u entries for " ENVNAME " (expected 1)\n", found); failedAlready = PR_TRUE; } else { if (verbose) { printf("env: PR_DuplicateEnvironment() found 1 entry for " ENVNAME "\n"); } } } } /* ---------------------------------------------------------------------- */ /* un-set the variable, using RAW name... should not work */ envBuf = NewBuffer( ENVBUFSIZE ); sprintf( envBuf, ENVNAME ); rc = PR_SetEnv( envBuf ); if ( PR_FAILURE == rc ) { if (verbose) { printf( "env: PR_SetEnv() not un-set using RAW name. Good!\n"); } } else { if (debug) { printf("env: PR_SetEnv() un-set using RAW name. Bad!\n" ); } failedAlready = PR_TRUE; } value = PR_GetEnv( ENVNAME ); if ( NULL == value ) { if (debug) { printf("env: PR_GetEnv() after un-set using RAW name. Bad!\n" ); } failedAlready = PR_TRUE; } else { if (verbose) { printf( "env: PR_GetEnv() after RAW un-set found: %s\n", value ); } } /* ---------------------------------------------------------------------- */ /* set it again ... */ envBuf = NewBuffer( ENVBUFSIZE ); sprintf( envBuf, ENVNAME "=" ENVVALUE ); rc = PR_SetEnv( envBuf ); if ( PR_FAILURE == rc ) { if (debug) { printf( "env: PR_SetEnv() failed setting the second time.\n"); } failedAlready = PR_TRUE; } else { if (verbose) { printf("env: PR_SetEnv() worked.\n"); } } /* un-set the variable using the form name= */ envBuf = NewBuffer( ENVBUFSIZE ); sprintf( envBuf, ENVNAME "=" ); rc = PR_SetEnv( envBuf ); if ( PR_FAILURE == rc ) { if (debug) { printf( "env: PR_SetEnv() failed un-setting using name=\n"); } failedAlready = PR_TRUE; } else { if (verbose) { printf("env: PR_SetEnv() un-set using name= worked\n" ); } } value = PR_GetEnv( ENVNAME ); if (( NULL == value ) || ( 0x00 == *value )) { if (verbose) { printf("env: PR_GetEnv() after un-set using name= worked\n" ); } } else { if (debug) { printf( "env: PR_GetEnv() after un-set using name=. Found: %s\n", value ); } failedAlready = PR_TRUE; } /* ---------------------------------------------------------------------- */ /* un-set the variable using the form name= */ envBuf = NewBuffer( ENVBUFSIZE ); sprintf( envBuf, ENVNAME "999=" ); rc = PR_SetEnv( envBuf ); if ( PR_FAILURE == rc ) { if (debug) { printf( "env: PR_SetEnv() failed un-setting using name=\n"); } failedAlready = PR_TRUE; } else { if (verbose) { printf("env: PR_SetEnv() un-set using name= worked\n" ); } } value = PR_GetEnv( ENVNAME "999" ); if (( NULL == value ) || ( 0x00 == *value )) { if (verbose) { printf("env: PR_GetEnv() after un-set using name= worked\n" ); } } else { if (debug) { printf( "env: PR_GetEnv() after un-set using name=. Found: %s\n", value ); } failedAlready = PR_TRUE; } /* ---------------------------------------------------------------------- */ if (debug || verbose) { printf("\n%s\n", (failedAlready)? "FAILED" : "PASSED" ); } return( (failedAlready)? 1 : 0 ); } /* main() */ /* env.c */