From e3e6552ef6e3121c24885c1c46409f3ea94d10b1 Mon Sep 17 00:00:00 2001 From: Ben Steffensmeier Date: Sun, 24 Nov 2024 16:16:41 -0600 Subject: [PATCH] Add configuration for setting program name. --- src/main/c/Include/pyembed.h | 2 +- src/main/c/Jep/maininterpreter.c | 5 ++-- src/main/c/Jep/pyembed.c | 17 +++++++++--- src/main/java/jep/MainInterpreter.java | 4 +-- src/main/java/jep/PyConfig.java | 36 ++++++++++++++++++-------- 5 files changed, 44 insertions(+), 20 deletions(-) diff --git a/src/main/c/Include/pyembed.h b/src/main/c/Include/pyembed.h index 583b09b7..87d34348 100644 --- a/src/main/c/Include/pyembed.h +++ b/src/main/c/Include/pyembed.h @@ -44,7 +44,7 @@ typedef struct __JepThread JepThread; void pyembed_preinit(JNIEnv*, jint, jint, jint, jint, jint, jint, jint, - jstring); + jstring, jstring); void pyembed_startup(JNIEnv*, jobjectArray); void pyembed_shutdown(JavaVM*); void pyembed_shared_import(JNIEnv*, jstring); diff --git a/src/main/c/Jep/maininterpreter.c b/src/main/c/Jep/maininterpreter.c index 6fc6fc76..c10680f8 100644 --- a/src/main/c/Jep/maininterpreter.c +++ b/src/main/c/Jep/maininterpreter.c @@ -47,11 +47,12 @@ JNIEXPORT void JNICALL Java_jep_MainInterpreter_setInitParams jint optimizeFlag, jint dontWriteBytecodeFlag, jint hashRandomizationFlag, - jstring pythonHome) + jstring pythonHome, + jstring programName) { pyembed_preinit(env, noSiteFlag, noUserSiteDirectory, ignoreEnvironmentFlag, verboseFlag, optimizeFlag, dontWriteBytecodeFlag, - hashRandomizationFlag, pythonHome); + hashRandomizationFlag, pythonHome, programName); } /* diff --git a/src/main/c/Jep/pyembed.c b/src/main/c/Jep/pyembed.c index 854af36f..775f6b78 100644 --- a/src/main/c/Jep/pyembed.c +++ b/src/main/c/Jep/pyembed.c @@ -239,7 +239,8 @@ void pyembed_preinit(JNIEnv *env, jint optimizeFlag, jint dontWriteBytecodeFlag, jint hashRandomizationFlag, - jstring pythonHome) + jstring pythonHome, + jstring programName) { if (noSiteFlag >= 0) { Py_NoSiteFlag = noSiteFlag; @@ -270,6 +271,14 @@ void pyembed_preinit(JNIEnv *env, // Python documentation says that the string should not be changed for // the duration of the program so it can never be freed. } + if (programName) { + const char* nameAsUTF = (*env)->GetStringUTFChars(env, programName, NULL); + wchar_t* nameForPython = Py_DecodeLocale(nameAsUTF, NULL); + (*env)->ReleaseStringUTFChars(env, programName, nameAsUTF); + Py_SetProgramName(nameForPython); + // Python documentation says that the string should not be changed for + // the duration of the program so it can never be freed. + } } /* @@ -561,7 +570,7 @@ void pyembed_shared_import(JNIEnv *env, jstring module) intptr_t pyembed_thread_init(JNIEnv *env, jobject cl, jobject caller, jboolean hasSharedModules, jboolean usesubinterpreter, jboolean isolated, jint useMainObmalloc, jint allowFork, - jint allowExec, jint allowThreads, jint allowDaemonThreads, + jint allowExec, jint allowThreads, jint allowDaemonThreads, jint checkMultiInterpExtensions, jint ownGIL) { JepThread *jepThread; @@ -587,8 +596,8 @@ intptr_t pyembed_thread_init(JNIEnv *env, jobject cl, jobject caller, PyEval_AcquireThread(mainThreadState); #if PY_MAJOR_VERSION > 3 || PY_MINOR_VERSION >= 12 - PyInterpreterConfig config; - if (isolated) { + PyInterpreterConfig config; + if (isolated) { config = (PyInterpreterConfig) _PyInterpreterConfig_INIT; } else { config = (PyInterpreterConfig) _PyInterpreterConfig_LEGACY_INIT; diff --git a/src/main/java/jep/MainInterpreter.java b/src/main/java/jep/MainInterpreter.java index 34c6d0f0..7212ffa1 100644 --- a/src/main/java/jep/MainInterpreter.java +++ b/src/main/java/jep/MainInterpreter.java @@ -138,7 +138,7 @@ protected void initialize() throws Error { setInitParams(pyConfig.noSiteFlag, pyConfig.noUserSiteDirectory, pyConfig.ignoreEnvironmentFlag, pyConfig.verboseFlag, pyConfig.optimizeFlag, pyConfig.dontWriteBytecodeFlag, - pyConfig.hashRandomizationFlag, pyConfig.pythonHome); + pyConfig.hashRandomizationFlag, pyConfig.pythonHome, pyConfig.programName); } thread = new Thread("JepMainInterpreter") { @@ -293,7 +293,7 @@ public static void setJepLibraryPath(String path) throws IllegalStateException { private static native void setInitParams(int noSiteFlag, int noUserSiteDiretory, int ignoreEnvironmentFlag, int verboseFlag, int optimizeFlag, int dontWriteBytecodeFlag, - int hashRandomizationFlag, String pythonHome); + int hashRandomizationFlag, String pythonHome, String programName); private static native void initializePython(String[] mainInterpreterArgv); diff --git a/src/main/java/jep/PyConfig.java b/src/main/java/jep/PyConfig.java index 246d129d..93c79a77 100644 --- a/src/main/java/jep/PyConfig.java +++ b/src/main/java/jep/PyConfig.java @@ -36,9 +36,9 @@ public class PyConfig { /* - * -1 is used to indicate not set, in which case we will not set it in - * the native code and the setting will be Python's default. A value of 0 - * or greater will cause the value to be set in the native code. + * -1 is used to indicate not set, in which case we will not set it in the + * native code and the setting will be Python's default. A value of 0 or + * greater will cause the value to be set in the native code. */ protected int noSiteFlag = -1; @@ -57,6 +57,8 @@ public class PyConfig { protected String pythonHome; + protected String programName; + /** * Set the Py_NoSiteFlag variable on the python interpreter. This * corresponds to the python "-S" flag and will prevent the "site" module @@ -155,7 +157,7 @@ public PyConfig setHashRandomizationFlag(int hashRandomizationFlag) { } /** - * Set the home location on the python interpreter. THis is the location of + * Set the home location on the python interpreter. This is the location of * the standard python libraries. This corresponds to the environment * variable PYTHONHOME. * @@ -168,15 +170,27 @@ public PyConfig setPythonHome(String pythonHome) { return this; } + /** + * Set the Py_SetProgramName variable on the python interpreter. This is + * used to initialize executable and in early error messages during python + * initialization. + * + * @param programName + * the name of the executable python program. + * @return a reference to this PyConfig + */ + public PyConfig setProgramName(String programName) { + this.programName = programName; + return this; + } + @Override public String toString() { - return "PyConfig [noSiteFlag=" + noSiteFlag + ", noUserSiteDirectory=" - + noUserSiteDirectory + ", ignoreEnvironmentFlag=" - + ignoreEnvironmentFlag + ", verboseFlag=" + verboseFlag - + ", optimizeFlag=" + optimizeFlag + ", dontWriteBytecodeFlag=" - + dontWriteBytecodeFlag + ", hashRandomizationFlag=" - + hashRandomizationFlag + ", pythonHome=" + pythonHome + "]"; + return "PyConfig [noSiteFlag=" + noSiteFlag + ", noUserSiteDirectory=" + noUserSiteDirectory + + ", ignoreEnvironmentFlag=" + ignoreEnvironmentFlag + ", verboseFlag=" + verboseFlag + + ", optimizeFlag=" + optimizeFlag + ", dontWriteBytecodeFlag=" + dontWriteBytecodeFlag + + ", hashRandomizationFlag=" + hashRandomizationFlag + ", pythonHome=" + pythonHome + ", programName=" + + programName + "]"; } - }