文件访问权限:更改用户ID
本文来探讨一下通过更改用户ID来获取合适的文件访问权限。由于更改组ID的规则与用户ID相同,我们在这里只探讨用户ID。
纸上得来终觉浅
先了解以下几个基本知识:
- 用户ID包括:实际用户ID、有效用户ID、保存的设置用户ID。其中保存的设置用户ID由exec函数保存。
- 实际用户ID标识我们究竟是谁,该字段在登录时取自口令文件中的登录项。通常,在一个登录会话期间该值不会改变,但root用户进程有方法改变它。
- 有效用户ID决定了我们的文件访问权限。
- 保存的设置用户ID在执行一个程序时包含了有效用户ID的副本。
- 通常,有效用户ID等于实际用户ID。
- 当执行一个程序文件时,进程的有效用户ID通常就是实际用户ID。但是可以在文件模式字(st_mode)中设置一个特殊标志,其含义是“当执行此文件时,将进程的有效用户ID设置为文件所有者的用户ID(st_uid)”。这个特殊标志被称为设置用户ID(set-user-ID)位。
更改这3个用户ID的不同方法总结如下:
| ID | exec | setuid(uid) | ||
| 设置用户ID位关闭 | 设置用户ID位打开 | 超级用户 | 非特权用户 | |
| 实际用户ID | 不变 | 不变 | 设为uid | 不变 |
| 有效用户ID | 不变 | 设置为程序文件的用户ID | 设为uid | 设为uid |
| 保存的设置用户ID | 从有效用户ID复制 | 从有效用户ID复制 | 设为uid | 不变 |
关于以上几点,我们参考一下《Unix高级环境编程》一书中给出的一个例子:
Unix系统程序/usr/bin/passwd是一个设置用户ID程序,该文件的所有者是root用户,而且设置了该文件的设置用户ID位。那么当这个程序文件由一个进程执行时,该进程具有root用户权限。不管执行此文件的进程的实际用户ID是什么,都会是这样,因为该进程的权限不是由实际用户ID决定的,而是由有效用户ID决定的。系统程序/usr/bin/passwd应该能将用户的新口令写入口令文件(一般是/etc/passwd 或 /etc/shadow)中,而只有root用户才具有对该文件的写权限,所以需要使用设置用户ID功能。因为运行设置用户ID程序的进程通常会得到额外的权限,所以编写这种程序时要特别谨慎。
绝知此事要躬行
假设一台Linux主机上有2个用户:Jim(实际用户ID为1000)和Lucy(实际用户ID为1001)。
Jim在自己的主目录下有一个文件test.txt,只有自己才能读写。该文件的访问权限如下所示:
Jim@linux:~$ ls -l test.txt
-rw------- 1 Jim Jim 13 Sep 13 17:07 /home/Jim/test.txt
Lucy想查看该文件,却没有相应的权限,因此会被拒绝:
Lucy@linux:~$ cat /home/Jim/test.txt
cat: /home/Jim/test.txt: Permission denied
Lucy@linux:~$ sudo cat /home/Jim/test.txt
[sudo] password for Lucy:
hello world!
结果Lucy使用root权限强行查看了Jim的test.txt文件。这就没什么好说的了。不过,我们还是建议,在Linux上能不用root权限就不用root权限,用多了可能会伤及无辜。
我们自己的数据文件要由我们自己来保护。Jim可以编写一个设置用户ID程序,Lucy通过执行这个程序,就可以查看Jim的test.txt文件。代码如下:
#include <stdio.h>
#include <unistd.h> #define BUF_LEN (256) int main(void)
{
FILE * fp = NULL;
char buf[BUF_LEN] = {}; uid_t real_uid, effect_uid, saved_uid; if (getresuid(&real_uid, &effect_uid, &saved_uid) == -) {
perror("getresuid");
return -;
} printf("===== before setuid =====\n"); printf("real_uid = %d, effect_uid = %d, saved_uid = %d\n\n", real_uid, effect_uid, saved_uid); /*
* the first important thing this program does is reduce its privilege by using setuid function.
* otherwise we can not protect our data file.
*/ // we reduce this program's privilege to real_uid.
if (setuid(real_uid) == -) {
perror("setuid");
return -;
} printf("====== after setuid with real_uid =====\n"); if (getresuid(&real_uid, &effect_uid, &saved_uid) == -) {
perror("getresuid");
return -;
} printf("real_uid = %d, effect_uid = %d, saved_uid = %d\n\n", real_uid, effect_uid, saved_uid); fp = fopen("/home/Jim/test.txt", "r");
if (fp == NULL) {
printf("\n##### error #####\n");
perror("fopen");
printf("##### error #####\n\n");
} // now we raise this program's privilege to saved_uid.
if (setuid(saved_uid) == -) {
perror("setuid");
return -;
} printf("====== after setuid with saved_uid =====\n"); if (getresuid(&real_uid, &effect_uid, &saved_uid) == -) {
perror("getresuid");
return -;
} printf("real_uid = %d, effect_uid = %d, saved_uid = %d\n\n", real_uid, effect_uid, saved_uid); fp = fopen("/home/Jim/test.txt", "r");
if (fp == NULL) {
printf("\n##### error #####\n");
perror("fopen");
printf("##### error #####\n\n"); return -;
} printf("\n----- file read -----\n");
while (fgets(buf, BUF_LEN, fp))
printf("%s", buf); if (feof(fp))
printf("\nfile reach end!\n");
else
ferror(fp);
printf("----- file close -----\n\n"); fclose(fp); // reduce this program's privilege by setting effect_uid to saved_uid
if (seteuid(saved_uid) == -) {
perror("seteuid");
return -;
} printf("====== after seteuid with saved_uid =====\n"); if (getresuid(&real_uid, &effect_uid, &saved_uid) == -) {
perror("getresuid");
return -;
} printf("real_uid = %d, effect_uid = %d, saved_uid = %d\n\n", real_uid, effect_uid, saved_uid); fp = fopen("/home/Jim/test.txt", "r");
if (fp == NULL) {
printf("\n##### error #####\n");
perror("fopen");
printf("##### error #####\n\n");
} else {
printf("file opened!\n\n");
fclose(fp);
} // reduce this program's privilege by setting effect_uid to real_uid
if (seteuid(real_uid) == -) {
perror("seteuid");
return -;
} printf("====== after seteuid with real_uid =====\n"); if (getresuid(&real_uid, &effect_uid, &saved_uid) == -) {
perror("getresuid");
return -;
} printf("real_uid = %d, effect_uid = %d, saved_uid = %d\n\n", real_uid, effect_uid, saved_uid); fp = fopen("/home/Jim/test.txt", "r");
if (fp == NULL) {
printf("\n##### error #####\n");
perror("fopen");
printf("##### error #####\n\n");
} else {
printf("file opened!\n\n");
fclose(fp);
} return ;
}
编译该程序的时候要加上_GNU_SOURCE标志(用于getresuid函数),编译完成之后还要打开该程序的设置用户ID标志位。
Jim@linux:~$ gcc uid_test.c -o uid_test -D _GNU_SOURCE
Jim@linux:~$ chmod u+s uid_test
Jim@linux:~$ ls -l uid_test
-rwsrwxr-x 1 Jim Jim 7726 Sep 14 17:19 /home/Jim/uid_test
从上面的命令输出中可以看到,uid_test程序的设置用户ID标志位已经打开。Lucy可以使用这个程序来查看Jim的test.txt文件。
Lucy@linux:~$ /home/Jim/uid_test
===== before setuid =====
real_uid = 1001, effect_uid = 1000, saved_uid = 1000
====== after setuid with real_uid =====
real_uid = 1001, effect_uid = 1001, saved_uid = 1000
##### error #####
fopen: Permission denied
##### error #####
====== after setuid with saved_uid =====
real_uid = 1001, effect_uid = 1000, saved_uid = 1000
----- file read -----
hello world!
file reach end!
----- file close -----
====== after seteuid with saved_uid =====
real_uid = 1001, effect_uid = 1000, saved_uid = 1000
file opened!
====== after seteuid with real_uid =====
real_uid = 1001, effect_uid = 1001, saved_uid = 1000
##### error #####
fopen: Permission denied
##### error #####
Lucy@linux:~$
从上面的输出中可以看出,通过改变用户ID可以控制程序的访问权限。
另外注意
setuid函数与seteuid函数的区别:
相同之处:对于一个非特权用户来说,两者都可以将进程的有效用户ID设置为其实际用户ID或其保存的设置用户ID。
不同之处:对于一个特权用户来说,setuid(uid)函数将实际用户ID、有效用户ID以及保存的设置用户ID设为uid;而seteuid(uid)函数只将进程的有效用户ID设为uid。
还是那句话:因为运行设置用户ID程序的进程通常会得到额外的权限,所以编写这种程序时要特别谨慎。
文件访问权限:更改用户ID的更多相关文章
- 用户id,组id和文件访问权限
实际用户ID和实际组ID:标示了我们究竟是谁,这两个字段在登录时取自口令文件中的登录项 有效用户ID和有效组ID以及附属组ID:决定了我们的文件的访问权限(通常有效用户ID等于实际用户ID,有效组ID ...
- UNIX环境编程学习笔记(9)——文件I/O之文件访问权限的屏蔽和更改
lienhua342014-09-10 1 文件访问权限 在文件访问权限和进程访问控制中,我们已经讲述过文件访问权限位,为了方便,我们重新列在下面, 表 1: 文件的 9 个访问权限位 st_mod ...
- C#更改文件访问权限所有者(适用于各个Windows版本)
前面也提到了,前段时间在做Online Judge系统,在正式上线前有几个比较老的版本,其中第一个版本使用ACL来控制权限以确保安全(但是这个版本完全建立在IIS上,所以这样做是没效果的),遇到了一些 ...
- 《UNIX环境高级编程》笔记--更改用户ID和组ID
在unix系统中,特权是基于用户和组ID的,当程序需要增加特权,或需要访问当前并不允许访问的资源时,我们需要更换自己 用户ID或组ID,使的新ID具有合适的特权或访问权限.与此类似,当程序需要降低其特 ...
- 进程控制之更改用户ID和组ID
在UNIX系统中,特权(例如能改变当前日期的表示法以及访问控制(例如,能否读.写一特定文件))是基于用户ID和组ID的.当程序需要增加特权,或需要访问当前并不允许访问的资源时,我们需要更换自己的用户I ...
- 更改用户id 和组id
转自 http://blog.csdn.net/todd911/article/details/16370577 在unix系统中,特权是基于用户和组ID的,当程序需要增加特权,或需要访问当前并不允许 ...
- [apue] linux 文件访问权限那些事儿
前言 说到 linux 上的文件权限,其实我们在说两个实体,一是文件,二是进程.一个进程能不能访问一个文件,其实由三部分内容决定: 文件的所有者.所在的组: 文件对所有者.组用户.其它用户设置的权限访 ...
- Linux下进程的文件访问权限
本文转自 http://blog.csdn.net/chosen0ne/article/details/10581883 对进程校验文件访问权限包括两个部分,一是确定进程的角色(属于哪个用户或者组), ...
- 《UNIX环境高级编程》笔记--文件访问权限和新文件、目录所有权
1.与进程关联的用户ID和组ID 与一个进程关联的ID有一下几个: 实际用户ID和实际组ID标识我们究竟是谁.通常在一个会话间值是不会改变的,但是超级用户进程有方法改变 他们,在以后的进程控制中会进行 ...
随机推荐
- C语言-第6次作业
1.本章学习总结 1.1思维导图 1.2 本章学习体会 学习感受:先是接触到网络工程导论的建立文件的方式,觉得很方便很好用,而后直接从代码的层面建立文件,觉得很新奇,更加方便,随着大作业的一步一步升级 ...
- Dingo 的安装
安装Dingo需要下面的环境: Laravel 5.1 或 lumen 5.1 + php 5.59+ 1. 修改composer.json 文件,添加下面的代码: "require&quo ...
- Generator
基本概念 Generator函数是ES6提供的一种异步编程解决办法,语法行为与传统函数完全不同. Generator函数有多种理解角度.语法上,首先可以把它理解成,Generator函数是一个状态机, ...
- Spring定时任务@Scheduled注解使用方式
1.开篇 spring的@Scheduled定时任务相信大家都是十分熟悉.最近在使用过程中发现了一些问题,写篇文章,和大家分享一下.结论在最后,不想看冗长过程的小伙伴可以直接拉到最后看结论. 2.简单 ...
- 函数嵌套定义,闭包及闭包的应用场景,装饰器,global.nonlocal关键字
函数的嵌套定义 在一个函数的内部定义另一个函数 为什么要有函数的嵌套定义: 1)函数fn2想直接使用fn1函数的局部变量,可以将fn2直接定义到fn1的内部,这样fn2就可以直接访问fn1的变凉了 2 ...
- Cocos Creator学习六:加载/释放图片资源
1.目的:学习加载图片资源.使用图片资源创建对象以及释放图片资源. 2.注意事项以及主要函数: ①注意事项:使用loadRes函数,资源必须放置在assets下的resources文件夹下(默认没有r ...
- 《R语言入门与实践》第六章:R 的环境系统
前言 这一章在对象的基础之上,讲解了对象所处的环境,进一步讲了环境对对象的作用,以及如何使用环境.结构如下: 环境的定义和操作 环境的规则 制作闭包 环境 R 环境的定义 在 R 中,每一个数据对象都 ...
- 关于PHP的mkdir函数
问题:dedecms5.7 php5.6 我想项目根目录下的uploads文件夹下动态创建一个文件夹/uploads/imgs $path = '/uploads/imgs'; mkdir($path ...
- jacoco覆盖率工具测试及性能分析
ant版本:https://ant.apache.org/bindownload.cgi jdk版本 注: ant 1.10 ---> jdk1.8 ant 1.9 ---& ...
- Matlab:双曲方程
tic; clear clc M=[, ];%空间步数 N=*M;%时间步数 :length(M) h=/M(k);%空间步长 tau=/N(k);%时间步长 s=tau/h;%步长比 x=:h:; ...