Skip to content

Commit

Permalink
fix: Support patches without a prefix by determining the prefix depth (
Browse files Browse the repository at this point in the history
…ocaml#10885)

Signed-off-by: Marek Kubica <[email protected]>
  • Loading branch information
Leonidas-from-XIV authored Sep 9, 2024
1 parent 7ec38ff commit b3bd89f
Show file tree
Hide file tree
Showing 2 changed files with 65 additions and 30 deletions.
87 changes: 64 additions & 23 deletions src/dune_patch/dune_patch.ml
Original file line number Diff line number Diff line change
Expand Up @@ -22,19 +22,24 @@ let re =
let junk = Re.rep Re.notnl in
Re.compile
@@ Re.seq
[ line [ Re.str {|--- |}; Re.opt (Re.str "a/"); filename; junk ]
; followed_by_line [ Re.str {|+++ |}; Re.opt (Re.str "b/"); filename; junk ]
[ line [ Re.str {|--- |}; filename; junk ]
; followed_by_line [ Re.str {|+++ |}; filename; junk ]
]
;;

module Patch = struct
(* CR-someday alizter: more parsed information about the patch should go here.
Eventually we wish to replace the patch command inside the patch action with a pure
OCaml implementation. *)
type t =
type operation =
| New of Path.Local.t
| Delete of Path.Local.t
| Replace of Path.Local.t

type t =
{ prefix : int
; op : operation
}
end

let patches_of_string patch_string =
Expand All @@ -50,13 +55,35 @@ let patches_of_string patch_string =
None
| true, false ->
(* New file *)
Some (Patch.New (Path.Local.of_string new_file))
let path = Path.Local.of_string new_file in
let prefix, path =
match Path.Local.split_first_component path with
| Some ("b", path) -> 1, path
| _ -> 0, path
in
Some { Patch.op = Patch.New path; prefix }
| false, true ->
(* Delete file *)
Some (Patch.Delete (Path.Local.of_string old_file))
let path = Path.Local.of_string old_file in
let prefix, path =
match Path.Local.split_first_component path with
| Some ("a", path) -> 1, path
| _ -> 0, path
in
Some { Patch.op = Patch.Delete path; prefix }
| false, false ->
let old_path = Path.Local.of_string old_file in
let new_path = Path.Local.of_string new_file in
let new_path, prefix =
match
( Path.Local.split_first_component old_path
, Path.Local.split_first_component new_path )
with
| Some ("a", _old_path), Some ("b", new_path) -> new_path, 1
| _, _ -> new_path, 0
in
(* Replace file *)
Some (Patch.Replace (Path.Local.of_string new_file)))
Some { Patch.op = Patch.Replace new_path; prefix })
;;

let prog =
Expand All @@ -69,29 +96,43 @@ let prog =
let exec display ~patch ~dir ~stderr =
let open Fiber.O in
let* () = Fiber.return () in
(* Read the patch file. *)
Io.read_file patch
(* Collect all the patches. *)
|> patches_of_string
(* Depending on whether it is creating a new file or modifying an existing file
prepare the files that will be modified accordingly. For modifying existing files
this means materializing any symlinks or hardlinks. *)
|> List.iter ~f:(function
let patches =
patch
(* Read the patch file. *)
|> Io.read_file
(* Collect all the patches. *)
|> patches_of_string
in
List.iter patches ~f:(fun { Patch.op; prefix = _ } ->
(* Depending on whether it is creating a new file or modifying an existing file
prepare the files that will be modified accordingly. For modifying existing files
this means materializing any symlinks or hardlinks. *)
match op with
| Patch.New _ | Delete _ -> ()
| Replace file ->
let file = Path.append_local dir file in
let temp = Path.extend_basename file ~suffix:".for_patch" in
Io.copy_file ~src:file ~dst:temp ();
Path.rename temp file);
Process.run
~dir
~display
~stdout_to:Process.(Io.null Out)
~stderr_to:stderr
~stdin_from:Process.(Io.null In)
Process.Failure_mode.Strict
(Lazy.force prog)
[ "-p1"; "-i"; Path.reach_for_running ~from:dir patch ]
match patches with
| [] -> User_error.raise [ Pp.text "No patches in patch file detected" ]
| { Patch.op = _; prefix } :: patches ->
(match
List.for_all ~f:(fun { Patch.op = _; prefix = p } -> Int.equal prefix p) patches
with
| false ->
User_error.raise [ Pp.text "Different prefix lengths in file unsupported" ]
| true ->
let p_flag = sprintf "-p%d" prefix in
Process.run
~dir
~display
~stdout_to:Process.(Io.null Out)
~stderr_to:stderr
~stdin_from:Process.(Io.null In)
Process.Failure_mode.Strict
(Lazy.force prog)
[ p_flag; "-i"; Path.reach_for_running ~from:dir patch ])
;;

module Spec = struct
Expand Down
8 changes: 1 addition & 7 deletions test/expect-tests/dune_patch/dune_patch_tests.ml
Original file line number Diff line number Diff line change
Expand Up @@ -224,13 +224,7 @@ let%expect_test "Using a patch from 'diff' with a timestamp" =
let%expect_test "patching a file without prefix" =
test [ "foo.ml", "This is wrong\n" ] ("foo.patch", no_prefix);
check "foo.ml";
[%expect.unreachable]
[@@expect.uncaught_exn
{|
(Dune_util__Report_error.Already_reported)
Trailing output
---------------
Command exited with code 1. |}]
[%expect {| This is right |}]
;;

let%expect_test "patching files with spaces" =
Expand Down

0 comments on commit b3bd89f

Please sign in to comment.