简体   繁体   English

从 Python 内部设置 LD_LIBRARY_PATH

[英]Setting LD_LIBRARY_PATH from inside Python

Is there a way to set specify during runtime where Python looks for shared libraries?有没有办法在运行时指定 Python 查找共享库的位置?

I have fontforge.so located in fontforge_bin and tried the following我有fontforge.so位于fontforge_bin并尝试了以下

os.environ['LD_LIBRARY_PATH']='fontforge_bin'
sys.path.append('fontforge_bin')
import fontforge

and get并得到

ImportError: fontforge_bin/fontforge.so: cannot open shared object file: No such file or directory

Doing ldd on fontforge_bin/fontforge.so gives the followingfontforge_bin/fontforge.so上执行ldd会得到以下结果

linux-vdso.so.1 =>  (0x00007fff2050c000)
libpthread.so.0 => /lib/libpthread.so.0 (0x00007f10ffdef000)
libc.so.6 => /lib/libc.so.6 (0x00007f10ffa6c000)
/lib64/ld-linux-x86-64.so.2 (0x00007f110022d000)

Your script can check for the existence/properness of the environment variable before you import your module, then set it in os.environ if it is missing, and then call os.execv() to restart the python interpreter using the same command line arguments but an updated set of environment variables.您的脚本可以在导入模块之前检查环境变量的存在/正确性,然后在 os.environ 中设置它是否丢失,然后调用os.execv()使用相同的命令行 arguments 重新启动 python 解释器但是一组更新的环境变量。

This is only advisable before any other imports (other than os and sys), because of potential module-import side-effects, like opened file descriptors or sockets, which may be challenging to close cleanly.这仅在任何其他导入(除了 os 和 sys)之前是可取的,因为潜在的模块导入副作用,如打开的文件描述符或 sockets,可能难以干净地关闭。

This code sets LD_LIBRARY_PATH and ORACLE_HOME:此代码设置 LD_LIBRARY_PATH 和 ORACLE_HOME:

#!/usr/bin/python
import os, sys
if 'LD_LIBRARY_PATH' not in os.environ:
    os.environ['LD_LIBRARY_PATH'] = '/usr/lib/oracle/XX.Y/client64/lib'
    os.environ['ORACLE_HOME'] = '/usr/lib/oracle/XX.Y/client64'
    try:
        os.execv(sys.argv[0], sys.argv)
    except Exception, exc:
        print 'Failed re-exec:', exc
        sys.exit(1)
#
# import yourmodule
print 'Success:', os.environ['LD_LIBRARY_PATH']
# your program goes here

It's probably cleaner to set that environment variable as part of the starting environment (in the parent process or systemd/etc job file).将该环境变量设置为启动环境的一部分(在父进程或 systemd/etc 作业文件中)可能更简洁。

...well sort of you could load all libraries from some folder of your choosing via ctypes and thus make them available for you regardless of the LD_LIBRARY_PATH. ...好吧,您可以通过ctypes从您选择的某个文件夹中加载所有库,从而使它们对您可用,而不管LD_LIBRARY_PATH如何。

from ctypes import *
lib1 = cdll.LoadLibrary('/home/username/lib/some_library.so')

or iterate through the files in that dir... you get the idea, once it is loaded it is there for you [if the dependencies are also out of the default path you should load them too...].或遍历该目录中的文件......你明白了,一旦加载它就在那里[如果依赖项也在默认路径之外,你也应该加载它们......]。

LD_LIBRARY_PATH sets the dynamic linker path; LD_LIBRARY_PATH设置动态 linker 路径; that generally can't be changed at runtime, since it's usually cached by the dynamic linker.这通常不能在运行时更改,因为它通常由动态 linker 缓存。

That's not where Python looks for imports , though, including module imports.不过,这不是 Python 寻找导入的地方,包括模块导入。 Changing sys.path is correct.更改sys.path是正确的。

# ls foo/
_csv.so
# python
Python 2.6.6 (r266:84292, Dec 26 2010, 22:31:48)
>>> import sys
>>> sys.path.insert(0, "foo")
>>> import _csv
>>> _csv.__file__
'foo/_csv.so'

(By the way, you may want to ldd the library to see if you have any odd import paths in the library. "ImportError: fontforge_bin/fontforge.so" looks strange.) (顺便说一句,您可能想 ldd 库以查看库中是否有任何奇怪的导入路径。“ImportError: fontforge_bin/fontforge.so”看起来很奇怪。)

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM