#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <pwd.h>
#include <grp.h>
#include <fcntl.h>
#include <dirent.h>
#include <unistd.h>
#include <string.h>
#include <time.h>
#include <stdlib.h>
#include <limits.h>
int islsflag( char * arg) ;
int printout( char * dir, int Rflag, int lflag, int gflag) ;
int main( int argc, char ** argv) {
int lflag= 0 , gflag= 0 , Rflag= 0 ;
int argexist= 0 ; //флаг существования аргументов - каталогов
for ( int i= 1 ; i< argc; i++ )
if ( islsflag( argv[ i] ) ) {
char c;
for ( int j= 1 ; ( c= argv[ i] [ j] ) != '\0 ' ; j++ ) {
if ( c== 'l' ) lflag= 1 ;
else if ( c== 'g' ) gflag= 1 ;
if ( c== 'R' ) Rflag= 1 ;
}
}
else argexist= 1 ;
if ( lflag && gflag) {
fprintf ( stderr
, "ls: wrong usage of flags" ) ; return 1 ;
}
char l[ PATH_MAX] ;
getcwd( l, PATH_MAX) ;
if ( argexist) {
for ( int i= 1 ; i< argc; i++ ) if ( ! islsflag( argv[ i] ) ) {
char s[ PATH_MAX] ;
if ( ! Rflag
) printf ( "%s:\n " , argv
[ i
] ) ; if ( ! islsflag( argv[ i] ) && printout( s, Rflag, lflag, gflag) )
return 1 ; //в printout произошла ошибка
}
return 0 ; //ошибок не произошло
}
return printout( l, Rflag, lflag, gflag) ;
}
/* возвращает 0, если аргумент - не флаг, и 1 - если флаг*/
int islsflag( char * arg) {
if ( arg[ 0 ] == '-' ) {
char c;
for ( int j= 1 ; ( c= arg[ j] ) != '\0 ' ; j++ ) {
if ( ( c!= 'l' ) && ( c!= 'g' ) && ( c!= 'R' ) )
return 0 ;
}
return 1 ;
}
return 0 ;
}
/* показать тип файла в первой позиции выходной строки */
void display_file_type( int st_mode) {
switch ( st_mode & S_IFMT ) {
case S_IFDIR
: putchar ( 'd' ) ; return ; case S_IFCHR
: putchar ( 'c' ) ; return ; case S_IFBLK
: putchar ( 'b' ) ; return ; case S_IFREG
: putchar ( '-' ) ; return ; case S_IFLNK
: putchar ( 'l' ) ; return ; case S_IFSOCK
: putchar ( 's' ) ; return ; }
}
/* показать права доступа для владельца, группы и прочих пользователей, а также все спец.флаги*/
void display_permission( int st_mode) {
static const char rwxmas[ 10 ] = "rwxrwxrwx" ;
char amode[ 10 ] ;
int i, j;
for ( i = 0 , j = ( 1 << 8 ) ; i < 9 ; i++, j >>= 1 )
amode[ i] = ( st_mode& j) ? rwxmas[ i] : '-' ;
if ( st_mode & S_ISUID ) amode[ 2 ] = 's' ;
if ( st_mode & S_ISGID ) amode[ 5 ] = 's' ;
if ( st_mode & S_ISVTX ) amode[ 8 ] = 't' ;
amode[ 9 ] = '\0 ' ;
}
char * settime( struct tm * u) {
char * s;
int size= 20 ;
char * tmp;
int length;
while ( ! ( length
= strftime ( s
, size
, "%d.%m.%Y %H:%M:%S" , u
) ) ) { size+= 10 ;
}
/* в строку s скопируем строку из кавычек, заменяя спецификаторы
* данными из структуры времени, не более size символов*/
return ( tmp) ;
}
/* перечислить атрибуты одного файла*/
/* от lflag (0 или 1) зависит, будет ли выводиться имя владельца*/
void long_list( char * path_name, char * fname, int lflag) {
struct stat statv;
struct passwd * pw_d;
struct group * grpv;
struct tm * timev;
if ( lstat( path_name,& statv) ) {
return ;
}
display_file_type( statv.st_mode ) ;
display_permission( statv.st_mode ) ;
printf ( "%4ld " , statv.
st_nlink ) ; /*значение счетчика жестких связей*/
if ( lflag) {
if ( ( pw_d= getpwuid( statv.st_uid ) ) == NULL) {
/* преобразовать UID в имя пользователя*/
return ;
}
}
if ( ( grpv= getgrgid( statv.st_gid ) ) == NULL) {
/* преобразовать gid в имя группы*/
return ;
}
/* показать размер файла*/
printf ( "%8ld " , statv.
st_size ) ;
/* показать время последнего изменения файла*/
const time_t
ctime = statv.
st_ctime ; char * strtime= settime( timev) ;
/* показать имя файла*/
}
/* вывод, возвращает 0, если ошибок нет, 1 - если есть*/
int printout( char * dir, int Rflag, int lflag, int gflag) {
DIR * d;
struct dirent * dd;
off_t o;
struct stat s;
char name[ PATH_MAX] ;
char * delim= "/" ;
/* если в качестве каталога передан корневой, то разделитель '/' не нужен*/
if ( ! strcmp ( dir
, "/" ) ) delim
= "" ;
if ( ! ( d= opendir( dir) ) ) {
return 1 ;
}
if ( Rflag
) printf ( "%s:\n " , dir
) ; while ( ( dd= readdir( d) ) ) {
continue ;
//сформируем полный путь
snprintf ( name
, PATH_MAX
, "%s%s%s" , dir
, delim
, dd
-> d_name
) ;
if ( lstat( name, & s) < 0 )
continue ;
if ( lflag || gflag) {
long_list( name, dd-> d_name, lflag) ; //вывод полной информации
}
else
printf ( "%s " , dd
-> d_name
) ; //только имя }
if ( ! lflag
&& ! gflag
) printf ( "\n " ) ;
if ( ! Rflag) return 0 ;
seekdir( d, 0 ) ;
snprintf ( name
, PATH_MAX
, "%s%s" , dir
, delim
) ; while ( ( dd= readdir( d) ) ) {
continue ;
char mas[ PATH_MAX] ;
if ( lstat( mas, & s) < 0 )
continue ;
if ( S_ISDIR( s.st_mode ) ) {
//если это каталог
char mas[ PATH_MAX] ;
o= telldir( d) ; //запомнить место в записи
closedir( d) ; //для экономии дескрипторов
chdir( mas) ;
if ( printout( mas, Rflag, lflag, gflag) ) return 1 ;
chdir( name) ;
//вернемся в прежнюю директорию
if ( ! ( d= opendir( dir) ) ) {
return 1 ;
}
seekdir( d, o) ; //вернуться на прежнее место
}
}
closedir( d) ;
return 0 ;
}
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <pwd.h>
#include <grp.h>
#include <fcntl.h>
#include <dirent.h>
#include <unistd.h>
#include <string.h>
#include <time.h>
#include <stdlib.h>
#include <limits.h>

int islsflag(char * arg);
int printout(char * dir, int Rflag, int lflag, int gflag);

int main(int argc,char ** argv){
	int lflag=0,gflag=0,Rflag=0;
	int argexist=0; //С„Р»Р°Рі СЃСѓС‰РµСЃС‚РІРѕРІР°РЅРёСЏ Р°СЂРіСѓРјРµРЅС‚РѕРІ - РєР°С‚Р°Р»РѕРіРѕРІ
	for (int i=1;i<argc;i++)
		if (islsflag(argv[i])){
			char c;
			for (int j=1;(c=argv[i][j])!='\0';j++){
				if (c=='l') lflag=1;
				else if (c=='g') gflag=1;
				if (c=='R') Rflag=1;
			}
		}
		else argexist=1;
	if (lflag && gflag){
		fprintf(stderr,"ls: wrong usage of flags");
		return 1;
	}

        char l[PATH_MAX];
	getcwd(l,PATH_MAX);

	if (argexist){
		for (int i=1;i<argc;i++) if (!islsflag(argv[i])){
			char s[PATH_MAX]; 
			strcpy(s,l);
			strcat(s,"/");
			strcat(s,argv[i]);
			if (!Rflag) printf("%s:\n",argv[i]);
			if (!islsflag(argv[i]) && printout(s,Rflag,lflag,gflag))
				return 1; //РІ printout РїСЂРѕРёР·РѕС€Р»Р° РѕС€РёР±РєР°
		}
		return 0; //РѕС€РёР±РѕРє РЅРµ РїСЂРѕРёР·РѕС€Р»Рѕ
	}
	return printout(l,Rflag,lflag,gflag);

}

/* РІРѕР·РІСЂР°С‰Р°РµС‚ 0, РµСЃР»Рё Р°СЂРіСѓРјРµРЅС‚ - РЅРµ С„Р»Р°Рі, Рё 1 - РµСЃР»Рё С„Р»Р°Рі*/
int islsflag(char * arg){
	if (arg[0]=='-'){
		char c;
		for (int j=1;(c=arg[j])!='\0';j++){
			if ( (c!='l') && (c!='g') && (c!='R') ) 
				return 0;
		}
		return 1;
	}
	return 0;
}

/* РїРѕРєР°Р·Р°С‚СЊ С‚РёРї С„Р°Р№Р»Р° РІ РїРµСЂРІРѕР№ РїРѕР·РёС†РёРё РІС‹С…РѕРґРЅРѕР№ СЃС‚СЂРѕРєРё */
void display_file_type(int st_mode){                                   
	switch ( st_mode & S_IFMT ){
        	case S_IFDIR:  putchar ( 'd' ); return;
	        case S_IFCHR:  putchar ( 'c' ); return;
	        case S_IFBLK:  putchar ( 'b' ); return;
	        case S_IFREG:  putchar ( '-' ); return;
	        case S_IFLNK:  putchar ( 'l' ); return;
	        case S_IFSOCK: putchar ( 's' ); return;
	}
} 
 
/* РїРѕРєР°Р·Р°С‚СЊ РїСЂР°РІР° РґРѕСЃС‚СѓРїР° РґР»СЏ РІР»Р°РґРµР»СЊС†Р°, РіСЂСѓРїРїС‹ Рё РїСЂРѕС‡РёС…  РїРѕР»СЊР·РѕРІР°С‚РµР»РµР№, Р° С‚Р°РєР¶Рµ РІСЃРµ СЃРїРµС†.С„Р»Р°РіРё*/
void display_permission(int st_mode){
	static const char rwxmas[10] = "rwxrwxrwx";
	char     amode[10];
	int      i, j;
	 
	for ( i = 0, j = ( 1 << 8 ); i < 9; i++, j >>= 1 )
		amode[i] = (st_mode&j) ? rwxmas[i]: '-';
	if ( st_mode & S_ISUID )   amode[2]= 's';
	if ( st_mode & S_ISGID )   amode[5]= 's';
	if ( st_mode & S_ISVTX )   amode[8]= 't';
	amode[9]='\0';
	printf ( "%s ",amode );
}

char * settime(struct tm *u){
	char *s;
	int size=20;
	char *tmp;
	int length;
	s=(char*)malloc(size);

	while (!(length=strftime(s,size,"%d.%m.%Y %H:%M:%S",u))){
		size+=10;
		s=(char*)realloc(s,size);
	}
	/* РІ СЃС‚СЂРѕРєСѓ s СЃРєРѕРїРёСЂСѓРµРј СЃС‚СЂРѕРєСѓ РёР· РєР°РІС‹С‡РµРє, Р·Р°РјРµРЅСЏСЏ СЃРїРµС†РёС„РёРєР°С‚РѕСЂС‹
	 * РґР°РЅРЅС‹РјРё РёР· СЃС‚СЂСѓРєС‚СѓСЂС‹ РІСЂРµРјРµРЅРё, РЅРµ Р±РѕР»РµРµ size СЃРёРјРІРѕР»РѕРІ*/

	tmp=(char*)malloc(sizeof(s));
	strcpy(tmp,s);
	return(tmp);
}

/* РїРµСЂРµС‡РёСЃР»РёС‚СЊ Р°С‚СЂРёР±СѓС‚С‹ РѕРґРЅРѕРіРѕ С„Р°Р№Р»Р°*/
/* РѕС‚ lflag (0 РёР»Рё 1) Р·Р°РІРёСЃРёС‚, Р±СѓРґРµС‚ Р»Рё РІС‹РІРѕРґРёС‚СЊСЃСЏ РёРјСЏ РІР»Р°РґРµР»СЊС†Р°*/
void long_list(char * path_name, char * fname, int lflag){
	struct stat     statv;
	struct passwd  *pw_d;
	struct group   *grpv;
	struct tm      *timev;

	if (lstat(path_name,&statv)){
		perror(path_name);
		return;
	}
	display_file_type(statv.st_mode);
	display_permission(statv.st_mode);
	printf("%4ld ",statv.st_nlink); /*Р·РЅР°С‡РµРЅРёРµ СЃС‡РµС‚С‡РёРєР° Р¶РµСЃС‚РєРёС… СЃРІСЏР·РµР№*/

	if (lflag){
		if ((pw_d=getpwuid(statv.st_uid))==NULL){ 
		/* РїСЂРµРѕР±СЂР°Р·РѕРІР°С‚СЊ UID РІ РёРјСЏ РїРѕР»СЊР·РѕРІР°С‚РµР»СЏ*/
			perror("");
			return;
		}	  
		printf ("%s ",pw_d->pw_name);   
	}

	if ((grpv=getgrgid(statv.st_gid))==NULL){
	  /* РїСЂРµРѕР±СЂР°Р·РѕРІР°С‚СЊ gid РІ РёРјСЏ РіСЂСѓРїРїС‹*/
		perror("");
		return;
	}
	printf("%s ",grpv->gr_name);
	
	/* РїРѕРєР°Р·Р°С‚СЊ СЂР°Р·РјРµСЂ С„Р°Р№Р»Р°*/
	printf("%8ld ",statv.st_size);
	
	/* РїРѕРєР°Р·Р°С‚СЊ РІСЂРµРјСЏ РїРѕСЃР»РµРґРЅРµРіРѕ РёР·РјРµРЅРµРЅРёСЏ С„Р°Р№Р»Р°*/
	const time_t ctime=statv.st_ctime;
	timev=localtime(&ctime);
	char * strtime=settime(timev);
	printf("%s",strtime);

	/* РїРѕРєР°Р·Р°С‚СЊ РёРјСЏ С„Р°Р№Р»Р°*/
	printf(" %s\n", fname);
}



/* РІС‹РІРѕРґ, РІРѕР·РІСЂР°С‰Р°РµС‚ 0, РµСЃР»Рё РѕС€РёР±РѕРє РЅРµС‚, 1 - РµСЃР»Рё РµСЃС‚СЊ*/
int printout(char * dir,int Rflag,int lflag,int gflag){
	DIR *d;
	struct dirent *dd;
	off_t o;
	struct stat s;
	char name[PATH_MAX];
	char *delim="/";
	
	/* РµСЃР»Рё РІ РєР°С‡РµСЃС‚РІРµ РєР°С‚Р°Р»РѕРіР° РїРµСЂРµРґР°РЅ РєРѕСЂРЅРµРІРѕР№, С‚Рѕ СЂР°Р·РґРµР»РёС‚РµР»СЊ '/' РЅРµ РЅСѓР¶РµРЅ*/
	if (!strcmp(dir, "/")) delim= "";
	
	if (!(d=opendir(dir))){
		perror(dir);
		return 1;
	}
 	if (Rflag) printf("%s:\n",dir);
	while ((dd=readdir(d))){
		if(!strcmp(dd->d_name, ".") || !strcmp(dd->d_name, ".."))
			continue;

		//СЃС„РѕСЂРјРёСЂСѓРµРј РїРѕР»РЅС‹Р№ РїСѓС‚СЊ
		snprintf(name, PATH_MAX, "%s%s%s", dir, delim, dd->d_name);

		if(lstat(name, &s) < 0) 
			continue;
		if (lflag || gflag){
			printf("  ");
			long_list(name, dd->d_name, lflag); //РІС‹РІРѕРґ РїРѕР»РЅРѕР№ РёРЅС„РѕСЂРјР°С†РёРё
		}
		else 
			printf("%s ",dd->d_name);//С‚РѕР»СЊРєРѕ РёРјСЏ
	}
	if (!lflag && !gflag) printf("\n");
	
	if (!Rflag) return 0;
	if (!strcmp(dir,"..") || !strcmp(dir,".")) return 0;
	printf("\n");
	seekdir(d,0);
	snprintf(name, PATH_MAX, "%s%s", dir, delim);
	while ((dd=readdir(d))){
		if(!strcmp(dd->d_name, ".") || !strcmp(dd->d_name, ".."))
			continue;
		char mas[PATH_MAX];
		strcpy(mas,name);
		strcat(mas,dd->d_name);
		if(lstat(mas, &s) < 0) 
			continue;
		if(S_ISDIR(s.st_mode)){
		//РµСЃР»Рё СЌС‚Рѕ РєР°С‚Р°Р»РѕРі
			char mas[PATH_MAX];
			o=telldir(d); //Р·Р°РїРѕРјРЅРёС‚СЊ РјРµСЃС‚Рѕ РІ Р·Р°РїРёСЃРё
			closedir(d); //РґР»СЏ СЌРєРѕРЅРѕРјРёРё РґРµСЃРєСЂРёРїС‚РѕСЂРѕРІ
			strcpy(mas,name);
			strcat(mas,dd->d_name);

			chdir(mas);
			if (printout(mas,Rflag,lflag,gflag)) return 1;
			chdir(name);
			//РІРµСЂРЅРµРјСЃСЏ РІ РїСЂРµР¶РЅСЋСЋ РґРёСЂРµРєС‚РѕСЂРёСЋ
			if(!(d=opendir(dir))) {
				perror(dir);
				return 1;
			}
			seekdir(d, o); //РІРµСЂРЅСѓС‚СЊСЃСЏ РЅР° РїСЂРµР¶РЅРµРµ РјРµСЃС‚Рѕ
		}
	}
	closedir(d);
	return 0;
}