APUE-文件和目录(三)函数chown 和lchown

简介: 下面的几个chown函数可用于更改文件的用户ID和组ID。如果两个参数owner或group中的任意一个是-1,则对应的ID不变。#includeint chown(const char *pathname,uid_t owner,gid_t group);int fchown(int fd,...

下面的几个chown函数可用于更改文件的用户ID和组ID。如果两个参数owner或group中的任意一个是-1,则对应的ID不变。

#include<unistd.h>
int chown(const char *pathname,uid_t owner,gid_t group);
int fchown(int fd,uid_t owner,gid_t group);
int fchownat(int fd,const char *pathname,uid_t owner,gid_t group,int flag);
int lchown(const char *pathname,uid_t owner,gid_t group);
若成功返回0;若出错返回-1

在符号链接下,lchown和fchownat(设置了AT_SYMLINK_NOFOLLOW标志)更改符号链接本身的所有者,而不是该符号链接所指向的文件的所有者。

看下面的例子:

#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <errno.h>
#define RWRWRW  (S_IRUSR |S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH)
//#define  _POSIX_CHOWN_RESTRICTED -1

int main(void)
{
    umask(0);//remove the mask
    int rv = creat("source.txt",RWRWRW);//creat a file whose mode is -rw-rw-rw-
    system("ln -s source.txt source_l.txt");//create a soft link to "source.txt" whose mode is lrwxrwxrwx
    rv = lchown("source_l.txt",0,0);//update the user ID and group ID to 0
    printf("rv:%d\n",rv);
    printf("errno:%d\n",errno);
    return 0;
}

首先创建一个文件source.txt,然后创建一个符号链接source_1.txt,最后修改此符号链接的所有者和所属组。

harlan@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples$ gcc 4-11.c
harlan@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples$ ./a.out          
rv:-1                                                                      
errno:1                                                                    
harlan@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples$ ll               
总用量 29                                                                     
-rw-r----- 1 harlan harlan  282  6月  5 22:52 4-10.c                        
-rw-rw-r-- 1 harlan harlan  516  6月  6 21:51 4-11.c                        
-rw-rw-r-- 1 harlan harlan  290  6月  1 22:13 4-7.c                         
-rw-rw-r-- 1 harlan harlan 1168  6月  5 22:15 4-9.c                         
-rw-rw-r-- 1 harlan harlan 4950  6月  1 09:07 4.c                           
-rwxrwxr-x 1 harlan harlan 8788  6月  6 21:51 a.out                         
lrwxrwxrwx 1 harlan harlan   10  6月  6 22:00 source_l.txt -> source.txt    
-rw-rw-rw- 1 harlan harlan    0  6月  6 22:00 source.txt                    

在harlan用户下生成可执行文件,并执行,文件和符号链接生成成功,但是lchown执行失败,errno为1,表示“Operation not permitted”。

原因因为如下:

基于BSD的系统一直规定只有超级用户才能更改一个文件的所有者。

切换到root用户下执行,最后成功:

harlan@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples$ su
密码:
root@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples# rm source*
root@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples# ./a.out
rv:0
errno:0
root@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples# ll
总用量 37
drwxrwsr-t 2 harlan harlan    0  6月  6 22:03 ./
drwxrwxrwx 2 harlan harlan    0  5月 18 21:16 ../
-rw-r----- 1 harlan harlan  282  6月  5 22:52 4-10.c
-rw-rw-r-- 1 harlan harlan  516  6月  6 21:51 4-11.c
-rw-rw-r-- 1 harlan harlan  290  6月  1 22:13 4-7.c
-rw-rw-r-- 1 harlan harlan 1168  6月  5 22:15 4-9.c
-rw-rw-r-- 1 harlan harlan 4950  6月  1 09:07 4.c
-rwxrwxr-x 1 harlan harlan 8788  6月  6 21:51 a.out*
lrwxrwxrwx 1 root   root     10  6月  6 22:03 source_l.txt -> source.txt
-rw-rw-rw- 1 root   harlan    0  6月  6 22:03 source.txt

接下来看一下怎么才能修改文件的组,满足如下条件就能够修改文件的组:
如果进程拥有此文件(有效用户ID等于该文件的用户ID),参数owner等于-1或文件的用户ID,并且参数group等于进程的有效组ID或进程的附属组ID之一,那么一个非超级用户进程可以更改该文件的组ID。
看下面的例子:

首先我们为当前用户harlan添加一个附属组ID,改之前:

harlan@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples$ id harlan
uid=1000(harlan) gid=1000(harlan) group=1000(harlan)

改之后:

harlan@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples$ sudo usermod -G zexu harlan   
[sudo] password for harlan:
harlan@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples$ id harlan
uid=1000(harlan) gid=1000(harlan) group=1000(harlan),1001(zexu)

为了是当前设置生效,执行如下命令:

harlan@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples$ newgrp zexu

最后看下面的代码:

#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <errno.h>
#define RWRWRW  (S_IRUSR |S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH)
void test_chownGID()
{
    int rv = creat("testGID.txt",RWRWRW);
    if(rv<0)
    {
        printf("create file error!\n");
    }
    struct stat statbuf;
    if(stat("testGID.txt",&statbuf)<0)
    {
        printf("stat error!\n");
    }
    else
    {
        printf("The current user's group ID is %d\n",statbuf.st_gid);
    }
    rv = chown("testGID.txt",-1,1001);
    printf("rv:%d\n",rv);
    if(stat("testGID.txt",&statbuf)<0)
    {
        printf("stat error!\n");
    }
    else
    {
        printf("After chown,the user's group ID is %d\n",statbuf.st_gid);
    }
}
int main(void)
{    
    test_chownGID();
    return 0;
}

执行结果:

harlan@DESKTOP-KU8C3K5:/github/APUE/chapter_4/myexamples$ ./a.out
The current user's group ID is 1000
rv:0
After chown,the user's group ID is 1001


作者: HarlanC

博客地址: http://www.cnblogs.com/harlanc/
个人博客: http://www.harlancn.me/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出, 原文链接

如果觉的博主写的可以,收到您的赞会是很大的动力,如果您觉的不好,您可以投反对票,但麻烦您留言写下问题在哪里,这样才能共同进步。谢谢!

目录
相关文章
|
7月前
|
Linux Windows
linux(十七)文件和目录相关命令-软连接、硬链接 ln命令
linux(十七)文件和目录相关命令-软连接、硬链接 ln命令
101 0
|
存储 Linux
【Linux命令200例】lsattr用于查看文件或目录的属性
lsattr 是 Linux 系统中的一个命令,用于查看文件或目录的属性。它可以列出文件或目录的扩展属性(Extended attributes),以及是否设置了一些特殊属性,如不可删除、不可修改等。本文将详细介绍 `lsattr` 命令的用法,并提供几个实际的例子。
542 0
|
Ubuntu Linux Shell
linux第三课:目录文档操作命令(内含绝对/相对路径+1.pwd+2.cd+3.mkdir(创建目录)+4. rmdir(删除目录)+5. ls+6. cp+7.rm+8cat+9touch命令)
linux第三课:目录文档操作命令(内含绝对/相对路径+1.pwd+2.cd+3.mkdir(创建目录)+4. rmdir(删除目录)+5. ls+6. cp+7.rm+8cat+9touch命令)
228 0
linux第三课:目录文档操作命令(内含绝对/相对路径+1.pwd+2.cd+3.mkdir(创建目录)+4. rmdir(删除目录)+5. ls+6. cp+7.rm+8cat+9touch命令)
|
Shell Linux
mkdir() 函数和 rmdir() 函数
mkdir() 函数和 rmdir() 函数
88 0
|
Windows
liunx之拷贝命令cp
cp(全称:copy file)用于复制文件或目录。
224 0