rhel - 显式声明 lib 时出现 UnsatisfiedLinkError 异常

标签 rhel java shared-libraries

尝试执行我的 Java 应用程序会产生 UnsatisfiedLinkError找不到 libmysqlclient.so.18 时的异常,即使它在 中显式声明也是如此LD_LIBRARY_PATH , -Djava.library.path /etc/ld.so.conf

问题描述

我正在尝试使用 pcap4j (https://github.com/kaitoy/pcap4j),它是 libpcap 的 Java 包装器,因此我可以从 Java 应用程序嗅探我机器 NIF 上的数据包。
由于 libpcap 需要 super 用户权限才能执行此任务,因此我必须以某种方式赋予执行此应用程序的非特权用户访问 NIF 的能力。

pcap4j 的维护者建议授予功能 CAP_NET_RAWCAP_NET_ADMIN到java命令如下:setcap cap_net_raw,cap_net_admin=eip /path/to/java
由于实现限制,我受到以下限制:

  • 由于安全策略,避免向非特权用户授予 sudo 访问权限。相同的推理可以应用于上述授予
    提到java命令的功能(不知道是否
    每个用户/命令对都授予功能),但是,从我的
    关于安全性的知识相对稀缺,后一种选择看起来
    就像我想要的更划定的权限授予方法
    实现(如果有替代许可,也欢迎解决方案
    授予方法看起来更适合我的目的),并且鉴于
    pcapj4 开发人员,可能更有经验的专业人士,建议
    所以,我遵循了能力授予路径。
  • 用户必须能够在没有密码提示的情况下执行应用程序
  • 权限授予只能进行一次,例如第一次创建用户时。

  • 授予 CAP_NET_RAW 后和 CAP_NET_ADMIN java命令的功能,出现了问题。执行我的应用程序时出现以下异常:
    Error creating entity
    java.lang.UnsatisfiedLinkError: /path/to/app/lib/libxpherejava.so: libmysqlclient.so.18: cannot open shared object file: No such file or directory
            at java.lang.ClassLoader$NativeLibrary.load(Native Method)
            at java.lang.ClassLoader.loadLibrary0(ClassLoader.java:1941)
            at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1857)
            at java.lang.Runtime.loadLibrary0(Runtime.java:870)
            at java.lang.System.loadLibrary(System.java:1122)
    

    细节

    JAVA: java-1.8.0-openjdk-1.8.0.171-8

    操作系统: Linux user-me 3.10.0-862.6.3.el7.x86_64 #1 SMP Fri Jun 15 17:57:37 EDT 2018 x86_64 x86_64 x86_64 GNU/Linux [Red Hat Enterprise Linux Server release 7.5 (Maipo)]

    LD_LIBRARY_PATH 包含未找到库的显式路径:
    LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/usr/lib/lwp:/usr/lib/jvm/java-1.8.0-openjdk-1.8.0.171-8.b10.el7_5.x86_64/jre/lib/amd64/jli/libjli.so:/usr/lib64/mysql/libmysqlclient.so.18
    export LD_LIBRARY_PATH
    

    LD_LIBRARY_PATH 通过使用 传递给 JVM java.library.path :
    exec java
    -XshowSettings:properties 
    -Djava.library.path=${LD_LIBRARY_PATH}
    -d64
    ...
    

    “-XshowSettings:属性”执行 java 命令时为我提供以下输出:
    java.library.path =
            /path/to/app/lib
            /path/to/app/lib/glib-2.0
            /usr/lib/lwp
            /usr/lib/jvm/java-1.8.0-openjdk-1.8.0.171-8.b10.el7_5.x86_64/jre/lib/amd64/jli/libjli.so
            /usr/lib64/mysql/libmysqlclient.so.18
    

    在哪里 /usr/lib64/mysql/libmysqlclient.so.18/usr/lib64/mysql/libmysqlclient.so.18.0.0 的符号链接(symbolic link)

    似乎 JVM(或任何从 libxpherejava.so 请求访问 libmysqlclient.so.18 的实体)找不到 libmysqlclient.so.18,即使它的路径已明确提供给
    java.library.path 和文件确实存在。

    LD_LIBRARY_PATH 包含 libmysqlclient.so.18 (/usr/lib64/mysql/libmysqlclient.so.18) 的路径,发出 libxpherejava.so 的 ldd 导致找不到 libmysqlclient.so.18
    [user@user-me log]$ ldd /path/to/app/lib/libxpherejava.so
            linux-vdso.so.1 =>  (0x00007ffe1a73d000)
            libcrypto.so.10 => /lib64/libcrypto.so.10 (0x00007fd727df4000)
            libssl.so.10 => /lib64/libssl.so.10 (0x00007fd727b83000)
            libxphereS.so => /path/to/app/lib/libxphereS.so (0x00007fd727973000)
            libmysqlclient.so.18 => not found
            libz.so.1 => /lib64/libz.so.1 (0x00007fd72775d000)
            libnsl.so.1 => /lib64/libnsl.so.1 (0x00007fd727543000)
            libpthread.so.0 => /lib64/libpthread.so.0 (0x00007fd727327000)
            libm.so.6 => /lib64/libm.so.6 (0x00007fd727025000)
            libglib-2.0.so.0 => /lib64/libglib-2.0.so.0 (0x00007fd726d11000)
            libc.so.6 => /lib64/libc.so.6 (0x00007fd726944000)
            libdl.so.2 => /lib64/libdl.so.2 (0x00007fd726740000)
            libgssapi_krb5.so.2 => /lib64/libgssapi_krb5.so.2 (0x00007fd7264f3000)
            libkrb5.so.3 => /lib64/libkrb5.so.3 (0x00007fd72620b000)
            libcom_err.so.2 => /lib64/libcom_err.so.2 (0x00007fd726007000)
            libk5crypto.so.3 => /lib64/libk5crypto.so.3 (0x00007fd725dd4000)
            /lib64/ld-linux-x86-64.so.2 (0x00007fd728718000)
            libpcre.so.1 => /lib64/libpcre.so.1 (0x00007fd725b72000)
            libkrb5support.so.0 => /lib64/libkrb5support.so.0 (0x00007fd725964000)
            libkeyutils.so.1 => /lib64/libkeyutils.so.1 (0x00007fd725760000)
            libresolv.so.2 => /lib64/libresolv.so.2 (0x00007fd725547000)
            libselinux.so.1 => /lib64/libselinux.so.1 (0x00007fd725320000)
    

    这是的内容ld.so.conf :
    [user@user-me lib]$ cat /etc/ld.so.conf
    /path/to/app/lib/
    /usr/lib/jvm/java-1.8.0-openjdk-1.8.0.171-8.b10.el7_5.x86_64/jre/lib/amd64/jli/libjli.so
    /usr/lib64/mysql/libmysqlclient.so.18
    

    两个库是 64 位 编译:
    [user@user-me lib]$ file libxpherejava.so
    libxpherejava.so: ELF 64-bit LSB shared object, x86-64, version 1 (SYSV), dynamically linked, BuildID[sha1]=82e1c673e732eb2d3770883b14facf3eff091243, not stripped
    [user@user-me lib]$ file /usr/lib64/mysql/libmysqlclient.so.18
    /usr/lib64/mysql/libmysqlclient.so.18: symbolic link to libmysqlclient.so.18.0.0'
    [user@user-me lib]$ file /usr/lib64/mysql/libmysqlclient.so.18.0.0
    /usr/lib64/mysql/libmysqlclient.so.18.0.0: ELF 64-bit LSB shared object, x86-64, version 1 (SYSV), dynamically linked, BuildID[sha1]=79978c5f4fb259a5a146614e260ea0720dd31d3b, stripped
    

    一个 追踪 在执行“exec java”命令的脚本上产生这个 --> https://dumpz.org/aGHQpNk9Znmk

    相关问题
  • JNI issue on Linux: cannot open shared object file --> 库是 64 位的,LD_LIBRARY_PATH 明确包含有问题的库的路径
  • https://github.com/kaitoy/pcap4j/issues/63 --> pcap4j 开发人员对类似问题的回答(解决方案适用于我的环境。错误已解决,但紧随其后出现当前错误)
  • http://blog.tinola.com/?e=7 --> 发布有关可以解释所有这些调整的能力授予缺陷

  • 有谁知道为什么找不到 libmysqlclient.so.18 ?

    最佳答案

    为有问题的库创建符号链接(symbolic link) libmysqlclient.so.18/usr/lib64 下解决问题。异常消失了。

    关于rhel - 显式声明 lib 时出现 UnsatisfiedLinkError 异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54407724/

    相关文章:

    C++ 可执行文件在运行时找不到库,即使它位于/usr/lib(在 Linux 上)

    linux - (rsyslog) 过滤设施?

    c - 使用 poll 在 C 中进行套接字连接超时

    java - Java 中的点长 - 找到它的定义

    java - 如何使用 mockito 为 Controller 类编写单元测试用例

    java - 从java代码运行xjc

    C++、全局变量和动态共享库

    c++ - 如何制作使用库的静态/动态库

    rpm 需要范围

    linux - REDHAT 环境中 postgresql 的 kerberos 身份验证设置