首页 > 编程语言 >解决php中通过exec调用python脚本报ModuleNotFoundError错误

解决php中通过exec调用python脚本报ModuleNotFoundError错误

时间:2023-07-17 10:23:49浏览次数:76  
标签:www exec python 用户 dotenv ModuleNotFoundError data

背景

出于某些原因,我们有时会在PHP中通过exec来调用Python代码,有可能是某些功能只能用Python实现(或用Python实现比较方便),有可能是出于性能考虑(Python可以执行耗时任务)。

但我们有时会发现,在控制台用命令行的方式运行python脚本一切正常,在 php 中用 exec 调用就报 ModuleNotFoundError: No module named 'xxx' 错误。

本文是在 Ubuntu 20.04 上以 ubuntu 用户身份进行的测试。

错误原因

用户不同

这种错误一般都是因为执行脚本的用户不同导致的,php用exec调用python脚本时,使用的用户一般是 www-data,而我们在控制台一般都是 rootubuntu 用户。

这个可以通过 whoami 命令来验证。

php代码如下:

$pythonScript = "whoami";
Log::info("exec script:" . $pythonScript);
exec($pythonScript, $output, $returnValue);
Log::info("exec output:" . json_encode($output));
Log::info("exec returnValue:" . $returnValue);

输出如下:

[2023-07-13 10:34:27] local.INFO: exec script:whoami  
[2023-07-13 10:34:27] local.INFO: exec output:["www-data"]  
[2023-07-13 10:34:27] local.INFO: exec returnValue:0  

为什么用户不同就会导致 ModuleNotFoundError: No module named 'xxx' 这个错误呢,根本原因还是权限问题。

权限问题

我们在控制台写python脚本时,一般会通过 pip[3] install [xxx]的形式安装依赖的包,这时包一般会安装在用户目录。

下面做个测试,我们安装 python-dotenv这个包,然后查看包的安装位置:

可以看到这个包安装在了 /home/ubuntu/.local/lib/python3.8/site-packages 这个目录。

下面我们试一下用 www-data 用户的身份是否有权限调用。

Python测试代码:

from dotenv import load_dotenv

load_dotenv()

分别用当前用户和www-data调用:

可以看到用www-data调用时果然报ModuleNotFoundError: No module named 'dotenv'错误。

我们查看一下我们安装的python-dotenvwww-data用户是否可用:

sudo -u www-data pip3 show python-dotenv

可以看到确实是没有的。

即然原因确定了,接下来就好办了。

解决方案

方案一:修改web服务器用户

即然是控制台用户可以运行脚本,我们把Web服务器用户改为控制台用户就可以了,以 apache 为例具体步骤如下:

1.打开apache配置文件:sudo vim /etc/apache2/apache2.conf

2.更改以下两行,将运行的用户和组设置为自己所需的:

User ${APACHE_RUN_USER}
Group ${APACHE_RUN_GROUP}
#更改为
User ubuntu
Group ubuntu

3.重启apache:sudo service apache2 restart

注:这种方案能解决问题,但并不好,因为权限给的太大了,有很大的安全风险,不建议用。

方案二:给 www-data 用户安装python依赖库

在安装之前我们确认一下www-data用户是否没有安装python-dotenv包:

sudo -u www-data pip3 show python-dotenv

下面我们给 www-data 用户安装python-dotenv包:

#安装
sudo -u www-data pip3 install python-dotenv

#显示安装路径
sudo -u www-data pip3 show python-dotenv

我们可以看到,安装到了 /var/www/.local/lib/python3.8/site-packages目录下。

我们来验证一下:

sudo -u www-data python3 pyscripts/test.py

可以看到不报错了。

大家还有别的方案吗?欢迎留言讨论。

标签:www,exec,python,用户,dotenv,ModuleNotFoundError,data
From: https://www.cnblogs.com/hotaigc/p/php-exec-python-ModuleNotFoundError.html

相关文章

  • python怎么读取txt文件
    1.简单的将文件读取到字符串中f=open("data.txt","r")#设置文件对象str=f.read()#将txt文件的所有内容读入到字符串str中f.close()#将文件关闭2.按行读取整个文件第一种方法f=open("data.txt","r")#设置文件对象line=f.readline()line=line[:-1]wh......
  • Dubbo接口+python的接口测试举例(用Python自带的telnetlib库进行dubbo测试)
      最近公司开发新的一套系统,开发出来的方案会基于dubbo分布式服务框架开发的,那么什么是dubbo,身为测试的我,第一眼看到这个,我得去了解了解dubbo是啥玩意,为开展的测试工作做准备,提前先学dubbo的相关知识。 1、什么是dubbo   Dubbo来源于阿里巴巴集团某个团队研发出来的一个......
  • python利用小列表中元素排序对整个大列表中的小列表进行排序
    一、了解sorted() 函数sorted()函数是Python内置的用于排序可迭代对象的函数,它可以接受多个参数来进行灵活的排序操作。下面是对sorted()函数的参数要求和使用方法的详细说明:参数列表:iterable(必需):表示要进行排序的可迭代对象,例如列表、元组、集合等。key(可选):指定一个函数......
  • 常用语言的线程模型(Java、go、C++、python3)
    背景知识软件是如何驱动硬件的?硬件是需要相关的驱动程序才能执行,而驱动程序是安装在操作系统内核中。如果写了一个程序A,A程序想操作硬件工作,首先需要进行系统调用,由内核去找对应的驱动程序驱使硬件工作。而驱动程序怎么让硬件工作的呢?驱动程序作为硬件和操作系统之间的媒介,可以......
  • Python | requests库
    一、基本概念1、简介requests模块是python基于urllib,采用Apache2Licensed开源协议的HTTP库。它比urllib更加方便,可以节约我们大量的工作,完全满足HTTP测试需求。Requests的哲学是以PEP20的习语为中心开发的,所以它比urllib更加Pythoner。2、获取通过pip......
  • subprocess Python执行系统命令最优选模块
    简介subprocess是Python中执行操作系统级别的命令的模块,所谓系级级别的命令就是如ls/etc/userifconfig等和操作系统有关的命令。subprocess创建子进程来执行相关命令,并连接它们的输入、输出和错误管道,获取它们的返回状态。subprocess来源Subprocess模块开发之前,标准......
  • Python报错 | 关于requests.exceptions.SSLError解决方案
    学习爬虫遇到的错误。报错信息:requests.exceptions.SSLError:HTTPSConnectionPool(host=’*****’,port=443):Maxretriesexceededwithurl:/(CausedbySSLError(“Can’tconnecttoHTTPSURLbecausetheSSLmoduleisnotavailable.”))解决方法:检查是否已安装......
  • python: xmlhelper
     xml:<?xmlversion="1.0"?><data><countryname="Liechtenstein"><rank>1</rank><year>2008</year><gdppc>141100</gdppc><neighborname=&quo......
  • python - 串口通讯
    1.安装pyserialpip3installpyserial2.使用方式config.pyimportserialport="COM1"baudrate=2400bytesize=serial.SEVENBITSstopbits=serial.STOPBITS_TWOparity=serial.PARITY_NONEtimeout=10main.pyimportserialimportconfigser=s......
  • [译]使用Python和Dash 创建一个仪表盘(上)
    介绍在数据科学和分析的领域,数据能力的释放不仅是通过提取见解的方式,同时也要能通过有效的方式来传达见解.这就是数据可视化发挥见解的地方.数据可视化是信息和数据的可视化呈现.它使用可视化元素,如图表、图形、地图,使其更容易看懂原始数据中的模式、趋势及异常值.对于数......