We're feeding a list of files from `lsdiff` to `filterdiff` via `xargs`.
That list is newline separated, and looks something like this:
```
$ lsdiff <(curl -s 1e86a219f5.patch)
a/README.md
b/files/The Answer to the Ultimate Question of Life, The Universe, and Everything.txt
```
However, if the list contains files with apostrophes in it, xargs freaks
out:
```
$ echo "there's an apostrophe here" | xargs -I{} python -c "import sys; print(sys.argv)" {}
xargs: unmatched single quote; by default quotes are special to xargs unless you use the -0 option
```
The fix is simple, just explicitly specify a delimiter:
```
$ echo "there's an apostrophe here" | xargs -I{} --delimiter='\n' python -c "import sys; print(sys.argv)" {}
['-c', "there's an apostrophe here"]
```
I added 2 tests here:
- `nix-build -A pkgs.tests.fetchpatch.fileWithApostrophe` fails without the code change.
- `nix-build -A pkgs.tests.fetchpatch.fileWithSpace` passes both before
and after this change, but I wanted to add it to prove that I didn't
break anything.
122 lines
3.5 KiB
Nix
122 lines
3.5 KiB
Nix
# This function downloads and normalizes a patch/diff file.
|
|
# This is primarily useful for dynamically generated patches,
|
|
# such as GitHub's or cgit's, where the non-significant content parts
|
|
# often change with updating of git or cgit.
|
|
# stripLen acts as the -p parameter when applying a patch.
|
|
|
|
{
|
|
lib,
|
|
fetchurl,
|
|
patchutils,
|
|
}:
|
|
|
|
{
|
|
relative ? null,
|
|
stripLen ? 0,
|
|
decode ? "cat", # custom command to decode patch e.g. base64 -d
|
|
extraPrefix ? null,
|
|
excludes ? [ ],
|
|
includes ? [ ],
|
|
revert ? false,
|
|
postFetch ? "",
|
|
nativeBuildInputs ? [ ],
|
|
...
|
|
}@args:
|
|
let
|
|
args' =
|
|
if relative != null then
|
|
{
|
|
stripLen = 1 + lib.length (lib.splitString "/" relative) + stripLen;
|
|
extraPrefix = lib.optionalString (extraPrefix != null) extraPrefix;
|
|
}
|
|
else
|
|
{
|
|
inherit stripLen extraPrefix;
|
|
};
|
|
in
|
|
let
|
|
inherit (args') stripLen extraPrefix;
|
|
in
|
|
lib.throwIfNot (excludes == [ ] || includes == [ ])
|
|
"fetchpatch: cannot use excludes and includes simultaneously"
|
|
fetchurl
|
|
(
|
|
{
|
|
nativeBuildInputs = [ patchutils ] ++ nativeBuildInputs;
|
|
postFetch =
|
|
''
|
|
tmpfile="$TMPDIR/patch"
|
|
|
|
if [ ! -s "$out" ]; then
|
|
echo "error: Fetched patch file '$out' is empty!" 1>&2
|
|
exit 1
|
|
fi
|
|
|
|
set +e
|
|
${decode} < "$out" > "$tmpfile"
|
|
if [ $? -ne 0 ] || [ ! -s "$tmpfile" ]; then
|
|
echo 'Failed to decode patch with command "'${lib.escapeShellArg decode}'"' >&2
|
|
echo 'Fetched file was (limited to 128 bytes):' >&2
|
|
od -A x -t x1z -v -N 128 "$out" >&2
|
|
exit 1
|
|
fi
|
|
set -e
|
|
mv "$tmpfile" "$out"
|
|
|
|
lsdiff \
|
|
${lib.optionalString (relative != null) "-p1 -i ${lib.escapeShellArg relative}/'*'"} \
|
|
"$out" \
|
|
| sort -u | sed -e 's/[*?]/\\&/g' \
|
|
| xargs -I{} --delimiter='\n' \
|
|
filterdiff \
|
|
--include={} \
|
|
--strip=${toString stripLen} \
|
|
${
|
|
lib.optionalString (extraPrefix != null) ''
|
|
--addoldprefix=a/${lib.escapeShellArg extraPrefix} \
|
|
--addnewprefix=b/${lib.escapeShellArg extraPrefix} \
|
|
''
|
|
} \
|
|
--clean "$out" > "$tmpfile"
|
|
|
|
if [ ! -s "$tmpfile" ]; then
|
|
echo "error: Normalized patch '$tmpfile' is empty (while the fetched file was not)!" 1>&2
|
|
echo "Did you maybe fetch a HTML representation of a patch instead of a raw patch?" 1>&2
|
|
echo "Fetched file was:" 1>&2
|
|
cat "$out" 1>&2
|
|
exit 1
|
|
fi
|
|
|
|
filterdiff \
|
|
-p1 \
|
|
${builtins.toString (builtins.map (x: "-x ${lib.escapeShellArg x}") excludes)} \
|
|
${builtins.toString (builtins.map (x: "-i ${lib.escapeShellArg x}") includes)} \
|
|
"$tmpfile" > "$out"
|
|
|
|
if [ ! -s "$out" ]; then
|
|
echo "error: Filtered patch '$out' is empty (while the original patch file was not)!" 1>&2
|
|
echo "Check your includes and excludes." 1>&2
|
|
echo "Normalized patch file was:" 1>&2
|
|
cat "$tmpfile" 1>&2
|
|
exit 1
|
|
fi
|
|
''
|
|
+ lib.optionalString revert ''
|
|
interdiff "$out" /dev/null > "$tmpfile"
|
|
mv "$tmpfile" "$out"
|
|
''
|
|
+ postFetch;
|
|
}
|
|
// builtins.removeAttrs args [
|
|
"relative"
|
|
"stripLen"
|
|
"decode"
|
|
"extraPrefix"
|
|
"excludes"
|
|
"includes"
|
|
"revert"
|
|
"postFetch"
|
|
"nativeBuildInputs"
|
|
]
|
|
)
|