FindFile (FF)
A replacement for find
(and fd
) that's simple to use, but much more powerful.
This repo is currently under active development, so most documentation is out of date or speculative.
Here's some examples of things I want to eventually support
- list all files in a directory:
ff 'isfile && depth=1'
- make a "tree" of files and their directories:
ff -n 'print "\t"*depth_from(start), basename'
- find all files that're at least 1 gig or are newer than 10 days ago:
ff 'size > 1g || modify > -10d'
- add the suffix
-YYYY-MM-DD
to all files but keep the extension:ff -n 'isfile && mv(file, "{dir}{base}-{ymd_date}.{suffix})'
- find files newer than 10 days with the enclosing folder is
log
:ff 'isfile && modify > -10d && basename(parent) = "log"'
- find all files that contain "hello" and "world", possibly on separate lines:
ff 'contents =~ /hello/ && contents =~ /world/'
- find the largest folder by its immediate files: (
${}
is run at script end):ff -n '${print maxdir} dirsize > dirsize(maxdir) then maxdir=dirsize'
Keywords
true
equivalent to1
false
equivalent to0
if else while continue break def return skip
/next
Functions
If a function takes no arguments, you can just omit the parens. eg file?
is the same as file?()
,
as file?()
is equivalent to file?(path)
.
Querying Info
name and args | aliases | what it does |
---|---|---|
file?(p=path) |
f? |
Returns whether p is a file. |
directory?(p=path) |
d? dir? |
Returns whether p is a directory. |
executable?(p=path) |
e? exe? |
Returns whether p is an executable. |
symlink?(p=path) |
s? sym? |
Returns whether p is a symlink. |
binary?(p=path) |
b? bin? |
Returns whether p is a binary file. |
gitignore?(p=path) |
gi? |
Returns whether p is ignored by a gitignore file. |
hidden?(p=path) |
gi? |
Returns whether p is starts with . |
ok?(msg) |
Prints msg out, then asks for confirmation. |
|
macos(...) |
future idea: stuff like macos tags or whatnot |
Path-Related functions
| root()
| r
| The root folder we started looking at |
| path()
| p
| The current path |
| dirname(p=path)
| d
dir
parent
| The parent directory |
| extname(p=path)
| e
ext
extension
| The extension, without a .
if it's present |
| extnamed(p=path)
| ed
extd
extensiond
| The extension, including the .
if it's present. |
| basename(p=path)
| b
bn
base
| Everything but the parent directory of p
|
| stemname(p=path)
| s
stem
| basename
, except without an extension (if present) |
Misc
| print(...)
| pr
| Prints its arguments out (with nothing between them) followed by a newline |
| printn(...)
| prn
| Prints its arguments out (with nothing between them) without a newline |
| next
| skip
| Ignores the current argument and continues onwards |
| exit(status)
| quit
| stops the entire script |
| pwd()
| Current working directory |
| depth(src=path, dst=root)
| how many directories down we are from the dst
. |
| date(<...>)
| The current date foromatted in a time |
| sleep(<...>)
| Sleeps |
Executable functions
Some of these functions are "destructive" (such as mv
): If a destructive file would overwrite another one, it'll check the command line arguments to see what to do (--interactive
implies always ask, --force
implies never ask; if neither is given, --force
is assumed.) You can use <fn>i
to always do interactive or <fn>f
to always force (like mvf
).
All these must be called with parens (maybe?)
name and args | what it does |
---|---|
exec(...) | .. |
mv{,f,i}(src=path, dst) |
Moves src to dst ; only confirms if overwriting a file when interactive |
rm{,f,i}(src=path) |
Removes the file at src ; always confirms when interactive. If given an empty directory, rm acts like rmdir . |
rmr{,f,i}(src=path) |
Removes the file at src , recursively; always confirms when interactive |
cp{,f,i}(src=path, dst) |
Copies src to dst only confirms if overwriting a file when interactive |
ln... |
|
mkdir(p) |
Creates a directory at p ; It'll also make all parent directories. |
touch(src=path) |
Creates a directory at p ; It'll also make all parent directories. |
IDEAS:
there was a problem with moving your file. What would you like to do:
(Q)uit: Stop the entire program
(C)ontinue: continue onwards,
(R)etry: try it again (maybe after you fix something)
(S)hell: Drop you into a shell where $cpath
is the variable for the current path