背景

出于某些原因,我们有时会在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

可以看到不报错了。

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

解决php中通过exec调用python脚本报ModuleNotFoundError错误的更多相关文章

  1. 在go中通过cmd调用python命令行参数量级过大问题解决

    问题描述如下: 在go中使用cmd调用python命令行 cmd := exec.Command("python", "dimine/Kriging/matrix.py& ...

  2. 运行python脚本报错:selenium.common.exceptions.SessionNotCreatedException: Message: session not created

    运行python脚本报错:selenium.common.exceptions.SessionNotCreatedException: Message: session not created 原因: ...

  3. QT中 使用c++调用python

    最近在做一个项目,开发环境用的是QT c++.项目中使用amazon云服务,调研发现有一个Amazon云的python接口.就有了标题中的问题,需要用C++来调用python脚本. 下面是一个c++调 ...

  4. Windows下运行python脚本报错“ImportError: No Module named ...”的解决方法

    之前遇到一个问题,在Pycharm或IPython之类的IDE上运行脚本正常,但是直接运行或cmd命令行运行的时候报了模块未能找到的错误--ImportError: No Module named . ...

  5. Python脚本报错AttributeError: ‘module’ object has no attribute’xxx’解决方法

    最近在编写Python脚本过程中遇到一个问题比较奇怪:Python脚本完全正常没问题,但执行总报错"AttributeError: 'module' object has no attrib ...

  6. 运行python脚本报错SyntaxError: (unicode error) 'unicodeescape' codec can't decode bytes in position 2-3: truncated \UXXXXXXXX escape

    运行python脚本报错 SyntaxError: (unicode error) 'unicodeescape' codec can't decode bytes  in position 2-3: ...

  7. j解决sparkr中使用某些r的原生函数 发生错误Error: class(objId) == "jobj" is not TRUE的问题

    Create table function in Spark in R not working João_Andre  (3) 询问的问题 | 2016年12月10日 06:03BLUEMIXRSPA ...

  8. Python脚本报错AttributeError: 'module' object has no attribute 'maketrans'

    出现此错误的原因:是此文件smtp02.py 所在的目录下有string.pyc 的文件存在,与python库里的string.pyc冲突造成无法确认编译所取的类库.

  9. 头疼的Python 脚本报错

    Python 脚本报错 检查是否用了制表符.变量声明前面不能用制表符,只能用空格,版本为2.7.14

  10. 在Java中调用Python

    写在前面 在微服务架构大行其道的今天,对于将程序进行嵌套调用的做法其实并不可取,甚至显得有些愚蠢.当然,之所以要面对这个问题,或许是因为一些历史原因,或者仅仅是为了简单.恰好我在项目中就遇到了这个问题 ...

随机推荐

  1. 计算机常用的快捷键以及常用的Dos命令

    计算机的常用快捷键有哪些? 今天我重温了Java基础的课程,计算机的快捷键大家肯定不陌生.   计算机的常用快捷键 ctrl+c 复制 ctrl+v 粘贴 ctrl+s 保存 ctrl+x 剪切 ct ...

  2. Junit启动测试mybatis xml文件BindingException: Invalid bound statement问题

    背景:1.正常启动,xml文件放在java目录和resource目录下均正常 2.junit启动,xml文件放在resource目录下正常,放在java目录下报BindingException错误 m ...

  3. [Pytorch框架] PyTorch 中文手册

    PyTorch 中文手册 书籍介绍 这是一本开源的书籍,目标是帮助那些希望和使用PyTorch进行深度学习开发和研究的朋友快速入门. 由于本人水平有限,在写此教程的时候参考了一些网上的资料,在这里对他 ...

  4. [OpenCV-Python] 9 图像的基础操作

    文章目录 OpenCV-Python: 核心操作 9 图像的基础操作 9.1 获取并修改像素值 9.2 获取图像属性 9.3 图像 ROI 9.4 拆分及合并图像通道 9.5 为图像扩边(填充) Op ...

  5. ArcGIS Desktop发布地形高程服务(DEM/DSM)

    在做ArcGIS三维时,地形服务的发布与普通地图服务的发布不一样,需要发布成ImageServer,切片格式选择LERC. 本文示例使用软件: ArcGIS Desktop10.3.1 注:ArcGI ...

  6. P1008 [NOIP1998 普及组] 三连击,置顶题解的问题

    题目链接: https://www.luogu.com.cn/problem/P1008 置顶题解 暴力,加简化的判断,数学原理,2个集合内所有数相加相乘结果一样,2个集合的内容一样(没错我自己编得, ...

  7. 2020-11-20:java中,听说过CMS的并发预处理和并发可中断预处理吗?

    福哥答案2020-11-20:[答案来自此链接:](http://bbs.xiangxueketang.cn/question/391)1.首先,CMS是一个关注停顿时间,以回收停顿时间最短为目标的垃 ...

  8. 2021-03-28:定义一种数:可以表示成若干(数量>1)连续正数和的数 。比如:5 = 2+3,5就是这样的数 ;12 = 3+4+5,12就是这样的数 。1不是这样的数,因为要求数量大于1个、连续正数和 。2 = 1 + 1,2也不是,因为等号右边不是连续正数 。给定一个参数N,返回是不是可以表示成若干连续正数和的数 。

    2021-03-28:定义一种数:可以表示成若干(数量>1)连续正数和的数 .比如:5 = 2+3,5就是这样的数 :12 = 3+4+5,12就是这样的数 .1不是这样的数,因为要求数量大于1 ...

  9. 2022-02-22:机器人大冒险。 力扣团队买了一个可编程机器人,机器人初始位置在原点(0, 0)。小伙伴事先给机器人输入一串指令command,机器人就会无限循环这条指令的步骤进行移动。指令有两种

    2022-02-22:机器人大冒险. 力扣团队买了一个可编程机器人,机器人初始位置在原点(0, 0).小伙伴事先给机器人输入一串指令command,机器人就会无限循环这条指令的步骤进行移动.指令有两种 ...

  10. transaction.atomic装饰器

    from django.shortcuts import renderfrom django.http import HttpResponsefrom django.views.generic imp ...