Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

FORFILES date -after- (date calc in cmd file)

I'd like to use FORFILES in a cmd file, to act on all files after a given date thru "today". I can use something like forfiles /d +07/10/2013 /c "cmd /c echo @fname" to act on everything after 7/10/13, but what I want is to just be able to calculate instead from 90 days before "today".

Is there a syntax for the date calculation that will work in a cmd file that will let me specify "x days before today" to feed into FORFILES?

I prefer to not use VBS (and found a code snippet that would work in VBS), though I could alternatively re-write my script for Powershell, but ideally I want to stick with cmd.

To clarify, "-90" would find all files older than 90 days; "+90" would find all files newer than 90 days -after- today (which is fundamentally useless, as files are rarely written with future dates), and "+7/30/2013" will find all files newer than 7/30/2013. I want that last time period, preferably able to take a number-of-days variable passed to the CMD file, that would say "after x number of days before today", i.e. "in the last x days". So instead of using the hard-coded date as shown above, I want to be able to calculate that date within the cmd file.

like image 968
Debra Avatar asked Jan 13 '23 04:01

Debra


2 Answers

Here is a work-around for the design flaw of forfiles concerning /D +dd days, supposing that no items can be modified in the future, and which is based on two nested forfiles loops and relies on the fact that forfiles sets the ErrorLevel in case no items match the provided search criteria:

rem Define minimum and maximum age in days here (0 means today):
set /A MINAGE=0, MAXAGE=90

set "MAXAGE=%MAXAGE:*-=%" & set "MINAGE=%MINAGE:*-=%" & set /A MAXINC=MAXAGE+1
> nul forfiles /D -%MINAGE% /C "cmd /C if @isdir==FALSE 2> nul forfiles /M @file /D -%MAXINC% || > con echo @fdate  @file"

The outer forfiles loop iterates through items that are at least as old as given by variable MINAGE. The inner forfiles loop, which iterates once at most as it recieves the iterated file @file from the outer loop, returns the same file if it is also at least as old as MAXINC (equals MAXAGE plus 1); if it is not, forfiles sets the ErrorLevel to 1, which in turn is captured by the || operator that executes the following command, namely echo, only in case ErrorLevel has been set to a non-zero value.

Both MINAGE and MAXAGE must not be negative numbers (the commands set "MAXAGE=%MAXAGE:*-=%" and set "MINAGE=%MINAGE:*-=%" remove the minus sign in case). The interim variable MAXINC has been introduced in order to include the age specified by MAXAGE itself into the returned result.

The redirection > nul prevents the outer forfiles from returning empty lines and the inner one from returning items that fulfil its search criteria (because we are interested in those that do not). 2> nul prevents the inner forfiles loop from returning error messages in case its search criteria are violated. > con overrides > nul for the echo command to actually return the required items.

The if @isdir==FALSE part filters out directories so that only files are processed. Change FALSE to TRUE if you want to process only directories; remove it completely if you want to process both.

like image 98
aschipfl Avatar answered Jan 18 '23 21:01

aschipfl


In PowerShell you could do something like this:

$refdate = (Get-Date).AddDays(-90)
Get-ChildItem | Where-Object {
    $_.LastWriteTime -ge $refdate
} | Select-Object -Expand Name
like image 26
Ansgar Wiechers Avatar answered Jan 18 '23 22:01

Ansgar Wiechers