package git-unix
Virtual package to install and configure ocaml-git's Unix backend
Install
Dune Dependency
Authors
Maintainers
Sources
git-3.7.0.tbz
sha256=ee30231a00f0a3548d3c2e1266537afe5f2a877d2c24369ed771cb91ec47610d
sha512=e0b0d209cf35edead52916fb215ef83739f0be00501b6898a91a14503544d96718b1c248febc1abbe9bf5bd9e07eb0ea3ea5f120c32065fd32a20632b534c481
doc/src/git-unix/git_unix.ml.html
Source file git_unix.ml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749
(* * Copyright (c) 2013-2017 Thomas Gazagnaire <thomas@gazagnaire.org> * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. *) open Lwt.Infix let ( >>? ) x f = let open Lwt.Infix in x >>= function Ok x -> f x | Error err -> Lwt.return_error err let ( <.> ) f g x = f (g x) (* XXX(dinosaure): NOTE! [Git_unix] wants to provide an implementation * which can fit into required modules by [Git.Store] __and__ the usual * layout of a non-bare Git repository. * * Nothing was done about performances - and provided implementations * are surely not the best. If someone wants a _fast_ implementation * of Git, this is the first entry-point. *) module Fold = struct let src = Logs.Src.create "git-unix.fold" ~doc:"logs git-unix's fold event" module Log = (val Logs.src_log src : Logs.LOG) let always x _ = x let rec contents ?(dotfiles = false) ?(rel = false) dir = let rec readdir dh acc = Lwt.catch (fun () -> Lwt_unix.readdir dh >>= Lwt.return_some) (fun _exn -> Lwt.return_none) >>= function | None -> Lwt.return acc | Some (".." | ".") -> readdir dh acc | Some f when dotfiles || not (f.[0] = '.') -> ( match Fpath.of_string f with | Ok f -> readdir dh ((if rel then f else Fpath.(dir // f)) :: acc) | Error (`Msg _) -> (* ignore *) readdir dh acc) | Some _ -> readdir dh acc in Lwt.catch (fun () -> Lwt_unix.opendir (Fpath.to_string dir) >>= fun dh -> readdir dh [] >>= fun res -> Lwt_unix.closedir dh >>= fun () -> Lwt.return res) (function | Unix.Unix_error (Unix.EINTR, _, _) -> contents ~dotfiles ~rel dir | Unix.Unix_error (err, _, _) -> let err = Fmt.str "directory contents %a: %s" Fpath.pp dir (Unix.error_message err) in Log.err (fun m -> m "%s" err); Lwt.return [] | exn -> Lwt.fail exn) let do_traverse_fun = function | `Any -> always true | `None -> always false | `Sat sat -> sat let rec exists path = Lwt.catch (fun () -> Lwt_unix.stat (Fpath.to_string path) >>= fun _ -> Lwt.return true) @@ function | Unix.Unix_error ((Unix.ENOENT | Unix.ENOTDIR), _, _) -> Lwt.return false | Unix.Unix_error (Unix.EINTR, _, _) -> exists path | exn -> Lwt.fail exn let rec file_exists path = Lwt.catch (fun () -> Lwt_unix.stat (Fpath.to_string path) >>= fun stat -> Lwt.return (stat.Unix.st_kind = Unix.S_REG)) @@ function | Unix.Unix_error (Unix.ENOENT, _, _) -> Lwt.return false | Unix.Unix_error (Unix.EINTR, _, _) -> file_exists path | exn -> Lwt.fail exn let rec dir_exists path = Lwt.catch (fun () -> Lwt_unix.stat (Fpath.to_string path) >>= fun stat -> Lwt.return (stat.Unix.st_kind = Unix.S_DIR)) @@ function | Unix.Unix_error (Unix.ENOENT, _, _) -> Lwt.return false | Unix.Unix_error (Unix.EINTR, _, _) -> dir_exists path | exn -> Lwt.fail exn let is_element_fun = function | `Any -> exists | `Files -> file_exists | `Dirs -> dir_exists | `Sat sat -> sat let readdir_fun = let readdir d = try Sys.readdir (Fpath.to_string d) with _exn -> [||] in Lwt.return <.> Array.to_list <.> readdir let fold ?(dotfiles = false) ?(elements = `Any) ?(traverse = `Any) f acc paths = let process () = let do_traverse = do_traverse_fun traverse in let is_element = is_element_fun elements in let is_dir = dir_exists in let readdir = readdir_fun in let process_path p (acc, to_traverse) = Lwt.both (is_element p) (is_dir p) >>= function | false, true when do_traverse p -> Lwt.return (acc, p :: to_traverse) | true, true when do_traverse p -> Lwt.both (f p acc) (Lwt.return (p :: to_traverse)) | true, _ -> Lwt.both (f p acc) (Lwt.return to_traverse) | _ -> Lwt.return (acc, to_traverse) in let dir_child d acc bname = if (not dotfiles) && bname.[0] = '.' then Lwt.return acc else process_path Fpath.(d / bname) acc in let rec loop acc = function | (d :: ds) :: up -> readdir d >>= fun childs -> Lwt_list.fold_left_s (dir_child d) (acc, []) childs >>= fun (acc, to_traverse) -> loop acc (to_traverse :: ds :: up) | [ [] ] -> Lwt.return acc | [] :: up -> loop acc up | _ -> assert false in let init acc p = let base = Fpath.(basename @@ normalize p) in if (not dotfiles) && base.[0] = '.' then Lwt.return acc else process_path p acc in Lwt_list.fold_left_s init (acc, []) paths >>= fun (acc, to_traverse) -> loop acc [ to_traverse ] in process () let fold ?dotfiles ?elements ?traverse f acc d = contents d >>= fold ?dotfiles ?elements ?traverse f acc end module Minor_heap (Digestif : Digestif.S) = struct let src = Logs.Src.create "git-unix.minor" ~doc:"logs git-unix's minor heap event" module Log = (val Logs.src_log src : Logs.LOG) type t = Fpath.t (* [.git/objects] *) type uid = Digestif.t type error = [ `Not_found of Digestif.t | `Msg of string ] let pp_error ppf = function | `Not_found uid -> Fmt.pf ppf "%a not found" Digestif.pp uid | `Msg err -> Fmt.string ppf err type +'a fiber = 'a Lwt.t let split uid = let hex = Digestif.to_hex uid in String.sub hex 0 2, String.sub hex 2 ((Digestif.digest_size * 2) - 2) let rec exists root uid = let hd, tl = split uid in let path = Fpath.(root / hd / tl) in let process () = Lwt_unix.stat (Fpath.to_string path) >>= fun _ -> Lwt.return true in let error = function | Unix.Unix_error (Unix.EACCES, _, _) -> Lwt.return false | Unix.Unix_error (Unix.ENOENT, _, _) -> Lwt.return false | Unix.Unix_error (Unix.EINTR, _, _) -> exists root uid | exn -> Lwt.fail exn in Lwt.catch process error let rec length root uid = let hd, tl = split uid in let path = Fpath.(root / hd / tl) in let process () = Lwt_unix.LargeFile.stat (Fpath.to_string path) >>= fun stat -> Lwt.return_ok stat.Unix.LargeFile.st_size in let error = function | Unix.Unix_error (Unix.EACCES, _, _) -> Lwt.return_error (`Not_found uid) | Unix.Unix_error (Unix.ENOENT, _, _) -> Lwt.return_error (`Not_found uid) | Unix.Unix_error (Unix.EINTR, _, _) -> length root uid | exn -> Lwt.fail exn in Lwt.catch process error let rec map root uid ~pos len = if pos < 0L || len < 0 then invalid_arg "Minor_heap.map: invalid bounds"; let hd, tl = split uid in let path = Fpath.(root / hd / tl) in let rec process () = Lwt_unix.LargeFile.stat (Fpath.to_string path) >>= fun stat -> try let len = if Int64.add pos (Int64.of_int len) > stat.Lwt_unix.LargeFile.st_size then Int64.to_int (Int64.sub stat.Lwt_unix.LargeFile.st_size pos) else len in let fd = Unix.openfile (Fpath.to_string path) Unix.[ O_RDONLY ] 0o400 in let rs = Mmap.V1.map_file fd ~pos Bigarray.char Bigarray.c_layout false [| len |] in Unix.close fd; Lwt.return (Bigarray.array1_of_genarray rs) with | Unix.Unix_error (Unix.EACCES, _, _) | Unix.Unix_error (Unix.ENOENT, _, _) -> Lwt.return Bigstringaf.empty | Unix.Unix_error (Unix.EINTR, _, _) -> process () in let error = function | Unix.Unix_error (Unix.EACCES, _, _) -> Lwt.return Bigstringaf.empty | Unix.Unix_error (Unix.ENOENT, _, _) -> Lwt.return Bigstringaf.empty | Unix.Unix_error (Unix.EINTR, _, _) -> map root uid ~pos len | exn -> Lwt.fail exn in Lwt.catch process error let append root uid payload = (* XXX(dinosaure): [irmin] expects an atomicity about the creation of [uid]. * This does not mean an atomicty about the creation and the filling of [uid]! * This assumption requires an atomicity about [mkdir / openfile]. * * A problem /can/ occur when we use [Lwt_unix.mkdir] which can /yield/. In such * case, [uid] still does not exist. However, [irmin] expects, at least, the existence * of it (whatever if we wrote entirely, partially or nothing). * * We should optimize this function but we need to keep this assumption - * which was not really clear. * * More precisely, a data-race condition exists when [irmin] wants to save 2 times the * same object from 2 different fibers. One can create the given object partially and * the other can try to read it, if such case appear, the second considers the object * as an non-existent object (but the second fiber is may be used to update a reference). * Finally, we assert the requirement of the atomicity about [append{v}]. However, the * bug discovered is really strange (replication of the bug can be done with [irmin-unix], * test [GIT.021]) *) Log.debug (fun m -> m "Minor.append %a" Digestif.pp uid); let hd, tl = split uid in let path = Fpath.(root / hd / tl) in let fiber () = let open Rresult in Bos.OS.Dir.create Fpath.(root / hd) >>= fun _ -> Bos.OS.File.write path (Bigstringaf.to_string payload) in Lwt.return (fiber ()) let f emitter (tmp, payloads) = let rec go pos = function | [] -> emitter None; Rresult.R.ok () | src :: rest as payloads -> let len = min (Bytes.length tmp) (Bigstringaf.length src - pos) in Bigstringaf.blit_to_bytes src ~src_off:pos tmp ~dst_off:0 ~len; emitter (Some (tmp, 0, len)); let pos = pos + len in if pos = Bigstringaf.length src then go 0 rest else go pos payloads in go 0 payloads let appendv root uid payloads = Log.debug (fun m -> m "Minor.appendv %a" Digestif.pp uid); let hd, tl = split uid in let path = Fpath.(root / hd / tl) in let fiber () = let open Rresult in Bos.OS.Dir.create Fpath.(root / hd) >>= fun _ -> Bos.OS.File.with_output path f (Bytes.create De.io_buffer_size, payloads) in Lwt.return (Rresult.R.join (fiber ())) let list root = let f x r = match List.rev (Fpath.segs x) with | tl :: hd :: _ -> let uid = Digestif.of_hex (hd ^ tl) in Lwt.return (uid :: r) | _ -> Lwt.return r in let elements path = match List.rev (Fpath.segs path) with | tl :: hd :: _ -> ( match Digestif.of_hex (hd ^ tl) with | _ -> Fold.file_exists path | exception _ -> Lwt.return false) | _ -> Lwt.return false in Fold.fold ~dotfiles:false ~elements:(`Sat elements) f [] root let reset root = list root >>= fun lst -> let rec f uid = let hd, tl = split uid in let path = Fpath.(root / hd / tl) in Lwt.catch (fun () -> Lwt_unix.unlink (Fpath.to_string path)) (function | Unix.Unix_error (Unix.EINTR, _, _) -> f uid | Unix.Unix_error (Unix.ENOENT, _, _) -> Lwt.return_unit | exn -> Lwt.fail exn) in Lwt_list.iter_p f lst >>= Lwt.return_ok end module Major_heap = struct let src = Logs.Src.create "git-unix.major" ~doc:"logs git-unix's major heap event" module Log = (val Logs.src_log src : Logs.LOG) type t = Fpath.t (* [.git/objects/pack] *) type uid = Fpath.t type 'a rd = < rd : unit ; .. > as 'a type 'a wr = < wr : unit ; .. > as 'a type 'a mode = | Rd : < rd : unit > mode | Wr : < wr : unit > mode | RdWr : < rd : unit ; wr : unit > mode type 'a fd = Lwt_unix.file_descr type error = [ `Not_found of uid ] type +'a fiber = 'a Lwt.t let pp_error : error Fmt.t = fun ppf -> function | `Not_found uid -> Fmt.pf ppf "%a not found" Fpath.pp uid (* XXX(dinosaure): currently, [Major_heap] has a read and a write access due to [append] which is only call by [Store.Sync]. We should provide 2 [Major_heap]: - one used by [Store] which is read-only - the second used by [Store.Sync] which is write-only A [mode] is better (to avoid duplicate) and safe. *) let create : type a. ?trunc:bool -> mode:a mode -> t -> uid -> (a fd, error) result Lwt.t = fun ?(trunc = true) ~mode root path -> let path = Fpath.(root // path) in let flags, perm = match mode with | Rd -> Unix.[ O_RDONLY ], 0o400 | Wr -> Unix.[ O_WRONLY; O_CREAT; O_APPEND ], 0o600 | RdWr -> Unix.[ O_RDWR; O_CREAT; O_APPEND ], 0o600 in let flags = if trunc then Unix.O_TRUNC :: flags else flags in let rec process () = Lwt_unix.openfile (Fpath.to_string path) flags perm >>= fun fd -> Lwt.return_ok fd and error = function | Unix.Unix_error (Unix.ENOENT, _, _) | Unix.Unix_error (Unix.EACCES, _, _) -> Printexc.print_backtrace stdout; flush stdout; Log.err (fun m -> m "%a does not exists." Fpath.pp path); Lwt.return_error (`Not_found path) | Unix.Unix_error (Unix.EINTR, _, _) -> Lwt.catch process error | exn -> Lwt.fail exn in Lwt.catch process error let map : t -> [> `Rd ] fd -> pos:int64 -> int -> Bigstringaf.t = fun _ fd ~pos len -> let fd = Lwt_unix.unix_file_descr fd in let payload = Mmap.V1.map_file fd ~pos Bigarray.char Bigarray.c_layout false [| len |] in Bigarray.array1_of_genarray payload let close _ fd = let rec process () = Lwt_unix.close fd >>= fun () -> Lwt.return_ok () and error = function | Unix.Unix_error (Unix.EINTR, _, _) -> Lwt.catch process error | exn -> Lwt.fail exn in Lwt.catch process error let length fd = let rec process () = Lwt_unix.LargeFile.fstat fd >>= fun st -> Lwt.return st.Unix.LargeFile.st_size and error = function | Unix.Unix_error (Unix.EINTR, _, _) -> Lwt.catch process error | exn -> Lwt.fail exn in Lwt.catch process error let list root = let res = let open Rresult in Bos.OS.Dir.contents ~dotfiles:false ~rel:true root >>| List.filter (Fpath.has_ext "pack") in match res with | Ok lst -> Lwt.return lst | Error (`Msg err) -> Log.warn (fun m -> m "Major.list: %s" err); Lwt.return [] let reset root = list root >>= fun lst -> let rec f path = Lwt.catch (fun () -> Lwt_unix.unlink (Fpath.to_string path) >>= fun () -> Lwt_unix.unlink (Fpath.to_string (Fpath.set_ext "idx" path))) (function | Unix.Unix_error (Unix.EINTR, _, _) -> f path | _exn -> Lwt.return_unit) in Lwt_list.iter_p f lst >>= Lwt.return_ok let move root ~src ~dst = let src = Fpath.(root // src) in let dst = Fpath.(root // dst) in Lwt_unix.rename (Fpath.to_string src) (Fpath.to_string dst) >>= fun () -> Lwt.return_ok () let append : t -> [> `Wr ] fd -> string -> unit fiber = fun _ fd str -> let rec go (off, len) = Lwt_unix.write_string fd str off len >>= fun len' -> if len = len' then Lwt.return () else go (off + len', len - len') in go (0, String.length str) end module Unix = struct include Unix let mkdir ?(path = true) ?(mode = 0o755) dir = let rec exists dir = Lwt.catch (fun () -> Lwt_unix.stat (Fpath.to_string dir) >>= fun stat -> Lwt.return_ok (stat.Unix.st_kind = Unix.S_DIR)) (function | Unix.Unix_error (Unix.ENOENT, _, _) -> Lwt.return_ok false | Unix.Unix_error (Unix.EINTR, _, _) -> exists dir | exn -> Lwt.fail exn) in let rec mkdir d mode = Lwt.catch (fun () -> Lwt_unix.mkdir (Fpath.to_string d) mode >>= Lwt.return_ok) (function | Unix.Unix_error (Unix.EEXIST, _, _) -> Lwt.return_ok () | Unix.Unix_error (Unix.EINTR, _, _) -> mkdir d mode | Unix.Unix_error (e, _, _) -> if d = dir then Lwt.return_error @@ Rresult.R.msgf "create directory %a: %s" Fpath.pp d (Unix.error_message e) else Lwt.return_error @@ Rresult.R.msgf "create directory %a: %a: %s" Fpath.pp dir Fpath.pp d (Unix.error_message e) | exn -> Lwt.fail exn) in exists dir >>? function | true -> Lwt.return_ok false | false -> ( match path with | false -> mkdir dir mode >>? fun () -> Lwt.return_ok false | true -> let rec dirs_to_create p acc = exists p >>? function | true -> Lwt.return_ok acc | false -> dirs_to_create (Fpath.parent p) (p :: acc) in let rec create_them dirs () = match dirs with | dir :: dirs -> mkdir dir mode >>? create_them dirs | [] -> Lwt.return_ok () in dirs_to_create dir [] >>? fun dirs -> create_them dirs () >>? fun () -> Lwt.return_ok true) end module Reference_heap = struct let src = Logs.Src.create "git-unix.reference" ~doc:"logs git-unix's reference heap event" module Log = (val Logs.src_log src : Logs.LOG) type +'a fiber = 'a (* XXX(dinosaure): ensure the atomicity. *) type t = Fpath.t (* [.git] *) type error = [ `Not_found of Git.Reference.t | `Msg of string ] let pp_error ppf = function | `Not_found refname -> Fmt.pf ppf "%a not found" Git.Reference.pp refname | `Msg err -> Fmt.string ppf err let safely_unlink filename = let rec unlink filename = try Unix.unlink filename with | Unix.Unix_error (Unix.ENOENT, _, _) -> () | Unix.Unix_error (Unix.EINTR, _, _) -> unlink filename | Unix.Unix_error (err, _, _) -> Fmt.failwith "unlink %s: %s" filename (Unix.error_message err) in unlink filename let atomic_wr root refname str = Log.debug (fun m -> m "Writing %a: %S." Git.Reference.pp refname str); let path = List.fold_left Fpath.add_seg root (Git.Reference.segs refname) in let base, _ = Fpath.split_base path in let open Rresult in Bos.OS.Dir.create ~path:true base >>= fun _ -> Bos.OS.Dir.exists path >>= fun res -> (if res then Bos.OS.Dir.delete ~must_exist:false ~recurse:true path else R.ok ()) >>= fun () -> Bos.OS.File.tmp "git-reference-%s" >>= fun src -> Bos.OS.File.write src str >>= fun () -> let fd = Unix.openfile (Fpath.to_string src) Unix.[ O_WRONLY ] 0o644 in Unix.close fd; (* XXX(dinosaure): on Windows, [rename] requires that [path] does not * exist! *) if Sys.os_type = "Win32" then safely_unlink (Fpath.to_string path); Unix.rename (Fpath.to_string src) (Fpath.to_string path); R.ok () let atomic_rd root refname = Log.debug (fun m -> m "Reading %a." Git.Reference.pp refname); let path = List.fold_left Fpath.add_seg root (Git.Reference.segs refname) in let open Rresult in Bos.OS.File.exists path >>= function | true -> let fd = Unix.openfile (Fpath.to_string path) Unix.[ O_RDONLY ] 0o644 in let { Unix.st_size; _ } = Unix.fstat fd in let rs = Bytes.create st_size in let ln = Unix.read fd rs 0 st_size in assert (ln = st_size); Unix.close fd; R.ok (Bytes.unsafe_to_string rs) | false -> R.error (`Not_found refname) let atomic_rm root refname = Log.debug (fun m -> m "Deleting %a." Git.Reference.pp refname); let path = List.fold_left Fpath.add_seg root (Git.Reference.segs refname) in Bos.OS.File.delete path let list root = let f x r = match Fpath.rem_prefix root x with | Some x -> ( Log.debug (fun l -> l "%a exists into the store." Fpath.pp x); match Git.Reference.of_string (Fpath.to_string x) with | Ok x -> x :: r | Error _ -> r) | None -> assert false (* XXX(dinosaure): see [elements]. *) in let elements path = match Option.map Fpath.segs (Fpath.rem_prefix root path) with | Some ("objects" :: _) -> Ok false | Some [ "HEAD" ] -> Bos.OS.File.exists path | Some ("refs" :: _) -> Bos.OS.File.exists path | _ -> Ok false in Log.debug (fun l -> l "Listing references into %a." Fpath.pp root); match Bos.OS.Dir.fold_contents ~dotfiles:false ~elements:(`Sat elements) f [] root with | Ok lst -> lst | Error (`Msg err) -> Log.warn (fun m -> m "error when we listing references: %s" err); [] let reset root = let open Rresult in let lst = list root in let f refname = let path = List.fold_left Fpath.add_seg root (Git.Reference.segs refname) in match Bos.OS.Path.delete path with | Ok () -> () | Error (`Msg err) -> Log.warn (fun m -> m "error when we deleting %a: %s" Fpath.pp path err) in List.iter f lst; R.ok () end module Make (Digestif : Digestif.S) = struct module Mn = Minor_heap (Digestif) include Git.Store.Make (Digestif) (Mn) (Major_heap) (Reference_heap) let major_uid = { Git.Store.pck_major_uid_of_uid = (fun _root uid -> Fpath.v (Fmt.str "pack-%s.pack" (Digestif.to_hex uid))); Git.Store.idx_major_uid_of_uid = (fun _root uid -> Fpath.v (Fmt.str "pack-%s.idx" (Digestif.to_hex uid))); Git.Store.uid_of_major_uid = (fun path -> let str = Fpath.basename (Fpath.rem_ext path) in match Astring.String.cut ~sep:"pack-" str with | Some ("", uid) -> Digestif.of_hex uid | _ -> Fmt.invalid_arg "Invalid major uniq ID: %a" Fpath.pp path); } let update_head refs = match Reference_heap.atomic_rd refs Git.Reference.head with | Error (`Not_found _) -> Reference_heap.atomic_wr refs Git.Reference.head (Fmt.str "ref: %a\n" Git.Reference.pp Git.Reference.master) |> Lwt.return | Ok _ -> Lwt.return_ok () | Error (`Msg _ as err) -> Lwt.return_error err let v ?dotgit root = let dotgit = match dotgit with Some v -> v | None -> Fpath.(root / ".git") in let packed = Packed_refs.load ~of_hex:Hash.of_hex dotgit in let minor = Fpath.(dotgit / "objects") in let major = Fpath.(dotgit / "objects" / "pack") in let temp = Fpath.(dotgit / "tmp") in let refs = dotgit in Bos.OS.Dir.set_default_tmp temp; Unix.mkdir ~path:true refs >>? fun _ -> Unix.mkdir ~path:true temp >>? fun _ -> Unix.mkdir ~path:true Fpath.(refs / "refs" / "heads") >>? fun _ -> Unix.mkdir ~path:true Fpath.(refs / "refs" / "tags") >>? fun _ -> update_head refs >>? fun _ -> Unix.mkdir ~path:true minor >>? fun _ -> Unix.mkdir ~path:true major >>? fun _ -> let open Lwt.Infix in (* TODO(dinosaure): [stat] directories. *) v ~dotgit ~minor ~major ~major_uid ~refs ~packed root >>= fun x -> Lwt.return_ok x end module Store = Make (Digestif.SHA1) let ctx = Git_unix_mimic.ctx module Sync (Git_store : Git.S) = struct let src = Logs.Src.create "git-unix.sync" ~doc:"logs git-unix's sync event" module Log = (val Logs.src_log src : Logs.LOG) include Git.Sync.Make (Git_store.Hash) (Major_heap) (Major_heap) (Git_store) let random_gen = lazy (Random.State.make_self_init ()) let random_path pat = let rand = Random.State.bits (Lazy.force random_gen) land 0xFFFFFF in Fpath.v (Fmt.str pat (Fmt.str "%06x" rand)) let failwithf fmt = Fmt.kstr (fun err -> Lwt.fail (Failure err)) fmt let create_tmp_path mode dir pat = let rec loop count = if count < 0 then failwithf "Create a temporary file %s in %a: too many failing attempts" (Fmt.str pat "XXXXXX") Fpath.pp dir else let file = random_path pat in let sfile = Fpath.to_string Fpath.(dir // file) in let open_flags = Unix.[ O_WRONLY; O_CREAT; O_EXCL; O_SHARE_DELETE ] in let process () = Lwt_unix.openfile sfile open_flags mode >>= fun fd -> Lwt.return (file, fd) in let error = function | Unix.Unix_error (Unix.EEXIST, _, _) -> loop (pred count) | Unix.Unix_error (Unix.EINTR, _, _) -> loop count | exn -> Lwt.fail exn in Lwt.catch process error in loop 10000 let tmp ?(mode = 0o600) dir pat = create_tmp_path mode dir pat >>= fun (file, fd) -> Lwt_unix.close fd >>= fun () -> Lwt.return file let stream_of_file ?(chunk = De.io_buffer_size) path = let stream, emitter = Lwt_stream.create () in let fill () = Lwt_unix.openfile (Fpath.to_string path) Unix.[ O_RDONLY ] 0o644 >>= fun fd -> let rec go () = let tmp = Bytes.create chunk in Lwt.catch (fun () -> Lwt_unix.read fd tmp 0 chunk >>= function | 0 -> emitter None; Lwt_unix.close fd | len -> emitter (Some (Bytes.sub_string tmp 0 len)); go ()) (fun _exn -> emitter None; Lwt_unix.close fd) in go () in Lwt.async fill; fun () -> Lwt_stream.get stream let fetch ?(push_stdout = ignore) ?(push_stderr = ignore) ?threads ~ctx edn store ?version ?capabilities ?deepen want = let dotgit = Git_store.dotgit store in let temp = Fpath.(dotgit / "tmp") in tmp temp "pack-%s.pack" >>= fun src -> tmp temp "pack-%s.pack" >>= fun dst -> tmp temp "pack-%s.idx" >>= fun idx -> let create_idx_stream () = stream_of_file Fpath.(temp // idx) in let create_pack_stream () = stream_of_file Fpath.(temp // dst) in fetch ~push_stdout ~push_stderr ?threads ~ctx edn store ?version ?capabilities ?deepen want ~src ~dst ~idx ~create_idx_stream ~create_pack_stream temp temp let push ~ctx edn store ?version ?capabilities cmds = push ~ctx edn store ?version ?capabilities cmds end
sectionYPositions = computeSectionYPositions($el), 10)"
x-init="setTimeout(() => sectionYPositions = computeSectionYPositions($el), 10)"
>