| /* -*- 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 <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| 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 */ |