Show More
@@ -269,7 +269,7 b' def _rundispatch(req):' | |||||
269 | ferr.write(inst.format()) |
|
269 | ferr.write(inst.format()) | |
270 | return -1 |
|
270 | return -1 | |
271 |
|
271 | |||
272 |
|
|
272 | formattedargs = _formatargs(req.args) | |
273 | starttime = util.timer() |
|
273 | starttime = util.timer() | |
274 | ret = 1 # default of Python exit code on unhandled exception |
|
274 | ret = 1 # default of Python exit code on unhandled exception | |
275 | try: |
|
275 | try: | |
@@ -308,7 +308,7 b' def _rundispatch(req):' | |||||
308 | req.ui.log( |
|
308 | req.ui.log( | |
309 | b"commandfinish", |
|
309 | b"commandfinish", | |
310 | b"%s exited %d after %0.2f seconds\n", |
|
310 | b"%s exited %d after %0.2f seconds\n", | |
311 |
|
|
311 | formattedargs, | |
312 | return_code, |
|
312 | return_code, | |
313 | duration, |
|
313 | duration, | |
314 | return_code=return_code, |
|
314 | return_code=return_code, |
@@ -3,6 +3,12 b'' | |||||
3 | version = 3 |
|
3 | version = 3 | |
4 |
|
4 | |||
5 | [[package]] |
|
5 | [[package]] | |
|
6 | name = "Inflector" | |||
|
7 | version = "0.11.4" | |||
|
8 | source = "registry+https://github.com/rust-lang/crates.io-index" | |||
|
9 | checksum = "fe438c63458706e03479442743baae6c88256498e6431708f6dfc520a26515d3" | |||
|
10 | ||||
|
11 | [[package]] | |||
6 | name = "adler" |
|
12 | name = "adler" | |
7 | version = "0.2.3" |
|
13 | version = "0.2.3" | |
8 | source = "registry+https://github.com/rust-lang/crates.io-index" |
|
14 | source = "registry+https://github.com/rust-lang/crates.io-index" | |
@@ -24,6 +30,12 b' dependencies = [' | |||||
24 | ] |
|
30 | ] | |
25 |
|
31 | |||
26 | [[package]] |
|
32 | [[package]] | |
|
33 | name = "aliasable" | |||
|
34 | version = "0.1.3" | |||
|
35 | source = "registry+https://github.com/rust-lang/crates.io-index" | |||
|
36 | checksum = "250f629c0161ad8107cf89319e990051fae62832fd343083bea452d93e2205fd" | |||
|
37 | ||||
|
38 | [[package]] | |||
27 | name = "ansi_term" |
|
39 | name = "ansi_term" | |
28 | version = "0.12.1" |
|
40 | version = "0.12.1" | |
29 | source = "registry+https://github.com/rust-lang/crates.io-index" |
|
41 | source = "registry+https://github.com/rust-lang/crates.io-index" | |
@@ -456,6 +468,7 b' dependencies = [' | |||||
456 | "log", |
|
468 | "log", | |
457 | "memmap2", |
|
469 | "memmap2", | |
458 | "micro-timer", |
|
470 | "micro-timer", | |
|
471 | "ouroboros", | |||
459 | "pretty_assertions", |
|
472 | "pretty_assertions", | |
460 | "rand 0.8.5", |
|
473 | "rand 0.8.5", | |
461 | "rand_distr", |
|
474 | "rand_distr", | |
@@ -464,7 +477,6 b' dependencies = [' | |||||
464 | "regex", |
|
477 | "regex", | |
465 | "same-file", |
|
478 | "same-file", | |
466 | "sha-1 0.10.0", |
|
479 | "sha-1 0.10.0", | |
467 | "stable_deref_trait", |
|
|||
468 | "tempfile", |
|
480 | "tempfile", | |
469 | "twox-hash", |
|
481 | "twox-hash", | |
470 | "zstd", |
|
482 | "zstd", | |
@@ -681,6 +693,30 b' source = "registry+https://github.com/ru' | |||||
681 | checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5" |
|
693 | checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5" | |
682 |
|
694 | |||
683 | [[package]] |
|
695 | [[package]] | |
|
696 | name = "ouroboros" | |||
|
697 | version = "0.15.0" | |||
|
698 | source = "registry+https://github.com/rust-lang/crates.io-index" | |||
|
699 | checksum = "9f31a3b678685b150cba82b702dcdc5e155893f63610cf388d30cd988d4ca2bf" | |||
|
700 | dependencies = [ | |||
|
701 | "aliasable", | |||
|
702 | "ouroboros_macro", | |||
|
703 | "stable_deref_trait", | |||
|
704 | ] | |||
|
705 | ||||
|
706 | [[package]] | |||
|
707 | name = "ouroboros_macro" | |||
|
708 | version = "0.15.0" | |||
|
709 | source = "registry+https://github.com/rust-lang/crates.io-index" | |||
|
710 | checksum = "084fd65d5dd8b3772edccb5ffd1e4b7eba43897ecd0f9401e330e8c542959408" | |||
|
711 | dependencies = [ | |||
|
712 | "Inflector", | |||
|
713 | "proc-macro-error", | |||
|
714 | "proc-macro2", | |||
|
715 | "quote", | |||
|
716 | "syn", | |||
|
717 | ] | |||
|
718 | ||||
|
719 | [[package]] | |||
684 | name = "output_vt100" |
|
720 | name = "output_vt100" | |
685 | version = "0.1.2" |
|
721 | version = "0.1.2" | |
686 | source = "registry+https://github.com/rust-lang/crates.io-index" |
|
722 | source = "registry+https://github.com/rust-lang/crates.io-index" | |
@@ -720,6 +756,30 b' dependencies = [' | |||||
720 | ] |
|
756 | ] | |
721 |
|
757 | |||
722 | [[package]] |
|
758 | [[package]] | |
|
759 | name = "proc-macro-error" | |||
|
760 | version = "1.0.4" | |||
|
761 | source = "registry+https://github.com/rust-lang/crates.io-index" | |||
|
762 | checksum = "da25490ff9892aab3fcf7c36f08cfb902dd3e71ca0f9f9517bea02a73a5ce38c" | |||
|
763 | dependencies = [ | |||
|
764 | "proc-macro-error-attr", | |||
|
765 | "proc-macro2", | |||
|
766 | "quote", | |||
|
767 | "syn", | |||
|
768 | "version_check", | |||
|
769 | ] | |||
|
770 | ||||
|
771 | [[package]] | |||
|
772 | name = "proc-macro-error-attr" | |||
|
773 | version = "1.0.4" | |||
|
774 | source = "registry+https://github.com/rust-lang/crates.io-index" | |||
|
775 | checksum = "a1be40180e52ecc98ad80b184934baf3d0d29f979574e439af5a55274b35f869" | |||
|
776 | dependencies = [ | |||
|
777 | "proc-macro2", | |||
|
778 | "quote", | |||
|
779 | "version_check", | |||
|
780 | ] | |||
|
781 | ||||
|
782 | [[package]] | |||
723 | name = "proc-macro2" |
|
783 | name = "proc-macro2" | |
724 | version = "1.0.24" |
|
784 | version = "1.0.24" | |
725 | source = "registry+https://github.com/rust-lang/crates.io-index" |
|
785 | source = "registry+https://github.com/rust-lang/crates.io-index" |
@@ -19,6 +19,7 b' im-rc = "15.0.0"' | |||||
19 | itertools = "0.10.3" |
|
19 | itertools = "0.10.3" | |
20 | lazy_static = "1.4.0" |
|
20 | lazy_static = "1.4.0" | |
21 | libc = "0.2.119" |
|
21 | libc = "0.2.119" | |
|
22 | ouroboros = "0.15.0" | |||
22 | rand = "0.8.5" |
|
23 | rand = "0.8.5" | |
23 | rand_pcg = "0.3.1" |
|
24 | rand_pcg = "0.3.1" | |
24 | rand_distr = "0.4.3" |
|
25 | rand_distr = "0.4.3" | |
@@ -27,7 +28,6 b' regex = "1.5.5"' | |||||
27 | sha-1 = "0.10.0" |
|
28 | sha-1 = "0.10.0" | |
28 | twox-hash = "1.6.2" |
|
29 | twox-hash = "1.6.2" | |
29 | same-file = "1.0.6" |
|
30 | same-file = "1.0.6" | |
30 | stable_deref_trait = "1.2.0" |
|
|||
31 | tempfile = "3.3.0" |
|
31 | tempfile = "3.3.0" | |
32 | crossbeam-channel = "0.5.2" |
|
32 | crossbeam-channel = "0.5.2" | |
33 | micro-timer = "0.4.0" |
|
33 | micro-timer = "0.4.0" |
@@ -136,8 +136,6 b' pub enum StatusError {' | |||||
136 | DirstateV2ParseError(DirstateV2ParseError), |
|
136 | DirstateV2ParseError(DirstateV2ParseError), | |
137 | } |
|
137 | } | |
138 |
|
138 | |||
139 | pub type StatusResult<T> = Result<T, StatusError>; |
|
|||
140 |
|
||||
141 | impl fmt::Display for StatusError { |
|
139 | impl fmt::Display for StatusError { | |
142 | fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { |
|
140 | fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | |
143 | match self { |
|
141 | match self { |
@@ -723,10 +723,11 b' where' | |||||
723 |
|
723 | |||
724 | impl OwningDirstateMap { |
|
724 | impl OwningDirstateMap { | |
725 | pub fn clear(&mut self) { |
|
725 | pub fn clear(&mut self) { | |
726 |
|
|
726 | self.with_dmap_mut(|map| { | |
727 | map.root = Default::default(); |
|
727 | map.root = Default::default(); | |
728 | map.nodes_with_entry_count = 0; |
|
728 | map.nodes_with_entry_count = 0; | |
729 | map.nodes_with_copy_source_count = 0; |
|
729 | map.nodes_with_copy_source_count = 0; | |
|
730 | }); | |||
730 | } |
|
731 | } | |
731 |
|
732 | |||
732 | pub fn set_entry( |
|
733 | pub fn set_entry( | |
@@ -734,9 +735,10 b' impl OwningDirstateMap {' | |||||
734 | filename: &HgPath, |
|
735 | filename: &HgPath, | |
735 | entry: DirstateEntry, |
|
736 | entry: DirstateEntry, | |
736 | ) -> Result<(), DirstateV2ParseError> { |
|
737 | ) -> Result<(), DirstateV2ParseError> { | |
737 |
|
|
738 | self.with_dmap_mut(|map| { | |
738 | map.get_or_insert(&filename)?.data = NodeData::Entry(entry); |
|
739 | map.get_or_insert(&filename)?.data = NodeData::Entry(entry); | |
739 | Ok(()) |
|
740 | Ok(()) | |
|
741 | }) | |||
740 | } |
|
742 | } | |
741 |
|
743 | |||
742 | pub fn add_file( |
|
744 | pub fn add_file( | |
@@ -745,8 +747,9 b' impl OwningDirstateMap {' | |||||
745 | entry: DirstateEntry, |
|
747 | entry: DirstateEntry, | |
746 | ) -> Result<(), DirstateError> { |
|
748 | ) -> Result<(), DirstateError> { | |
747 | let old_state = self.get(filename)?.map(|e| e.state()); |
|
749 | let old_state = self.get(filename)?.map(|e| e.state()); | |
748 |
|
|
750 | self.with_dmap_mut(|map| { | |
749 | Ok(map.add_or_remove_file(filename, old_state, entry)?) |
|
751 | Ok(map.add_or_remove_file(filename, old_state, entry)?) | |
|
752 | }) | |||
750 | } |
|
753 | } | |
751 |
|
754 | |||
752 | pub fn remove_file( |
|
755 | pub fn remove_file( | |
@@ -777,9 +780,10 b' impl OwningDirstateMap {' | |||||
777 | if size == 0 { |
|
780 | if size == 0 { | |
778 | self.copy_map_remove(filename)?; |
|
781 | self.copy_map_remove(filename)?; | |
779 | } |
|
782 | } | |
780 |
|
|
783 | self.with_dmap_mut(|map| { | |
781 | let entry = DirstateEntry::new_removed(size); |
|
784 | let entry = DirstateEntry::new_removed(size); | |
782 | Ok(map.add_or_remove_file(filename, old_state, entry)?) |
|
785 | Ok(map.add_or_remove_file(filename, old_state, entry)?) | |
|
786 | }) | |||
783 | } |
|
787 | } | |
784 |
|
788 | |||
785 | pub fn drop_entry_and_copy_source( |
|
789 | pub fn drop_entry_and_copy_source( | |
@@ -789,7 +793,6 b' impl OwningDirstateMap {' | |||||
789 | let was_tracked = self |
|
793 | let was_tracked = self | |
790 | .get(filename)? |
|
794 | .get(filename)? | |
791 | .map_or(false, |e| e.state().is_tracked()); |
|
795 | .map_or(false, |e| e.state().is_tracked()); | |
792 | let map = self.get_map_mut(); |
|
|||
793 | struct Dropped { |
|
796 | struct Dropped { | |
794 | was_tracked: bool, |
|
797 | was_tracked: bool, | |
795 | had_entry: bool, |
|
798 | had_entry: bool, | |
@@ -826,10 +829,20 b' impl OwningDirstateMap {' | |||||
826 | )? { |
|
829 | )? { | |
827 | dropped = d; |
|
830 | dropped = d; | |
828 | if dropped.had_entry { |
|
831 | if dropped.had_entry { | |
829 |
node.descendants_with_entry_count |
|
832 | node.descendants_with_entry_count = node | |
|
833 | .descendants_with_entry_count | |||
|
834 | .checked_sub(1) | |||
|
835 | .expect( | |||
|
836 | "descendants_with_entry_count should be >= 0", | |||
|
837 | ); | |||
830 | } |
|
838 | } | |
831 | if dropped.was_tracked { |
|
839 | if dropped.was_tracked { | |
832 |
node.tracked_descendants_count |
|
840 | node.tracked_descendants_count = node | |
|
841 | .tracked_descendants_count | |||
|
842 | .checked_sub(1) | |||
|
843 | .expect( | |||
|
844 | "tracked_descendants_count should be >= 0", | |||
|
845 | ); | |||
833 | } |
|
846 | } | |
834 |
|
847 | |||
835 | // Directory caches must be invalidated when removing a |
|
848 | // Directory caches must be invalidated when removing a | |
@@ -843,21 +856,22 b' impl OwningDirstateMap {' | |||||
843 | return Ok(None); |
|
856 | return Ok(None); | |
844 | } |
|
857 | } | |
845 | } else { |
|
858 | } else { | |
846 |
let |
|
859 | let entry = node.data.as_entry(); | |
|
860 | let was_tracked = entry.map_or(false, |entry| entry.tracked()); | |||
|
861 | let had_entry = entry.is_some(); | |||
847 | if had_entry { |
|
862 | if had_entry { | |
848 | node.data = NodeData::None |
|
863 | node.data = NodeData::None | |
849 | } |
|
864 | } | |
|
865 | let mut had_copy_source = false; | |||
850 | if let Some(source) = &node.copy_source { |
|
866 | if let Some(source) = &node.copy_source { | |
851 | DirstateMap::count_dropped_path(unreachable_bytes, source); |
|
867 | DirstateMap::count_dropped_path(unreachable_bytes, source); | |
|
868 | had_copy_source = true; | |||
852 | node.copy_source = None |
|
869 | node.copy_source = None | |
853 | } |
|
870 | } | |
854 | dropped = Dropped { |
|
871 | dropped = Dropped { | |
855 |
was_tracked |
|
872 | was_tracked, | |
856 | .data |
|
|||
857 | .as_entry() |
|
|||
858 | .map_or(false, |entry| entry.state().is_tracked()), |
|
|||
859 | had_entry, |
|
873 | had_entry, | |
860 |
had_copy_source |
|
874 | had_copy_source, | |
861 | }; |
|
875 | }; | |
862 | } |
|
876 | } | |
863 | // After recursion, for both leaf (rest_of_path is None) nodes and |
|
877 | // After recursion, for both leaf (rest_of_path is None) nodes and | |
@@ -876,6 +890,7 b' impl OwningDirstateMap {' | |||||
876 | Ok(Some((dropped, remove))) |
|
890 | Ok(Some((dropped, remove))) | |
877 | } |
|
891 | } | |
878 |
|
892 | |||
|
893 | self.with_dmap_mut(|map| { | |||
879 | if let Some((dropped, _removed)) = recur( |
|
894 | if let Some((dropped, _removed)) = recur( | |
880 | map.on_disk, |
|
895 | map.on_disk, | |
881 | &mut map.unreachable_bytes, |
|
896 | &mut map.unreachable_bytes, | |
@@ -883,22 +898,29 b' impl OwningDirstateMap {' | |||||
883 | filename, |
|
898 | filename, | |
884 | )? { |
|
899 | )? { | |
885 | if dropped.had_entry { |
|
900 | if dropped.had_entry { | |
886 |
map.nodes_with_entry_count |
|
901 | map.nodes_with_entry_count = map | |
|
902 | .nodes_with_entry_count | |||
|
903 | .checked_sub(1) | |||
|
904 | .expect("nodes_with_entry_count should be >= 0"); | |||
887 | } |
|
905 | } | |
888 | if dropped.had_copy_source { |
|
906 | if dropped.had_copy_source { | |
889 |
map.nodes_with_copy_source_count |
|
907 | map.nodes_with_copy_source_count = map | |
|
908 | .nodes_with_copy_source_count | |||
|
909 | .checked_sub(1) | |||
|
910 | .expect("nodes_with_copy_source_count should be >= 0"); | |||
890 | } |
|
911 | } | |
891 | } else { |
|
912 | } else { | |
892 | debug_assert!(!was_tracked); |
|
913 | debug_assert!(!was_tracked); | |
893 | } |
|
914 | } | |
894 | Ok(()) |
|
915 | Ok(()) | |
|
916 | }) | |||
895 | } |
|
917 | } | |
896 |
|
918 | |||
897 | pub fn has_tracked_dir( |
|
919 | pub fn has_tracked_dir( | |
898 | &mut self, |
|
920 | &mut self, | |
899 | directory: &HgPath, |
|
921 | directory: &HgPath, | |
900 | ) -> Result<bool, DirstateError> { |
|
922 | ) -> Result<bool, DirstateError> { | |
901 |
|
|
923 | self.with_dmap_mut(|map| { | |
902 | if let Some(node) = map.get_node(directory)? { |
|
924 | if let Some(node) = map.get_node(directory)? { | |
903 | // A node without a `DirstateEntry` was created to hold child |
|
925 | // A node without a `DirstateEntry` was created to hold child | |
904 | // nodes, and is therefore a directory. |
|
926 | // nodes, and is therefore a directory. | |
@@ -907,13 +929,14 b' impl OwningDirstateMap {' | |||||
907 | } else { |
|
929 | } else { | |
908 | Ok(false) |
|
930 | Ok(false) | |
909 | } |
|
931 | } | |
|
932 | }) | |||
910 | } |
|
933 | } | |
911 |
|
934 | |||
912 | pub fn has_dir( |
|
935 | pub fn has_dir( | |
913 | &mut self, |
|
936 | &mut self, | |
914 | directory: &HgPath, |
|
937 | directory: &HgPath, | |
915 | ) -> Result<bool, DirstateError> { |
|
938 | ) -> Result<bool, DirstateError> { | |
916 |
|
|
939 | self.with_dmap_mut(|map| { | |
917 | if let Some(node) = map.get_node(directory)? { |
|
940 | if let Some(node) = map.get_node(directory)? { | |
918 | // A node without a `DirstateEntry` was created to hold child |
|
941 | // A node without a `DirstateEntry` was created to hold child | |
919 | // nodes, and is therefore a directory. |
|
942 | // nodes, and is therefore a directory. | |
@@ -922,6 +945,7 b' impl OwningDirstateMap {' | |||||
922 | } else { |
|
945 | } else { | |
923 | Ok(false) |
|
946 | Ok(false) | |
924 | } |
|
947 | } | |
|
948 | }) | |||
925 | } |
|
949 | } | |
926 |
|
950 | |||
927 | #[timed] |
|
951 | #[timed] | |
@@ -973,16 +997,29 b' impl OwningDirstateMap {' | |||||
973 | on_disk::write(map, can_append) |
|
997 | on_disk::write(map, can_append) | |
974 | } |
|
998 | } | |
975 |
|
999 | |||
976 | pub fn status<'a>( |
|
1000 | /// `callback` allows the caller to process and do something with the | |
977 | &'a mut self, |
|
1001 | /// results of the status. This is needed to do so efficiently (i.e. | |
978 | matcher: &'a (dyn Matcher + Sync), |
|
1002 | /// without cloning the `DirstateStatus` object with its paths) because | |
|
1003 | /// we need to borrow from `Self`. | |||
|
1004 | pub fn with_status<R>( | |||
|
1005 | &mut self, | |||
|
1006 | matcher: &(dyn Matcher + Sync), | |||
979 | root_dir: PathBuf, |
|
1007 | root_dir: PathBuf, | |
980 | ignore_files: Vec<PathBuf>, |
|
1008 | ignore_files: Vec<PathBuf>, | |
981 | options: StatusOptions, |
|
1009 | options: StatusOptions, | |
982 | ) -> Result<(DirstateStatus<'a>, Vec<PatternFileWarning>), StatusError> |
|
1010 | callback: impl for<'r> FnOnce( | |
983 | { |
|
1011 | Result<(DirstateStatus<'r>, Vec<PatternFileWarning>), StatusError>, | |
984 | let map = self.get_map_mut(); |
|
1012 | ) -> R, | |
985 | super::status::status(map, matcher, root_dir, ignore_files, options) |
|
1013 | ) -> R { | |
|
1014 | self.with_dmap_mut(|map| { | |||
|
1015 | callback(super::status::status( | |||
|
1016 | map, | |||
|
1017 | matcher, | |||
|
1018 | root_dir, | |||
|
1019 | ignore_files, | |||
|
1020 | options, | |||
|
1021 | )) | |||
|
1022 | }) | |||
986 | } |
|
1023 | } | |
987 |
|
1024 | |||
988 | pub fn copy_map_len(&self) -> usize { |
|
1025 | pub fn copy_map_len(&self) -> usize { | |
@@ -1030,7 +1067,7 b' impl OwningDirstateMap {' | |||||
1030 | &mut self, |
|
1067 | &mut self, | |
1031 | key: &HgPath, |
|
1068 | key: &HgPath, | |
1032 | ) -> Result<Option<HgPathBuf>, DirstateV2ParseError> { |
|
1069 | ) -> Result<Option<HgPathBuf>, DirstateV2ParseError> { | |
1033 |
|
|
1070 | self.with_dmap_mut(|map| { | |
1034 | let count = &mut map.nodes_with_copy_source_count; |
|
1071 | let count = &mut map.nodes_with_copy_source_count; | |
1035 | let unreachable_bytes = &mut map.unreachable_bytes; |
|
1072 | let unreachable_bytes = &mut map.unreachable_bytes; | |
1036 | Ok(DirstateMap::get_node_mut( |
|
1073 | Ok(DirstateMap::get_node_mut( | |
@@ -1046,6 +1083,7 b' impl OwningDirstateMap {' | |||||
1046 | } |
|
1083 | } | |
1047 | node.copy_source.take().map(Cow::into_owned) |
|
1084 | node.copy_source.take().map(Cow::into_owned) | |
1048 | })) |
|
1085 | })) | |
|
1086 | }) | |||
1049 | } |
|
1087 | } | |
1050 |
|
1088 | |||
1051 | pub fn copy_map_insert( |
|
1089 | pub fn copy_map_insert( | |
@@ -1053,7 +1091,7 b' impl OwningDirstateMap {' | |||||
1053 | key: HgPathBuf, |
|
1091 | key: HgPathBuf, | |
1054 | value: HgPathBuf, |
|
1092 | value: HgPathBuf, | |
1055 | ) -> Result<Option<HgPathBuf>, DirstateV2ParseError> { |
|
1093 | ) -> Result<Option<HgPathBuf>, DirstateV2ParseError> { | |
1056 |
|
|
1094 | self.with_dmap_mut(|map| { | |
1057 | let node = DirstateMap::get_or_insert_node( |
|
1095 | let node = DirstateMap::get_or_insert_node( | |
1058 | map.on_disk, |
|
1096 | map.on_disk, | |
1059 | &mut map.unreachable_bytes, |
|
1097 | &mut map.unreachable_bytes, | |
@@ -1066,6 +1104,7 b' impl OwningDirstateMap {' | |||||
1066 | map.nodes_with_copy_source_count += 1 |
|
1104 | map.nodes_with_copy_source_count += 1 | |
1067 | } |
|
1105 | } | |
1068 | Ok(node.copy_source.replace(value.into()).map(Cow::into_owned)) |
|
1106 | Ok(node.copy_source.replace(value.into()).map(Cow::into_owned)) | |
|
1107 | }) | |||
1069 | } |
|
1108 | } | |
1070 |
|
1109 | |||
1071 | pub fn len(&self) -> usize { |
|
1110 | pub fn len(&self) -> usize { | |
@@ -1113,7 +1152,7 b' impl OwningDirstateMap {' | |||||
1113 | >, |
|
1152 | >, | |
1114 | DirstateError, |
|
1153 | DirstateError, | |
1115 | > { |
|
1154 | > { | |
1116 |
let map = self.get_map |
|
1155 | let map = self.get_map(); | |
1117 | let on_disk = map.on_disk; |
|
1156 | let on_disk = map.on_disk; | |
1118 | Ok(Box::new(filter_map_results( |
|
1157 | Ok(Box::new(filter_map_results( | |
1119 | map.iter_nodes(), |
|
1158 | map.iter_nodes(), |
@@ -1,105 +1,89 b'' | |||||
|
1 | use crate::{DirstateError, DirstateParents}; | |||
|
2 | ||||
1 | use super::dirstate_map::DirstateMap; |
|
3 | use super::dirstate_map::DirstateMap; | |
2 | use stable_deref_trait::StableDeref; |
|
|||
3 | use std::ops::Deref; |
|
4 | use std::ops::Deref; | |
4 |
|
5 | |||
|
6 | use ouroboros::self_referencing; | |||
|
7 | ||||
5 | /// Keep a `DirstateMap<'on_disk>` next to the `on_disk` buffer that it |
|
8 | /// Keep a `DirstateMap<'on_disk>` next to the `on_disk` buffer that it | |
6 | /// borrows. |
|
9 | /// borrows. | |
7 | /// |
|
10 | #[self_referencing] | |
8 | /// This is similar to [`OwningRef`] which is more limited because it |
|
|||
9 | /// represents exactly one `&T` reference next to the value it borrows, as |
|
|||
10 | /// opposed to a struct that may contain an arbitrary number of references in |
|
|||
11 | /// arbitrarily-nested data structures. |
|
|||
12 | /// |
|
|||
13 | /// [`OwningRef`]: https://docs.rs/owning_ref/0.4.1/owning_ref/struct.OwningRef.html |
|
|||
14 | pub struct OwningDirstateMap { |
|
11 | pub struct OwningDirstateMap { | |
15 | /// Owned handle to a bytes buffer with a stable address. |
|
|||
16 | /// |
|
|||
17 | /// See <https://docs.rs/owning_ref/0.4.1/owning_ref/trait.StableAddress.html>. |
|
|||
18 | on_disk: Box<dyn Deref<Target = [u8]> + Send>, |
|
12 | on_disk: Box<dyn Deref<Target = [u8]> + Send>, | |
19 |
|
13 | #[borrows(on_disk)] | ||
20 | /// Pointer for `Box<DirstateMap<'on_disk>>`, typed-erased because the |
|
14 | #[covariant] | |
21 | /// language cannot represent a lifetime referencing a sibling field. |
|
15 | map: DirstateMap<'this>, | |
22 | /// This is not quite a self-referencial struct (moving this struct is not |
|
|||
23 | /// a problem as it doesnβt change the address of the bytes buffer owned |
|
|||
24 | /// by `on_disk`) but touches similar borrow-checker limitations. |
|
|||
25 | ptr: *mut (), |
|
|||
26 | } |
|
16 | } | |
27 |
|
17 | |||
28 | impl OwningDirstateMap { |
|
18 | impl OwningDirstateMap { | |
29 | pub fn new_empty<OnDisk>(on_disk: OnDisk) -> Self |
|
19 | pub fn new_empty<OnDisk>(on_disk: OnDisk) -> Self | |
30 | where |
|
20 | where | |
31 |
OnDisk: Deref<Target = [u8]> + |
|
21 | OnDisk: Deref<Target = [u8]> + Send + 'static, | |
32 | { |
|
22 | { | |
33 | let on_disk = Box::new(on_disk); |
|
23 | let on_disk = Box::new(on_disk); | |
34 | let bytes: &'_ [u8] = &on_disk; |
|
|||
35 | let map = DirstateMap::empty(bytes); |
|
|||
36 |
|
24 | |||
37 | // Like in `bytes` above, this `'_` lifetime parameter borrows from |
|
25 | OwningDirstateMapBuilder { | |
38 | // the bytes buffer owned by `on_disk`. |
|
26 | on_disk, | |
39 | let ptr: *mut DirstateMap<'_> = Box::into_raw(Box::new(map)); |
|
27 | map_builder: |bytes| DirstateMap::empty(&bytes), | |
40 |
|
28 | } | ||
41 | // Erase the pointed type entirely in order to erase the lifetime. |
|
29 | .build() | |
42 | let ptr: *mut () = ptr.cast(); |
|
|||
43 |
|
||||
44 | Self { on_disk, ptr } |
|
|||
45 | } |
|
30 | } | |
46 |
|
31 | |||
47 | pub fn get_pair_mut<'a>( |
|
32 | pub fn new_v1<OnDisk>( | |
48 | &'a mut self, |
|
33 | on_disk: OnDisk, | |
49 | ) -> (&'a [u8], &'a mut DirstateMap<'a>) { |
|
34 | ) -> Result<(Self, DirstateParents), DirstateError> | |
50 | // SAFETY: We cast the type-erased pointer back to the same type it had |
|
35 | where | |
51 | // in `new`, except with a different lifetime parameter. This time we |
|
36 | OnDisk: Deref<Target = [u8]> + Send + 'static, | |
52 | // connect the lifetime to that of `self`. This cast is valid because |
|
37 | { | |
53 | // `self` owns the same `on_disk` whose buffer `DirstateMap` |
|
38 | let on_disk = Box::new(on_disk); | |
54 | // references. That buffer has a stable memory address because our |
|
39 | let mut parents = DirstateParents::NULL; | |
55 | // `Self::new_empty` counstructor requires `StableDeref`. |
|
40 | ||
56 | let ptr: *mut DirstateMap<'a> = self.ptr.cast(); |
|
41 | Ok(( | |
57 | // SAFETY: we dereference that pointer, connecting the lifetime of the |
|
42 | OwningDirstateMapTryBuilder { | |
58 | // new `&mut` to that of `self`. This is valid because the |
|
43 | on_disk, | |
59 | // raw pointer is to a boxed value, and `self` owns that box. |
|
44 | map_builder: |bytes| { | |
60 | (&self.on_disk, unsafe { &mut *ptr }) |
|
45 | DirstateMap::new_v1(&bytes).map(|(dmap, p)| { | |
|
46 | parents = p.unwrap_or(DirstateParents::NULL); | |||
|
47 | dmap | |||
|
48 | }) | |||
|
49 | }, | |||
61 | } |
|
50 | } | |
62 |
|
51 | .try_build()?, | ||
63 | pub fn get_map_mut<'a>(&'a mut self) -> &'a mut DirstateMap<'a> { |
|
52 | parents, | |
64 | self.get_pair_mut().1 |
|
53 | )) | |
65 | } |
|
54 | } | |
66 |
|
55 | |||
67 | pub fn get_map<'a>(&'a self) -> &'a DirstateMap<'a> { |
|
56 | pub fn new_v2<OnDisk>( | |
68 | // SAFETY: same reasoning as in `get_pair_mut` above. |
|
57 | on_disk: OnDisk, | |
69 | let ptr: *mut DirstateMap<'a> = self.ptr.cast(); |
|
58 | data_size: usize, | |
70 | unsafe { &*ptr } |
|
59 | metadata: &[u8], | |
|
60 | ) -> Result<Self, DirstateError> | |||
|
61 | where | |||
|
62 | OnDisk: Deref<Target = [u8]> + Send + 'static, | |||
|
63 | { | |||
|
64 | let on_disk = Box::new(on_disk); | |||
|
65 | ||||
|
66 | OwningDirstateMapTryBuilder { | |||
|
67 | on_disk, | |||
|
68 | map_builder: |bytes| { | |||
|
69 | DirstateMap::new_v2(&bytes, data_size, metadata) | |||
|
70 | }, | |||
71 | } |
|
71 | } | |
72 |
|
72 | .try_build() | ||
73 | pub fn on_disk<'a>(&'a self) -> &'a [u8] { |
|
|||
74 | &self.on_disk |
|
|||
75 | } |
|
|||
76 | } |
|
73 | } | |
77 |
|
74 | |||
78 | impl Drop for OwningDirstateMap { |
|
75 | pub fn with_dmap_mut<R>( | |
79 |
|
|
76 | &mut self, | |
80 | // Silence a "field is never read" warning, and demonstrate that this |
|
77 | f: impl FnOnce(&mut DirstateMap) -> R, | |
81 | // value is still alive. |
|
78 | ) -> R { | |
82 | let _: &Box<dyn Deref<Target = [u8]> + Send> = &self.on_disk; |
|
79 | self.with_map_mut(f) | |
83 | // SAFETY: this cast is the same as in `get_mut`, and is valid for the |
|
80 | } | |
84 | // same reason. `self.on_disk` still exists at this point, drop glue |
|
81 | ||
85 | // will drop it implicitly after this `drop` method returns. |
|
82 | pub fn get_map(&self) -> &DirstateMap { | |
86 | let ptr: *mut DirstateMap<'_> = self.ptr.cast(); |
|
83 | self.borrow_map() | |
87 | // SAFETY: `Box::from_raw` takes ownership of the box away from `self`. |
|
84 | } | |
88 | // This is fine because drop glue does nothing for `*mut ()` and weβre |
|
85 | ||
89 | // in `drop`, so `get` and `get_mut` cannot be called again. |
|
86 | pub fn on_disk(&self) -> &[u8] { | |
90 | unsafe { drop(Box::from_raw(ptr)) } |
|
87 | self.borrow_on_disk() | |
91 | } |
|
88 | } | |
92 | } |
|
89 | } | |
93 |
|
||||
94 | fn _static_assert_is_send<T: Send>() {} |
|
|||
95 |
|
||||
96 | fn _static_assert_fields_are_send() { |
|
|||
97 | _static_assert_is_send::<Box<DirstateMap<'_>>>(); |
|
|||
98 | } |
|
|||
99 |
|
||||
100 | // SAFETY: we donβt get this impl implicitly because `*mut (): !Send` because |
|
|||
101 | // thread-safety of raw pointers is unknown in the general case. However this |
|
|||
102 | // particular raw pointer represents a `Box<DirstateMap<'on_disk>>` that we |
|
|||
103 | // own. Since that `Box` is `Send` as shown in above, it is sound to mark |
|
|||
104 | // this struct as `Send` too. |
|
|||
105 | unsafe impl Send for OwningDirstateMap {} |
|
@@ -40,13 +40,14 b' use std::time::SystemTime;' | |||||
40 | /// exists in one of the two trees, depending on information requested by |
|
40 | /// exists in one of the two trees, depending on information requested by | |
41 | /// `options` we may need to traverse the remaining subtree. |
|
41 | /// `options` we may need to traverse the remaining subtree. | |
42 | #[timed] |
|
42 | #[timed] | |
43 |
pub fn status<' |
|
43 | pub fn status<'dirstate>( | |
44 |
dmap: &' |
|
44 | dmap: &'dirstate mut DirstateMap, | |
45 | matcher: &(dyn Matcher + Sync), |
|
45 | matcher: &(dyn Matcher + Sync), | |
46 | root_dir: PathBuf, |
|
46 | root_dir: PathBuf, | |
47 | ignore_files: Vec<PathBuf>, |
|
47 | ignore_files: Vec<PathBuf>, | |
48 | options: StatusOptions, |
|
48 | options: StatusOptions, | |
49 |
) -> Result<(DirstateStatus<' |
|
49 | ) -> Result<(DirstateStatus<'dirstate>, Vec<PatternFileWarning>), StatusError> | |
|
50 | { | |||
50 | // Force the global rayon threadpool to not exceed 16 concurrent threads. |
|
51 | // Force the global rayon threadpool to not exceed 16 concurrent threads. | |
51 | // This is a stop-gap measure until we figure out why using more than 16 |
|
52 | // This is a stop-gap measure until we figure out why using more than 16 | |
52 | // threads makes `status` slower for each additional thread. |
|
53 | // threads makes `status` slower for each additional thread. |
@@ -1,7 +1,6 b'' | |||||
1 | use crate::changelog::Changelog; |
|
1 | use crate::changelog::Changelog; | |
2 | use crate::config::{Config, ConfigError, ConfigParseError}; |
|
2 | use crate::config::{Config, ConfigError, ConfigParseError}; | |
3 | use crate::dirstate::DirstateParents; |
|
3 | use crate::dirstate::DirstateParents; | |
4 | use crate::dirstate_tree::dirstate_map::DirstateMap; |
|
|||
5 | use crate::dirstate_tree::on_disk::Docket as DirstateDocket; |
|
4 | use crate::dirstate_tree::on_disk::Docket as DirstateDocket; | |
6 | use crate::dirstate_tree::owning::OwningDirstateMap; |
|
5 | use crate::dirstate_tree::owning::OwningDirstateMap; | |
7 | use crate::errors::HgResultExt; |
|
6 | use crate::errors::HgResultExt; | |
@@ -316,25 +315,19 b' impl Repo {' | |||||
316 | .set(Some(docket.uuid.to_owned())); |
|
315 | .set(Some(docket.uuid.to_owned())); | |
317 | let data_size = docket.data_size(); |
|
316 | let data_size = docket.data_size(); | |
318 | let metadata = docket.tree_metadata(); |
|
317 | let metadata = docket.tree_metadata(); | |
319 |
|
|
318 | if let Some(data_mmap) = self | |
320 | .hg_vfs() |
|
319 | .hg_vfs() | |
321 | .mmap_open(docket.data_filename()) |
|
320 | .mmap_open(docket.data_filename()) | |
322 | .io_not_found_as_none()? |
|
321 | .io_not_found_as_none()? | |
323 | { |
|
322 | { | |
324 |
OwningDirstateMap::new_ |
|
323 | OwningDirstateMap::new_v2(data_mmap, data_size, metadata) | |
325 | } else { |
|
324 | } else { | |
326 |
OwningDirstateMap::new_ |
|
325 | OwningDirstateMap::new_v2(Vec::new(), data_size, metadata) | |
327 |
} |
|
326 | } | |
328 | let (on_disk, placeholder) = map.get_pair_mut(); |
|
|||
329 | *placeholder = DirstateMap::new_v2(on_disk, data_size, metadata)?; |
|
|||
330 | Ok(map) |
|
|||
331 | } else { |
|
327 | } else { | |
332 | let mut map = OwningDirstateMap::new_empty(dirstate_file_contents); |
|
328 | let (map, parents) = | |
333 | let (on_disk, placeholder) = map.get_pair_mut(); |
|
329 | OwningDirstateMap::new_v1(dirstate_file_contents)?; | |
334 | let (inner, parents) = DirstateMap::new_v1(on_disk)?; |
|
330 | self.dirstate_parents.set(parents); | |
335 | self.dirstate_parents |
|
|||
336 | .set(parents.unwrap_or(DirstateParents::NULL)); |
|
|||
337 | *placeholder = inner; |
|
|||
338 | Ok(map) |
|
331 | Ok(map) | |
339 | } |
|
332 | } | |
340 | } |
|
333 | } |
@@ -144,15 +144,8 b' impl From<HgPathError> for std::io::Erro' | |||||
144 | /// On Unix, it's just byte-to-byte conversion. On Windows, it has to be |
|
144 | /// On Unix, it's just byte-to-byte conversion. On Windows, it has to be | |
145 | /// decoded from MBCS to WTF-8. If WindowsUTF8Plan is implemented, the source |
|
145 | /// decoded from MBCS to WTF-8. If WindowsUTF8Plan is implemented, the source | |
146 | /// character encoding will be determined on a per-repository basis. |
|
146 | /// character encoding will be determined on a per-repository basis. | |
147 | // |
|
|||
148 | // FIXME: (adapted from a comment in the stdlib) |
|
|||
149 | // `HgPath::new()` current implementation relies on `Slice` being |
|
|||
150 | // layout-compatible with `[u8]`. |
|
|||
151 | // When attribute privacy is implemented, `Slice` should be annotated as |
|
|||
152 | // `#[repr(transparent)]`. |
|
|||
153 | // Anyway, `Slice` representation and layout are considered implementation |
|
|||
154 | // detail, are not documented and must not be relied upon. |
|
|||
155 | #[derive(Eq, Ord, PartialEq, PartialOrd, Hash)] |
|
147 | #[derive(Eq, Ord, PartialEq, PartialOrd, Hash)] | |
|
148 | #[repr(transparent)] | |||
156 | pub struct HgPath { |
|
149 | pub struct HgPath { | |
157 | inner: [u8], |
|
150 | inner: [u8], | |
158 | } |
|
151 | } |
@@ -23,7 +23,6 b' use crate::{' | |||||
23 | }; |
|
23 | }; | |
24 | use hg::{ |
|
24 | use hg::{ | |
25 | dirstate::StateMapIter, |
|
25 | dirstate::StateMapIter, | |
26 | dirstate_tree::dirstate_map::DirstateMap as TreeDirstateMap, |
|
|||
27 | dirstate_tree::on_disk::DirstateV2ParseError, |
|
26 | dirstate_tree::on_disk::DirstateV2ParseError, | |
28 | dirstate_tree::owning::OwningDirstateMap, |
|
27 | dirstate_tree::owning::OwningDirstateMap, | |
29 | revlog::Node, |
|
28 | revlog::Node, | |
@@ -53,18 +52,12 b' py_class!(pub class DirstateMap |py| {' | |||||
53 | on_disk: PyBytes, |
|
52 | on_disk: PyBytes, | |
54 | ) -> PyResult<PyObject> { |
|
53 | ) -> PyResult<PyObject> { | |
55 | let on_disk = PyBytesDeref::new(py, on_disk); |
|
54 | let on_disk = PyBytesDeref::new(py, on_disk); | |
56 |
let |
|
55 | let (map, parents) = OwningDirstateMap::new_v1(on_disk) | |
57 | let (on_disk, map_placeholder) = map.get_pair_mut(); |
|
|||
58 |
|
||||
59 | let (actual_map, parents) = TreeDirstateMap::new_v1(on_disk) |
|
|||
60 | .map_err(|e| dirstate_error(py, e))?; |
|
56 | .map_err(|e| dirstate_error(py, e))?; | |
61 | *map_placeholder = actual_map; |
|
|||
62 | let map = Self::create_instance(py, map)?; |
|
57 | let map = Self::create_instance(py, map)?; | |
63 | let parents = parents.map(|p| { |
|
58 | let p1 = PyBytes::new(py, parents.p1.as_bytes()); | |
64 |
|
|
59 | let p2 = PyBytes::new(py, parents.p2.as_bytes()); | |
65 | let p2 = PyBytes::new(py, p.p2.as_bytes()); |
|
60 | let parents = (p1, p2); | |
66 | (p1, p2) |
|
|||
67 | }); |
|
|||
68 | Ok((map, parents).to_py_object(py).into_object()) |
|
61 | Ok((map, parents).to_py_object(py).into_object()) | |
69 | } |
|
62 | } | |
70 |
|
63 | |||
@@ -79,9 +72,7 b' py_class!(pub class DirstateMap |py| {' | |||||
79 | PyErr::new::<exc::OSError, _>(py, format!("Dirstate error: {:?}", e)) |
|
72 | PyErr::new::<exc::OSError, _>(py, format!("Dirstate error: {:?}", e)) | |
80 | }; |
|
73 | }; | |
81 | let on_disk = PyBytesDeref::new(py, on_disk); |
|
74 | let on_disk = PyBytesDeref::new(py, on_disk); | |
82 |
let |
|
75 | let map = OwningDirstateMap::new_v2( | |
83 | let (on_disk, map_placeholder) = map.get_pair_mut(); |
|
|||
84 | *map_placeholder = TreeDirstateMap::new_v2( |
|
|||
85 | on_disk, data_size, tree_metadata.data(py), |
|
76 | on_disk, data_size, tree_metadata.data(py), | |
86 | ).map_err(dirstate_error)?; |
|
77 | ).map_err(dirstate_error)?; | |
87 | let map = Self::create_instance(py, map)?; |
|
78 | let map = Self::create_instance(py, map)?; |
@@ -127,11 +127,16 b' pub fn status_wrapper(' | |||||
127 | // The caller may call `copymap.items()` separately |
|
127 | // The caller may call `copymap.items()` separately | |
128 | let list_copies = false; |
|
128 | let list_copies = false; | |
129 |
|
129 | |||
|
130 | let after_status = |res: Result<(DirstateStatus<'_>, _), StatusError>| { | |||
|
131 | let (status_res, warnings) = | |||
|
132 | res.map_err(|e| handle_fallback(py, e))?; | |||
|
133 | build_response(py, status_res, warnings) | |||
|
134 | }; | |||
|
135 | ||||
130 | match matcher.get_type(py).name(py).borrow() { |
|
136 | match matcher.get_type(py).name(py).borrow() { | |
131 | "alwaysmatcher" => { |
|
137 | "alwaysmatcher" => { | |
132 | let matcher = AlwaysMatcher; |
|
138 | let matcher = AlwaysMatcher; | |
133 | let (status_res, warnings) = dmap |
|
139 | dmap.with_status( | |
134 | .status( |
|
|||
135 |
|
|
140 | &matcher, | |
136 |
|
|
141 | root_dir.to_path_buf(), | |
137 |
|
|
142 | ignore_files, | |
@@ -143,9 +148,8 b' pub fn status_wrapper(' | |||||
143 |
|
|
148 | list_copies, | |
144 |
|
|
149 | collect_traversed_dirs, | |
145 |
|
|
150 | }, | |
|
151 | after_status, | |||
146 |
|
|
152 | ) | |
147 | .map_err(|e| handle_fallback(py, e))?; |
|
|||
148 | build_response(py, status_res, warnings) |
|
|||
149 | } |
|
153 | } | |
150 | "exactmatcher" => { |
|
154 | "exactmatcher" => { | |
151 | let files = matcher.call_method( |
|
155 | let files = matcher.call_method( | |
@@ -167,8 +171,7 b' pub fn status_wrapper(' | |||||
167 | let files = files?; |
|
171 | let files = files?; | |
168 | let matcher = FileMatcher::new(files.as_ref()) |
|
172 | let matcher = FileMatcher::new(files.as_ref()) | |
169 | .map_err(|e| PyErr::new::<ValueError, _>(py, e.to_string()))?; |
|
173 | .map_err(|e| PyErr::new::<ValueError, _>(py, e.to_string()))?; | |
170 | let (status_res, warnings) = dmap |
|
174 | dmap.with_status( | |
171 | .status( |
|
|||
172 |
|
|
175 | &matcher, | |
173 |
|
|
176 | root_dir.to_path_buf(), | |
174 |
|
|
177 | ignore_files, | |
@@ -180,9 +183,8 b' pub fn status_wrapper(' | |||||
180 |
|
|
183 | list_copies, | |
181 |
|
|
184 | collect_traversed_dirs, | |
182 |
|
|
185 | }, | |
|
186 | after_status, | |||
183 |
|
|
187 | ) | |
184 | .map_err(|e| handle_fallback(py, e))?; |
|
|||
185 | build_response(py, status_res, warnings) |
|
|||
186 | } |
|
188 | } | |
187 | "includematcher" => { |
|
189 | "includematcher" => { | |
188 | // Get the patterns from Python even though most of them are |
|
190 | // Get the patterns from Python even though most of them are | |
@@ -219,8 +221,7 b' pub fn status_wrapper(' | |||||
219 | let matcher = IncludeMatcher::new(ignore_patterns) |
|
221 | let matcher = IncludeMatcher::new(ignore_patterns) | |
220 | .map_err(|e| handle_fallback(py, e.into()))?; |
|
222 | .map_err(|e| handle_fallback(py, e.into()))?; | |
221 |
|
223 | |||
222 | let (status_res, warnings) = dmap |
|
224 | dmap.with_status( | |
223 | .status( |
|
|||
224 |
|
|
225 | &matcher, | |
225 |
|
|
226 | root_dir.to_path_buf(), | |
226 |
|
|
227 | ignore_files, | |
@@ -232,10 +233,8 b' pub fn status_wrapper(' | |||||
232 |
|
|
233 | list_copies, | |
233 |
|
|
234 | collect_traversed_dirs, | |
234 |
|
|
235 | }, | |
|
236 | after_status, | |||
235 |
|
|
237 | ) | |
236 | .map_err(|e| handle_fallback(py, e))?; |
|
|||
237 |
|
||||
238 | build_response(py, status_res, warnings) |
|
|||
239 | } |
|
238 | } | |
240 | e => Err(PyErr::new::<ValueError, _>( |
|
239 | e => Err(PyErr::new::<ValueError, _>( | |
241 | py, |
|
240 | py, |
@@ -25,6 +25,9 b' use hg::utils::files::get_bytes_from_os_' | |||||
25 | use hg::utils::files::get_bytes_from_path; |
|
25 | use hg::utils::files::get_bytes_from_path; | |
26 | use hg::utils::files::get_path_from_bytes; |
|
26 | use hg::utils::files::get_path_from_bytes; | |
27 | use hg::utils::hg_path::{hg_path_to_path_buf, HgPath}; |
|
27 | use hg::utils::hg_path::{hg_path_to_path_buf, HgPath}; | |
|
28 | use hg::DirstateStatus; | |||
|
29 | use hg::PatternFileWarning; | |||
|
30 | use hg::StatusError; | |||
28 | use hg::StatusOptions; |
|
31 | use hg::StatusOptions; | |
29 | use log::info; |
|
32 | use log::info; | |
30 | use std::io; |
|
33 | use std::io; | |
@@ -230,12 +233,12 b' pub fn run(invocation: &crate::CliInvoca' | |||||
230 | list_copies, |
|
233 | list_copies, | |
231 | collect_traversed_dirs: false, |
|
234 | collect_traversed_dirs: false, | |
232 | }; |
|
235 | }; | |
233 | let (mut ds_status, pattern_warnings) = dmap.status( |
|
236 | ||
234 | &AlwaysMatcher, |
|
237 | type StatusResult<'a> = | |
235 | repo.working_directory_path().to_owned(), |
|
238 | Result<(DirstateStatus<'a>, Vec<PatternFileWarning>), StatusError>; | |
236 | ignore_files(repo, config), |
|
239 | ||
237 | options, |
|
240 | let after_status = |res: StatusResult| -> Result<_, CommandError> { | |
238 | )?; |
|
241 | let (mut ds_status, pattern_warnings) = res?; | |
239 | for warning in pattern_warnings { |
|
242 | for warning in pattern_warnings { | |
240 | match warning { |
|
243 | match warning { | |
241 | hg::PatternFileWarning::InvalidSyntax(path, syntax) => ui |
|
244 | hg::PatternFileWarning::InvalidSyntax(path, syntax) => ui | |
@@ -338,10 +341,25 b' pub fn run(invocation: &crate::CliInvoca' | |||||
338 | output.display(b"C ", "status.clean", ds_status.clean)?; |
|
341 | output.display(b"C ", "status.clean", ds_status.clean)?; | |
339 | } |
|
342 | } | |
340 |
|
343 | |||
341 |
let |
|
344 | let dirstate_write_needed = ds_status.dirty; | |
342 | let filesystem_time_at_status_start = |
|
345 | let filesystem_time_at_status_start = | |
343 | ds_status.filesystem_time_at_status_start; |
|
346 | ds_status.filesystem_time_at_status_start; | |
344 |
|
347 | |||
|
348 | Ok(( | |||
|
349 | fixup, | |||
|
350 | dirstate_write_needed, | |||
|
351 | filesystem_time_at_status_start, | |||
|
352 | )) | |||
|
353 | }; | |||
|
354 | let (fixup, mut dirstate_write_needed, filesystem_time_at_status_start) = | |||
|
355 | dmap.with_status( | |||
|
356 | &AlwaysMatcher, | |||
|
357 | repo.working_directory_path().to_owned(), | |||
|
358 | ignore_files(repo, config), | |||
|
359 | options, | |||
|
360 | after_status, | |||
|
361 | )?; | |||
|
362 | ||||
345 | if (fixup.is_empty() || filesystem_time_at_status_start.is_none()) |
|
363 | if (fixup.is_empty() || filesystem_time_at_status_start.is_none()) | |
346 | && !dirstate_write_needed |
|
364 | && !dirstate_write_needed | |
347 | { |
|
365 | { |
General Comments 0
You need to be logged in to leave comments.
Login now