BookmarkSubscribeRSS Feed
☑ This topic is solved. Need further help from the community? Please sign in and ask a new question.
Lost_Gary
Quartz | Level 8

I don't seem to be able to figure this out, but I'd like to pull into a dataset a listing of filenames and modification dates from a network folder.  I am thinking this is simple, but I don't seem be able find a solution that works.  Any thoughts would be appreciated.  

1 ACCEPTED SOLUTION
6 REPLIES 6
SASJedi
Ammonite | Level 13

You can use the %findfiles macro from my public GitHub sas-macros repository to do this pretty easily. 

 

/* Get the macros straight from GitHub */ 
/* Supporting macro utilities */
filename attrib "fileattribs.sas";
filename exist "exist.sas";
filename trans "translate.sas";
/* The main FindFiles macro */
filename find "findfiles.sas";

proc http 
	url="https://raw.githubusercontent.com/SASJedi/sas-macros/master/findfiles.sas"
	out=find;
run;
proc http 
	url="https://raw.githubusercontent.com/SASJedi/sas-macros/master/exist.sas"
	out=exist;
run;
proc http 
	url="https://raw.githubusercontent.com/SASJedi/sas-macros/master/fileattribs.sas"
	out=attrib;
run;
proc http 
	url="https://raw.githubusercontent.com/SASJedi/sas-macros/master/translate.sas"
	out=trans;
run;
	 /* Review the contents of the macro programs before running them! */
data _null_;
	infile attrib;
	file print;
	input; 
	put _infile_;
run;
data _null_;
	infile exist;
	file print;
	input; 
	put _infile_;
run;
data _null_;
	infile trans;
	file print;
	input; 
	put _infile_;
run;
data _null_;
	infile find;
	file print;
	input; 
	put _infile_;
run;

/* Run the macro programs to compile the macros */
%include attrib;
%include exist;
%include trans;
%include find;

/* Get syntax help for FindFiles in the SAS log */
%FindFiles(?)

From the log:

NOTE:  *FINDFILES Documentation *******************************

       Produces a list of files with a specified extension in the log
       or optionally writes them to a dataset.

       SYNTAX: %FindFiles(dir,<ext,dsn,sub>)
          dir=fully qualified directory path
          ext=Space delimited list of file extensions (Optional, default is ALL)
          dsn=name of data set to store filenames (Optional, otherwise writes to log.)
          sub=look in subfolders? (Y|N default is Y)

       Example:
       %FindFiles(c:\temp, csv)
       %FindFiles(\\server\folder\, xls xlsx xlsm, work.myfiles)
       %FindFiles(s:/workshop,sas,work.pgm_files,N)

       *************************************************************

Now, use the macro to produce a dataset named work.files containing information for all the files in \\localhost\c$\temp and all subfolders:

%FindFiles(\\localhost\c$\temp,,work.Files)
proc print data=work.files(obs=3);
run;

A sample of the results:

 

Obs Item Path Filename Size CRDate CRTime ModDate ModTime
1 1 //localhost/c$/temp/subfolder another_solution.sas 878 11/17/2023 8:43:59 11/10/2023 16:42:43
2 1 //localhost/c$/temp have.csv 208 10/13/2023 8:29:05 10/13/2023 8:29:05
3 2 //localhost/c$/temp my.html 943 07/11/2023 16:38:10 07/11/2023 16:38:43

 

May the SAS be with you!
Mark

Check out my Jedi SAS Tricks for SAS Users
Lost_Gary
Quartz | Level 8

Sorry, but I cannot seem to get that macro to work. 

 

I am able to get this code to work, but only for the filename.  I am not exactly familiar with the loop and how to insert the date modified into this process.  Any thoughts are appreciated.  

 

data filenames;
length fref $8 fname $200 ;
did = filename(fref,"\\My Folder");
did = dopen(fref);

do i = 1 to dnum(did);
fname = dread(did,i);

output;
end;
did = dclose(did);
did = filename(fref);
keep fname;
run;

Tom
Super User Tom
Super User

I think that macro is calling other macros. So you will need to download and compile those also.

Or you could use this one which does not depend on other macros. https://github.com/sasutils/macros/blob/master/dirtree.sas

 

Essentially you need to also use the FILENAME(), FOPEN(), FCLOSE() and FINFO() and perhaps FOPTNAME() functions to make a fileref pointing to each file and then open the file and retrieve information about the file, like the last modified date.

 

If the XCMD option is enabled you could also just use a PIPE and read the output of the DIR command.

So something like:

data want;
  infile "dir \\My Folder" pipe truncover;
  input .... ;
run;

 

s_lassen
Meteorite | Level 14

I think the following code is pretty dangerous:

/* Get the macro straight from GitHub */ 
filename code url "https://raw.githubusercontent.com/SASJedi/sas-macros/master/findfiles.sas";
%include code;

and should never be accepted in a serious working environment. I assume (but that is only an assumption) that you are a nice guy that would never put malicious code into your programs on github, but if you yourself are equally careless about running code that you have not even looked at, there is a serious risk that somebody else will hack your computer and plant malicious code in your programs.

 

Imported code should be revisited before running, and should be copied to a local file to secure against unauthorized change.

SASJedi
Ammonite | Level 13

That is a very valid observation, @s_lassen. I trust this code because I wrote and maintain it, but others should review before execution.

I'll modify the solution to use PROC HTTP to first download the file, giving the user the opportunity to review the code before running it. Thanks for keeping me honest!

Mark

 

Check out my Jedi SAS Tricks for SAS Users

hackathon24-white-horiz.png

The 2025 SAS Hackathon has begun!

It's finally time to hack! Remember to visit the SAS Hacker's Hub regularly for news and updates.

Latest Updates

How to Concatenate Values

Learn how use the CAT functions in SAS to join values from multiple variables into a single value.

Find more tutorials on the SAS Users YouTube channel.

SAS Training: Just a Click Away

 Ready to level-up your skills? Choose your own adventure.

Browse our catalog!

Discussion stats
  • 6 replies
  • 2827 views
  • 4 likes
  • 4 in conversation