bpo-32030: Add _Py_FindEnvConfigValue() (#4963)
Add a new _Py_FindEnvConfigValue() function: code shared between Windows and Unix implementations of _PyPathConfig_Calculate() to read the pyenv.cfg file. _Py_FindEnvConfigValue() now uses _Py_DecodeUTF8_surrogateescape() instead of using a Python Unicode string, the Python API must not be used early during Python initialization. Same change in Unix search_for_exec_prefix(): use _Py_DecodeUTF8_surrogateescape(). Cleanup also encode_current_locale(): PyMem_RawFree/PyMem_Free can be called with NULL. Fix also "NUL byte" => "NULL byte" typo.
This commit is contained in:
parent
9dd762013f
commit
9bee329130
|
@ -19,6 +19,13 @@ PyAPI_FUNC(char*) _Py_EncodeLocaleRaw(
|
||||||
size_t *error_pos);
|
size_t *error_pos);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef Py_BUILD_CORE
|
||||||
|
PyAPI_FUNC(wchar_t*) _Py_DecodeUTF8_surrogateescape(
|
||||||
|
const char *s,
|
||||||
|
Py_ssize_t size,
|
||||||
|
size_t *p_wlen);
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifndef Py_LIMITED_API
|
#ifndef Py_LIMITED_API
|
||||||
PyAPI_FUNC(PyObject *) _Py_device_encoding(int);
|
PyAPI_FUNC(PyObject *) _Py_device_encoding(int);
|
||||||
|
|
||||||
|
|
|
@ -118,6 +118,11 @@ PyAPI_FUNC(wchar_t *) Py_GetPath(void);
|
||||||
#ifdef Py_BUILD_CORE
|
#ifdef Py_BUILD_CORE
|
||||||
PyAPI_FUNC(_PyInitError) _PyPathConfig_Init(const _PyCoreConfig *core_config);
|
PyAPI_FUNC(_PyInitError) _PyPathConfig_Init(const _PyCoreConfig *core_config);
|
||||||
PyAPI_FUNC(PyObject*) _PyPathConfig_ComputeArgv0(int argc, wchar_t **argv);
|
PyAPI_FUNC(PyObject*) _PyPathConfig_ComputeArgv0(int argc, wchar_t **argv);
|
||||||
|
PyAPI_FUNC(int) _Py_FindEnvConfigValue(
|
||||||
|
FILE *env_file,
|
||||||
|
const wchar_t *key,
|
||||||
|
wchar_t *value,
|
||||||
|
size_t value_size);
|
||||||
#endif
|
#endif
|
||||||
PyAPI_FUNC(void) Py_SetPath(const wchar_t *);
|
PyAPI_FUNC(void) Py_SetPath(const wchar_t *);
|
||||||
#ifdef MS_WINDOWS
|
#ifdef MS_WINDOWS
|
||||||
|
|
|
@ -296,62 +296,6 @@ absolutize(wchar_t *path)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* search for a prefix value in an environment file. If found, copy it
|
|
||||||
to the provided buffer, which is expected to be no more than MAXPATHLEN
|
|
||||||
bytes long.
|
|
||||||
*/
|
|
||||||
static int
|
|
||||||
find_env_config_value(FILE * env_file, const wchar_t * key, wchar_t * value)
|
|
||||||
{
|
|
||||||
int result = 0; /* meaning not found */
|
|
||||||
char buffer[MAXPATHLEN*2+1]; /* allow extra for key, '=', etc. */
|
|
||||||
|
|
||||||
fseek(env_file, 0, SEEK_SET);
|
|
||||||
while (!feof(env_file)) {
|
|
||||||
char * p = fgets(buffer, MAXPATHLEN*2, env_file);
|
|
||||||
wchar_t tmpbuffer[MAXPATHLEN*2+1];
|
|
||||||
PyObject * decoded;
|
|
||||||
int n;
|
|
||||||
|
|
||||||
if (p == NULL) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
n = strlen(p);
|
|
||||||
if (p[n - 1] != '\n') {
|
|
||||||
/* line has overflowed - bail */
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (p[0] == '#') {
|
|
||||||
/* Comment - skip */
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
decoded = PyUnicode_DecodeUTF8(buffer, n, "surrogateescape");
|
|
||||||
if (decoded != NULL) {
|
|
||||||
Py_ssize_t k;
|
|
||||||
wchar_t * state;
|
|
||||||
k = PyUnicode_AsWideChar(decoded,
|
|
||||||
tmpbuffer, MAXPATHLEN * 2);
|
|
||||||
Py_DECREF(decoded);
|
|
||||||
if (k >= 0) {
|
|
||||||
wchar_t * tok = wcstok(tmpbuffer, L" \t\r\n", &state);
|
|
||||||
if ((tok != NULL) && !wcscmp(tok, key)) {
|
|
||||||
tok = wcstok(NULL, L" \t", &state);
|
|
||||||
if ((tok != NULL) && !wcscmp(tok, L"=")) {
|
|
||||||
tok = wcstok(NULL, L"\r\n", &state);
|
|
||||||
if (tok != NULL) {
|
|
||||||
wcsncpy(value, tok, MAXPATHLEN);
|
|
||||||
result = 1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* search_for_prefix requires that argv0_path be no more than MAXPATHLEN
|
/* search_for_prefix requires that argv0_path be no more than MAXPATHLEN
|
||||||
bytes long.
|
bytes long.
|
||||||
*/
|
*/
|
||||||
|
@ -501,24 +445,17 @@ search_for_exec_prefix(const _PyCoreConfig *core_config,
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
char buf[MAXPATHLEN+1];
|
char buf[MAXPATHLEN+1];
|
||||||
PyObject *decoded;
|
wchar_t *rel_builddir_path;
|
||||||
wchar_t rel_builddir_path[MAXPATHLEN+1];
|
|
||||||
n = fread(buf, 1, MAXPATHLEN, f);
|
n = fread(buf, 1, MAXPATHLEN, f);
|
||||||
buf[n] = '\0';
|
buf[n] = '\0';
|
||||||
fclose(f);
|
fclose(f);
|
||||||
decoded = PyUnicode_DecodeUTF8(buf, n, "surrogateescape");
|
rel_builddir_path = _Py_DecodeUTF8_surrogateescape(buf, n, NULL);
|
||||||
if (decoded != NULL) {
|
if (rel_builddir_path != NULL) {
|
||||||
Py_ssize_t k;
|
wcsncpy(exec_prefix, calculate->argv0_path, MAXPATHLEN);
|
||||||
k = PyUnicode_AsWideChar(decoded,
|
exec_prefix[MAXPATHLEN] = L'\0';
|
||||||
rel_builddir_path, MAXPATHLEN);
|
joinpath(exec_prefix, rel_builddir_path);
|
||||||
Py_DECREF(decoded);
|
PyMem_RawFree(rel_builddir_path );
|
||||||
if (k >= 0) {
|
return -1;
|
||||||
rel_builddir_path[k] = L'\0';
|
|
||||||
wcsncpy(exec_prefix, calculate->argv0_path, MAXPATHLEN);
|
|
||||||
exec_prefix[MAXPATHLEN] = L'\0';
|
|
||||||
joinpath(exec_prefix, rel_builddir_path);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -784,7 +721,7 @@ calculate_read_pyenv(PyCalculatePath *calculate)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Look for a 'home' variable and set argv0_path to it, if found */
|
/* Look for a 'home' variable and set argv0_path to it, if found */
|
||||||
if (find_env_config_value(env_file, L"home", tmpbuffer)) {
|
if (_Py_FindEnvConfigValue(env_file, L"home", tmpbuffer, MAXPATHLEN)) {
|
||||||
wcscpy(calculate->argv0_path, tmpbuffer);
|
wcscpy(calculate->argv0_path, tmpbuffer);
|
||||||
}
|
}
|
||||||
fclose(env_file);
|
fclose(env_file);
|
||||||
|
|
|
@ -564,58 +564,6 @@ done:
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int
|
|
||||||
find_env_config_value(FILE * env_file, const wchar_t * key, wchar_t * value)
|
|
||||||
{
|
|
||||||
int result = 0; /* meaning not found */
|
|
||||||
char buffer[MAXPATHLEN*2+1]; /* allow extra for key, '=', etc. */
|
|
||||||
|
|
||||||
fseek(env_file, 0, SEEK_SET);
|
|
||||||
while (!feof(env_file)) {
|
|
||||||
char * p = fgets(buffer, MAXPATHLEN*2, env_file);
|
|
||||||
wchar_t tmpbuffer[MAXPATHLEN*2+1];
|
|
||||||
PyObject * decoded;
|
|
||||||
size_t n;
|
|
||||||
|
|
||||||
if (p == NULL) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
n = strlen(p);
|
|
||||||
if (p[n - 1] != '\n') {
|
|
||||||
/* line has overflowed - bail */
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (p[0] == '#') {
|
|
||||||
/* Comment - skip */
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
decoded = PyUnicode_DecodeUTF8(buffer, n, "surrogateescape");
|
|
||||||
if (decoded != NULL) {
|
|
||||||
Py_ssize_t k;
|
|
||||||
k = PyUnicode_AsWideChar(decoded,
|
|
||||||
tmpbuffer, MAXPATHLEN * 2);
|
|
||||||
Py_DECREF(decoded);
|
|
||||||
if (k >= 0) {
|
|
||||||
wchar_t * context = NULL;
|
|
||||||
wchar_t * tok = wcstok_s(tmpbuffer, L" \t\r\n", &context);
|
|
||||||
if ((tok != NULL) && !wcscmp(tok, key)) {
|
|
||||||
tok = wcstok_s(NULL, L" \t", &context);
|
|
||||||
if ((tok != NULL) && !wcscmp(tok, L"=")) {
|
|
||||||
tok = wcstok_s(NULL, L"\r\n", &context);
|
|
||||||
if (tok != NULL) {
|
|
||||||
wcsncpy(value, tok, MAXPATHLEN);
|
|
||||||
result = 1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
read_pth_file(_PyPathConfig *config, wchar_t *prefix, const wchar_t *path,
|
read_pth_file(_PyPathConfig *config, wchar_t *prefix, const wchar_t *path,
|
||||||
int *isolated, int *nosite)
|
int *isolated, int *nosite)
|
||||||
|
@ -765,9 +713,11 @@ calculate_pyvenv_file(PyCalculatePath *calculate)
|
||||||
FILE *env_file = _Py_wfopen(envbuffer, L"r");
|
FILE *env_file = _Py_wfopen(envbuffer, L"r");
|
||||||
if (env_file == NULL) {
|
if (env_file == NULL) {
|
||||||
errno = 0;
|
errno = 0;
|
||||||
|
|
||||||
reduce(envbuffer);
|
reduce(envbuffer);
|
||||||
reduce(envbuffer);
|
reduce(envbuffer);
|
||||||
join(envbuffer, env_cfg);
|
join(envbuffer, env_cfg);
|
||||||
|
|
||||||
env_file = _Py_wfopen(envbuffer, L"r");
|
env_file = _Py_wfopen(envbuffer, L"r");
|
||||||
if (env_file == NULL) {
|
if (env_file == NULL) {
|
||||||
errno = 0;
|
errno = 0;
|
||||||
|
@ -780,7 +730,7 @@ calculate_pyvenv_file(PyCalculatePath *calculate)
|
||||||
|
|
||||||
/* Look for a 'home' variable and set argv0_path to it, if found */
|
/* Look for a 'home' variable and set argv0_path to it, if found */
|
||||||
wchar_t tmpbuffer[MAXPATHLEN+1];
|
wchar_t tmpbuffer[MAXPATHLEN+1];
|
||||||
if (find_env_config_value(env_file, L"home", tmpbuffer)) {
|
if (_Py_FindEnvConfigValue(env_file, L"home", tmpbuffer, MAXPATHLEN)) {
|
||||||
wcscpy_s(calculate->argv0_path, MAXPATHLEN+1, tmpbuffer);
|
wcscpy_s(calculate->argv0_path, MAXPATHLEN+1, tmpbuffer);
|
||||||
}
|
}
|
||||||
fclose(env_file);
|
fclose(env_file);
|
||||||
|
|
|
@ -20,8 +20,6 @@ extern int winerror_to_errno(int);
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#endif /* HAVE_FCNTL_H */
|
#endif /* HAVE_FCNTL_H */
|
||||||
|
|
||||||
extern wchar_t* _Py_DecodeUTF8_surrogateescape(const char *s, Py_ssize_t size,
|
|
||||||
size_t *p_wlen);
|
|
||||||
extern char* _Py_EncodeUTF8_surrogateescape(const wchar_t *text,
|
extern char* _Py_EncodeUTF8_surrogateescape(const wchar_t *text,
|
||||||
size_t *error_pos, int raw_malloc);
|
size_t *error_pos, int raw_malloc);
|
||||||
|
|
||||||
|
@ -194,7 +192,7 @@ encode_ascii_surrogateescape(const wchar_t *text, size_t *error_pos, int raw_mal
|
||||||
|
|
||||||
len = wcslen(text);
|
len = wcslen(text);
|
||||||
|
|
||||||
/* +1 for NUL byte */
|
/* +1 for NULL byte */
|
||||||
if (raw_malloc) {
|
if (raw_malloc) {
|
||||||
result = PyMem_RawMalloc(len + 1);
|
result = PyMem_RawMalloc(len + 1);
|
||||||
}
|
}
|
||||||
|
@ -467,13 +465,11 @@ encode_current_locale(const wchar_t *text, size_t *error_pos, int raw_malloc)
|
||||||
else
|
else
|
||||||
converted = wcstombs(NULL, buf, 0);
|
converted = wcstombs(NULL, buf, 0);
|
||||||
if (converted == (size_t)-1) {
|
if (converted == (size_t)-1) {
|
||||||
if (result != NULL) {
|
if (raw_malloc) {
|
||||||
if (raw_malloc) {
|
PyMem_RawFree(result);
|
||||||
PyMem_RawFree(result);
|
}
|
||||||
}
|
else {
|
||||||
else {
|
PyMem_Free(result);
|
||||||
PyMem_Free(result);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (error_pos != NULL)
|
if (error_pos != NULL)
|
||||||
*error_pos = i;
|
*error_pos = i;
|
||||||
|
|
|
@ -354,6 +354,56 @@ _PyPathConfig_ComputeArgv0(int argc, wchar_t **argv)
|
||||||
return PyUnicode_FromWideChar(argv0, n);
|
return PyUnicode_FromWideChar(argv0, n);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Search for a prefix value in an environment file (pyvenv.cfg).
|
||||||
|
If found, copy it into the provided buffer. */
|
||||||
|
int
|
||||||
|
_Py_FindEnvConfigValue(FILE *env_file, const wchar_t *key,
|
||||||
|
wchar_t *value, size_t value_size)
|
||||||
|
{
|
||||||
|
int result = 0; /* meaning not found */
|
||||||
|
char buffer[MAXPATHLEN*2+1]; /* allow extra for key, '=', etc. */
|
||||||
|
|
||||||
|
fseek(env_file, 0, SEEK_SET);
|
||||||
|
while (!feof(env_file)) {
|
||||||
|
char * p = fgets(buffer, MAXPATHLEN*2, env_file);
|
||||||
|
wchar_t *tmpbuffer;
|
||||||
|
int n;
|
||||||
|
|
||||||
|
if (p == NULL) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
n = strlen(p);
|
||||||
|
if (p[n - 1] != '\n') {
|
||||||
|
/* line has overflowed - bail */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (p[0] == '#') {
|
||||||
|
/* Comment - skip */
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
tmpbuffer = _Py_DecodeUTF8_surrogateescape(buffer, n, NULL);
|
||||||
|
if (tmpbuffer != NULL) {
|
||||||
|
wchar_t * state;
|
||||||
|
wchar_t * tok = wcstok(tmpbuffer, L" \t\r\n", &state);
|
||||||
|
if ((tok != NULL) && !wcscmp(tok, key)) {
|
||||||
|
tok = wcstok(NULL, L" \t", &state);
|
||||||
|
if ((tok != NULL) && !wcscmp(tok, L"=")) {
|
||||||
|
tok = wcstok(NULL, L"\r\n", &state);
|
||||||
|
if (tok != NULL) {
|
||||||
|
wcsncpy(value, tok, MAXPATHLEN);
|
||||||
|
result = 1;
|
||||||
|
PyMem_RawFree(tmpbuffer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
PyMem_RawFree(tmpbuffer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Reference in New Issue