如何在unix系统中用别的用户运行一个程序?

1、问题的缘由

实际开发系统的时候,经常需要用别的用户运行一个程序。比如,有些系统为保证系统安全,不允许使用root来运行。这里,我们总结了unix系统下如何解决这个问题的一些方法。同时,我们还讨论如何在python脚本里实现。

2、unix系统的方法

2.1、runuser

runuser允许使用替代用户和组ID来运行命令。如果选项-u没有给出,则回退到su兼容的语义和shell被执行。

  • runuser和su命令之间的区别在于runuser不要求输入密码(因为它可能仅由root用户执行),它使用不同的PAM配置。命令runuser不必与suid权限一起安装。
  • 当没有参数调用时,runuser默认以root用户身份运行交互式shell。
  • 为了向后兼容,runuser默认不更改当前目录并仅设置环境变量HOME和SHELL(加上USER和LOG- 如果目标用户不是root用户,则为NAME)。此版本的runuser使用PAM进行会话管理

用法:

runuser [options] -u <USER> COMMAND
runuser [options] [-] [USER [arg]...]

案例:

 runuser -l elastic -c "ls /var/opt/elastic"

如果当前用户不是root:

runuser: may not be used by non-root users

2.2、su

su允许使用替代用户和组ID来运行命令。

  • 当没有参数的情况下调用su时默认以root身份运行交互式shell。
  • 为了向后兼容,su默认不更改当前目录并仅设置环境变量HOME和SHELL(如果目标用户不是root的话,加上USER和LOGNAME)。
  • 建议始终使用--login选项(而不是快捷方式-),以避免混合环境导致的副作用,
  • su使用PAM进行认证,帐户和会话管理。在其他su实现中有一些配置选项,例如 必须通过PAM配置wheel组的支持。

案例:

$ su -l elastic
Password:*****

$ whoami
elastic

2.3、sudo

sudo允许允许的用户按照安全策略的规定,以超级用户或其他用户的身份执行命令。

  • sudo支持安全策略和输入/输出日志记录的插件体系结构。第三方可以开发和分发自己的策略和I / O日志插件与sudo前端无缝协作。
  • 默认安全策略是sudoers,通过/ etc / sudoers文件或通过LDAP配置。看PLUGINS部分获取更多信息。
  • 安全策略确定用户必须运行sudo的权限(如果有的话)。该政策可能要求用户使用密码或另一种认证机制验证自己身份。如果需要验证,如果用户的密码未在可配置的时间限制内输入,则sudo将退出。sudoers安全策略的默认密码提示超时为5分钟。安全策略可以支持凭证缓存,以允许用户在不需要认证的情况下再次运行sudo一段时间。 sudoers 安全策略的缓存凭证5分钟,除非在sudoers(5)中覆盖。通过使用-v选项运行sudo,用户无需更新缓存的凭据
         运行一个命令。当作为sudoedit被调用时,隐含-e选项。安全策略可能会记录成功和失败的尝试使用sudo。如果配置了I / O插件,则记录运行命令的输入和输出。

案例:

sudo -u jim -g audio vi ~jim/sound.txt

2.4、总结


命令


root到用户


用户到root


任何用户间


身份验证


系统支持


日志


解释


runuser


Y


N


N


不需要


Linux:是

Solaris:不

MacOS:是


N/A


不用身份验证,效率较高


su


Y


Y


Y


目标用户的密码


Linux:是

Solaris:是

MacOS:是


/var/log/auth.log

/var/log/secure


必须提供用户密码


sudo


Y


Y


Y


用户需要验证本人身份


Linux:是

Solaris:不

MacOS:是


/var/log/auth.log

/var/log/secure


用户必须是wheel组的成员,可以执行管理员的工作

3、python脚本的方法

在实际工作中,越来越多需要用python来自动化任务、集成不同的部件、处理数据。很多时候会碰到同样问题:如何在python脚本中调用不同用户来运行程序。

  • 定义一个函数来设置正在运行的进程的gid和uid。
  • 将此函数作为preexec_fn参数传递给subprocess.Popen

subprocess.Popen将使用fork / exec模型来使用preexec_fn。这等同于按顺序调用

  • os.fork(),
  • preexec_fn()(在子进程中),
  • os.exec()(在子进程中)

必须注意,由于os.setuid,os.setgid和preexec_fn仅在Unix上受支持,因此此解决方案仅支持unix及相关系统。

import os, pwd, subprocess, sys

# set gid and uid
def demote(user_uid, user_gid):
    def result():
        os.setgid(user_gid)
        os.setuid(user_uid)
    return result

# run a command under a new user
def run_command(cmd, user_name, cwd):
    pw_record = pwd.getpwnam(user_name)
    user_name      = pw_record.pw_name
    user_uid       = pw_record.pw_uid
    user_gid       = pw_record.pw_gid

    # get env variable
    env = os.environ.copy()
    env[ ‘HOME‘     ]  = pw_record.pw_dir
    env[ ‘LOGNAME‘  ]  = user_name
    env[ ‘PWD‘      ]  = cwd
    env[ ‘USER‘     ]  = user_name
    proc = subprocess.Popen(cmd,
                          preexec_fn=demote(user_uid, user_gid),
                          cwd=cwd, env=env )
    return proc.communicate()

原文地址:https://www.cnblogs.com/yshen4/p/8897690.html

时间: 2024-11-25 13:51:15

如何在unix系统中用别的用户运行一个程序?的相关文章

多次单击快捷方式,只运行一个程序实例

在应用程序安装之后,单击一次快捷方式,就运行一个程序实例,对于资源独占型程序来说,这样是不可以的,比如该程序使用了当前系统的某个端口,当同样的程序再次运行,再次试图占用同一个端口次,会提示"端口已经被占用的"异常.如此,必须在启动应用程序时,必须判断该程序是否已经有一个实例在运行.下面这个类中先判断该程序的实例有没有在运行,使用线程同步类EventWaitHandle(Boolean, EventResetMode, String)及注册正在等待 WaitHandle 的委托方法Reg

Unix系统编程()进程和程序

进程(process)是一个可执行程序(program)的实例. 程序是包含了一系列信息的文件,这些信息描述了如何在运行时创建一个进程,所包括的内容如下所示. 二进制格式标识:每个程序文件都包含用于描述可执行文件格式的元信息(metainformation).内核(kernel)利用此信息来解释文件中的其他信息. 历史上,UNIX可执行文件曾有两种广泛使用的格式,分别是最初的a.out(汇编程序输出)和更加复杂的COFF(通用对象文件格式). 现在,大多数UNIX实现(包括Linux)采用可执行

如何在linux下安装jdk并运行java程序

一.进入root 大家可以看到我这里用的是CentOS 6.5 系统 二.测试网络与YUM是否可用 1.测试网络 ping www.baidu.com,如下图就是通了 参考: 一.JDK安装1.linux下 yum直接安装:https://blog.csdn.net/youzhouliu/article/details/511831152.安装包安装(如果linux有网也可以去下或通过windows传输):(1)https://www.cnblogs.com/zeze/p/5902124.htm

C#:只运行一个程序

一.通过系统事件 1.实现如下: using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Windows.Forms; using System.Threading; using System.Runtime.InteropServices; namespace Example { public class SinglonProgram { #region

今天在Linux下运行一个程序遇到的问题

我运行一个使用PF_RING+DNA进行捕包的程序,结果在程序运行结束后,我发现我退出不了DNA模式了! 以前都是进入驱动PF_RING-aware中安装对应驱动就可以了,但是这里一直显示pfring in use 和ixgbe.ko in use ,那么,我该怎么办呢? 前段时间总是重启了事,但是今天试着解决之下竟然真的解决了! 用top查看发现一个我的程序占用了98%以上的CPU,正在运行...用kill 19432根本杀死不了这个进程,所以... 最后用kill -9 19432强行杀死这

Winform程序只允许运行一个程序实例

/// <summary> /// 应用程序的主入口点. /// </summary> [STAThread] static void Main() { Application.EnableVisualStyles(); Application.SetCompatibleTextRenderingDefault(false); Application.Run(new Form1()); /* * 利用互斥变量来控制应用程序只能运行一个 */ bool bRun = true; va

nohup命令:以守护进程方式运行一个程序

nohup命令:如果你正在运行一个进程,而且你觉得在退出帐户时该进程还不会结束,那么可以使用nohup命令.该命令可以在你退出帐户/关闭终端之后继续运行相应的进程.nohup就是不挂断的意思( no hang up). 该命令的一般形式为:nohup command & 使用nohup命令提交作业 如果使用nohup命令提交作业,那么在缺省情况下该作业的所有输出都被重定向到一个名为nohup.out的文件中,除非另外指定了输出文件: nohup command > myout.file 2&

在VMware装了linux系统,如何在windows系统中用xshell连接

网上有好几种方法,不过我觉得这种比较简单 1.找到VMware菜单  打开 编辑>虚拟网络编辑器 如图: 点下面的更改设置 点确定就可以了,什么都不用改.然后回到linux系统中ifconfig 连接这个ip即可

1202.1——运行一个程序的完整的过程

1.预编译: 将#include头文件展开 gcc -E printName.c 2.汇编:生产汇编代码 会产生一个.s的文件 判断代码是否正确 gcc -S printName.c 3.编译:将汇编代码转化成目标文件(二进制文件) 产生一个.o的文件 gcc -c printfName.s 4.链接  将所有的.o文件链接为一个可执行的文件 gcc -o outputName printName.o xx.o 5.运行 ./outputName