1. 问题

在自动化应用的时候 ,有时候环境变量与运行需要不一致。这时候有两种选择:

  1. 改变节点环境变量,使得其和运行需求保持一致;
  2. 在自动化脚本中设置环境变量,其范围只在脚本运行环境中有效。

显然,当需要运行多个自动化脚本,每个的需求不一致的时候,选择2是更好的设计。

2. 解决方案和案例

假设我们有两个java的运行版本,一个是本节点上的java,版本7;另一个是云盘上共享的版本8。我们需要运行elasticsearch 6.2.3,它需要java8。而我们不希望更新节点上的java:

  • java 8路径:/shared/java8/bin
  • elasticsearch 6.2.3路径:/opt/elasticsearch6.2.3/bin
  •  elasticsearch用户:elasticsearch
import subprocess, re, os, socket, resource, pwd, crypt, logging, sys, platform

def demote(user_uid, user_gid):
def result():
os.setgid(user_gid)
os.setuid(user_uid)
return result class setuputil:
@staticmethod
def run_command(args, username=None, usercwd=None):
try:
if username:
pw_record = pwd.getpwnam(username)
env = os.environ.copy()
env['HOME'] = pw_record.pw_dir
env['LOGNAME'] = username
env['USER'] = username
if usercwd:
env['PWD'] = usercwd
p = subprocess.Popen(args, preexec_fn=demote(pw_record.pw_uid, pw_record.pw_gid),
cwd=usercwd, env=env, close_fds=True)
else:
p = subprocess.Popen(args, stdout=subprocess.PIPE, stderr=subprocess.PIPE, close_fds = True) outputs = p.communicate()
for out in outputs:
if out:
return (p.returncode, out)
except OSError as os_e:
return (os_e.errno, os_e.strerror)
except ValueError as val_e:
return (val_e.errno, val_e.strerror)
else:
return (p.returncode, outputs) @staticmethod
def java_version():
java_ver_pattern = '\"(\d+\.\d+).*\"'
java_cmd = "java"
if 'JAVA_HOME' in os.environ:
java_cmd = os.environ['JAVA_HOME'] + "/bin/java"
version = subprocess.check_output([java_cmd, '-version'], stderr=subprocess.STDOUT)
if version:
return re.search(java_ver_pattern, version).groups()[0]
else:
return None @staticmethod
def start_app(es_exec, pid_file, es_user):
rtcode, msg = setuputil.run_command([es_exec, '-d', '-p', pid_file], es_user)
if rtcode:
return (rtcode, msg)
else:
return (0, "OK") MIN_JAVA_VERSION = 1.8

ELASTIC_USER = 'elastic'
ELASTIC_PASSWORD = 'elastic'
ELASTIC_PID = '/tmp/elasticpid' def main():
# set up java home
os.environ['JAVA_HOME'] = '/shared/java8' j_ver = setuputil.java_version()
if not j_ver or float(j_ver) < MIN_JAVA_VERSION:
logging.error("Java isn't installed or java version is not up to date: %s", j_ver if j_ver else "None")
return (1, "Java isn't installed or java version is not up to date: %s"%(j_ver if j_ver else "None")) rtcode, msg = setuputil.start_app('/opt/elasticsearch6.2.3/bin/elasticsearch', ELASTIC_PID, ELASTIC_USER)
if rtcode:
logging.error("Failed to start elasticsearch: %d, %s", rtcode, msg)
return (rtcode, msg)
else:
logging.debug("Start elasticsearch: %s", msg)
return (0, None) if __name__ == '__main__':
main()

3. 注意的问题

有几个需要注意的问题:

  • 环境变量的适用范围(scope)

    • 一般只在运行的当前shell有效
    • 就上例而言,JAVA_HOME只在运行脚本的进程里是有效
  • 在哪里设置环境变量?
    • 需要在运行命令之前设置  
    • 就上例而言,JAVA_HOME需要在运行java版本检查,和运行elasticsearch之前设置

  

在python脚本中设置环境变量,并运行相关应用的更多相关文章

  1. python 程序中设置环境变量

    python 中调用系统命令有三种方法: 1.os.system('command') ,这个方法是直接调用标准C的system() 函数,仅仅在一个子终端运行系统命令,而不能获取命令执行后的返回信息 ...

  2. ant脚本中设置环境变量

    http://blog.csdn.net/quqi99/article/details/5329841

  3. 怎样在Ubuntu中设置环境变量

    首先启动终端. 单击屏幕左上角的Ubuntu图标,在弹出的窗口中点击搜索栏,输入"terminal", 稍等片刻,终端就会赫然在目!二话不说,直接点击!     然后打开环境设置文 ...

  4. Python IDLE arcpy设置环境变量

    在IDLE中 import arcpy help(arcpy) 得到的路径为: 但是在arcmap中,路径为: 说明IDLE的环境变量设置有问题: 在windows的环境变量中设置环境变量PYTHON ...

  5. Ubuntu中设置环境变量详解

    1, 为单一用户:.bashrc: 为每一个运行bash shell的用户执行此文件.当bash shell被打开时,该文件被读取.打开用户主目录下的.bashrc,在这个文件中加入export PA ...

  6. dockerfile中设置环境变量

    设置环境变量要用 ENV ENV JAVA_HOME /usr/local/jdk1.8.0_171 ENV PATH $JAVA_HOME/bin:$PATH 用 echo "JAVA_H ...

  7. Mac笔记本中是用Idea开发工具在Java项目中调用python脚本遇到的环境变量问题解决

    问题描述: mac笔记本本身会自带几个python版本,比如python2.7版本,我没有改动mac默认的python版本,只是安装了python3.7版本. 使用Pycharm开发Python项目没 ...

  8. 如何在pycharm中设置环境变量

    今天运行tensorflow的时候,发现在pycharm下,程序无法找到CUDA的libcupti.so文件.而在添加完环境变量: export LD_LIBRARY_PATH=$LD_LIBRARY ...

  9. MySQL在Windows中设置环境变量

    在桌面选择“计算机”的图标(或者是我的电脑),右键-->属性-->点击“高级系统设置” 然后选择 高级  选项点击环境变量 然后点击新建-> 变量名为MYSQL_HOME 变量值为m ...

随机推荐

  1. 根据需求设计类并且画UML类图练习

    题目如下: // 打车时,可以打专车或者快车.任何车都有车牌号和名称// 不同车价格不同,快车每公里1元,专车每公里2元// 行程开始时,显示车辆信息// 行程结束时,显示打车金额(假定行程就5公里) ...

  2. Spark系列-核心概念

    Spark系列-初体验(数据准备篇) Spark系列-核心概念 一. Spark核心概念 Master,也就是架构图中的Cluster Manager.Spark的Master和Workder节点分别 ...

  3. jenkins ansible

    ansible-playbook - hosts: test remote_user: root vars: warpath: "/root/.jenkins/workspace/172.1 ...

  4. filebeat+logstash+elasticsearch收集haproxy日志

    filebeat用于是日志收集,感觉和 flume相同,但是用go开发,性能比较好 在2.4版本中, 客户机部署logstash收集匹配日志,传输到 kafka,在用logstash 从消息队列中抓取 ...

  5. git checkout -b

    创建分支: $ git branch mybranch切换分支: $ git checkout mybranch创建并切换分支: $ git checkout -b mybranch 更新master ...

  6. Kafka设计解析(四)Kafka Consumer设计解析

    转载自 技术世界,原文链接 Kafka设计解析(四)- Kafka Consumer设计解析 目录 一.High Level Consumer 1. Consumer Group 2. High Le ...

  7. 关于ISP、IAP、DFU和bootloader

    这是嵌入式开发中常用的几个专业术语,其诞生的背景和其具体作用大概如下   在很久很久以前,那是8051单片机流行的时代,做单片机开发都需要一个专用工具,就是单片机的编程器,或者叫烧写器.说“烧”写一点 ...

  8. CSS+JQuery实现Tabs效果,点击更改背景色(不含图片)

    1,Html代码 <body> <div id="box"> <ul id="tab_nav"> <li class= ...

  9. SQLAlchemy Table(表)类方式 - Table类和Column类

    Table 构造方法 Table(name, metadata[, *column_list][, **kwargs]) 参数说明: name 表名 metadata 元数据对象 column_lis ...

  10. 【10.21总结】一个渗透测试练习实例——发现未知的漏洞(Race condition)

    Write-up地址:Exploiting an unknown vulnerability 作者:Abhishek Bundela 这篇文章跟我之前看到的文章不太一样,作者是按照一个练习的方式简单描 ...